普法网(宪法小卫士)课后练习、考试自动答题

全国学生“学宪法 讲宪法”活动自动答题脚本,因缺少测试账号无法保证每年都能用,欢迎大家提供测试账号以支持此脚本长期可用,测试账号登录(不可用)信息请发送至我们的邮箱nawlgzs@gmail.com

  1. // ==UserScript==
  2. // @name 普法网(宪法小卫士)课后练习、考试自动答题
  3. // @namespace Ne-21
  4. // @version 1.4.1
  5. // @description 全国学生“学宪法 讲宪法”活动自动答题脚本,因缺少测试账号无法保证每年都能用,欢迎大家提供测试账号以支持此脚本长期可用,测试账号登录(不可用)信息请发送至我们的邮箱nawlgzs@gmail.com
  6. // @author Ne-21
  7. // @match *://static.qspfw.moe.gov.cn/*
  8. // @icon 
  9. // @run-at document-end
  10. // @grant unsafeWindow
  11. // @grant GM_setValue
  12. // @grant GM_getValue
  13. // @license MIT
  14. // @require https://lf9-cdn-tos.bytecdntp.com/cdn/expire-1-M/jquery/3.6.0/jquery.min.js
  15. // @require https://lf26-cdn-tos.bytecdntp.com/cdn/expire-1-M/limonte-sweetalert2/11.0.1/sweetalert2.all.min.js
  16. // @require https://lf26-cdn-tos.bytecdntp.com/cdn/expire-1-M/crypto-js/4.1.1/crypto-js.min.js
  17. // ==/UserScript==
  18.  
  19. var _self = unsafeWindow,
  20. $ = _self.jQuery || top.jQuery,
  21. Swal = Swal || window.Swal,
  22. columnId = getQueryVariable("columnId"),
  23. answer_list = [],
  24. exam_list = [],
  25. time = 3e3, // 答题间隔时间,最好为5秒
  26. save_key = "xfxws2024",
  27. num = {"A": 1,"B": 2, "C": 3, "D": 4};
  28.  
  29. (function() {
  30. // 题目切换其实可以不依靠定时器,不过懒得改了,能用就行。。。
  31. if (window.location.pathname.indexOf('learn_exam.html') != -1) {
  32. Swal.fire('宪法小助手提示','点击确定开始考试','info').then(()=>{
  33. Swal.fire({
  34. position: 'top-end',
  35. title: '脚本将在3秒后开始自动作答!',
  36. showConfirmButton: false,
  37. timer: 3000
  38. })
  39. getExam();
  40. let t = setInterval( function() {
  41. doExam(t)
  42. },time);
  43. })
  44. } else if (window.location.pathname.indexOf('learn-practice.html') != -1) {
  45. Swal.fire('宪法小助手提示','点击确定开始练习,脚本会自动收录本练习题目数据','info').then(()=>{
  46. getAnswer(columnId);
  47. let t = setInterval( function() {
  48. doQuestion(t)
  49. },time);
  50. })
  51. } else if (window.location.pathname.indexOf('learn_practice_list.html') != -1) {
  52. Swal.fire('宪法小助手提示','<div style="font-size: 13px;">脚本最后更新时间:2024.10.26<br />使用说明:<br /><div><span style="color: red;">1.脚本题库数据托管于本地,请在做综合评价前逐个完成练习,收集答案!!!</span><br />2.脚本运行故障如综合测评无操作等,请使用Edge浏览器+ScriptCat。<br />3.问题联系邮箱nawlgzs@gmail.com<br />4.脚本数据来自本网站后端返回的明文JSON数据包,脚本不涉及任何逆向操作!<br />5.脚本仅供学习交流,请勿用于商业用途,否则后果自负!</div></div>')
  53. } else if (window.location.pathname.indexOf('evaluation.html') != -1) {
  54. } else {
  55. console.log(window.location.pathname)
  56. }
  57. })();
  58.  
  59. // 解析url参数
  60. function getQueryVariable(variable) {
  61. var query = window.location.search.substring(1);
  62. var vars = query.split("&");
  63. for (var i=0;i<vars.length;i++) {
  64. var pair = vars[i].split("=");
  65. if(pair[0] == variable){return pair[1];}
  66. }
  67. return(false);
  68. };
  69.  
  70. // 正则匹配
  71. function getStr(str, start, end) {
  72. let res = str.match(new RegExp(`${start}(.*?)${end}`))
  73. return res ? res[1] : null
  74. }
  75.  
  76. // 获取答案
  77. function getAnswer(columnId) {
  78. $.ajax({
  79. url: _self.config.practice.host + _self.config.practice.practice + "?columnId="+ columnId + "&taskId=" + _self.config.taskId,
  80. headers: _self.config.apiConfig.header,
  81. async: false,
  82. success: function (res) {
  83. const { data, status } = res;
  84. if (status === "0") {
  85. var questionBankList = data.questionBankList
  86. answer_list = questionBankList;
  87. upload(answer_list)
  88. } else if (status === "1") {
  89. alert("请先学习当前模块");
  90. window.history.go(-1);
  91. } else if (status === "-2") {
  92. alert("请重新登陆");
  93. } else {
  94.  
  95. }
  96. },
  97. error: function (err) {
  98. }
  99. });
  100. }
  101.  
  102. // 答题操作
  103. function doQuestion(t) {
  104. var cur_topic = $('#currentTopic').text(),
  105. tol_topic = $('#totalTopic').text(),
  106. answer = answer_list[cur_topic - 1].answer;
  107. $('#exam_answer > div:nth-child(' + num[answer] + ')').click();
  108. if (cur_topic == tol_topic) {
  109. clearInterval(t);
  110. setTimeout(function(){Swal.fire('宪法小助手提示','答题完成','info')},time / 2);
  111. } else{
  112. setTimeout(function(){$('#next_question').click()},time / 2);
  113. };
  114. }
  115.  
  116. // 获取考试题目
  117. function getExam(){
  118. $.ajax({
  119. url: _self.config.wexam.host + _self.config.wexam.getPaper + "?taskId=" + _self.config.taskId,
  120. headers: _self.config.apiConfig.header,
  121. async: false,
  122. success: function (res) {
  123. const { data, status, message } = res;
  124. if (status === "0") {
  125. var question_data = res.data;
  126. var paper = question_data.paper;
  127. var paperInfo = paper.paperInfo;
  128. exam_list = paperInfo;
  129. } else {
  130. alert('获取考试题目失败!')
  131. }
  132. },
  133. error: function (err) {
  134. }
  135. });
  136. }
  137.  
  138. // 考试答题操作
  139. function doExam(t){
  140. let db_json = []
  141. if (GM_getValue(save_key) && JSON.parse(GM_getValue(save_key)).length >= 0) {
  142. db_json = JSON.parse(GM_getValue(save_key))
  143. } else {
  144. $('#ne21ans')[0] ? $('#ne21ans').html('<p style="color: red;">未匹配到答案,请手动作答~</p>') : $('#exam_question').append('<div id="ne21ans"><p style="color: red;">未匹配到答案,请手动作答~</p></div>')
  145. return
  146. }
  147.  
  148. console.log(JSON.parse(GM_getValue(save_key)))
  149. $('#ne21ans')[0] ? $('#ne21ans').html('<p style="color: red;">正在搜索答案~</p>') : $('#exam_question').append('<div id="ne21ans"><p style="color: red;">正在搜索答案~</p></div>')
  150.  
  151. var cur_topic = $('#currentTopic').text(),
  152. tol_topic = $('#totalTopic').text(),
  153. questionInfo = exam_list[cur_topic - 1];
  154.  
  155.  
  156. ans_index = []
  157. let question = filterStr(questionInfo.content)
  158. let ops = questionInfo.answerOptions.split("@!@")
  159. ops.map((el)=> filterStr(el) )
  160. for (var i = 0; i < ops.length; i++) {
  161. hash_tmp = MD555(question+"|"+ops[i])
  162. db_json.forEach((item)=>{
  163. if (item.hash == hash_tmp) {
  164. ans_index.push(i)
  165. }
  166. })
  167. }
  168.  
  169. if (ans_index.length == 0) {
  170. $('#ne21ans')[0] ? $('#ne21ans').html('<p style="color: red;">无题库数据,请先收集答案或自己作答~</p>') : $('#exam_question').append('<div id="ne21ans"><p style="color: red;">无题库数据,请先收集答案或自己作答~</p></div>')
  171. return
  172. }
  173.  
  174. ans_index.forEach((item1)=>{
  175. $('#ne21ans').html('<p style="color: red;">参考答案:'+ ops[item1] + '</p>')
  176. $('#exam_answer > div:nth-child(' + (item1+1) + ')').click();
  177. })
  178.  
  179. if (cur_topic == tol_topic) {
  180. clearInterval(t);
  181. setTimeout(function(){Swal.fire('宪法小助手提示','答题完成,请自己点击交卷!','info')},time / 2);
  182. } else{
  183. setTimeout(function(){$('#next_question').click()},time / 2);
  184. };
  185.  
  186. }
  187.  
  188. function upload(question_data) {
  189. let db_json = []
  190. if (GM_getValue(save_key) && JSON.parse(GM_getValue(save_key)).length >= 0) {
  191. db_json = JSON.parse(GM_getValue(save_key))
  192. }
  193. question_data.forEach((item)=>{
  194. let question = filterStr(item.content)
  195. let ans_index = []
  196. item.answer.split().forEach(((item1)=>{
  197. let index_tmp = "ABCDEFG".indexOf(item1)
  198. ans_index.push(index_tmp)
  199. }))
  200. let ans_ops = item.answerOptions.split("@!@")
  201. ans_ops.map((el)=> filterStr(el) )
  202. ans_index.forEach((item3)=>{
  203. db_json.push({
  204. "hash":MD555(question+"|"+ans_ops[item3]),
  205. "question": question,
  206. "answer": ans_ops[item3]
  207. })
  208. })
  209. })
  210.  
  211. let dbJson = uniqueByField(db_json,"hash")
  212. GM_setValue(save_key,JSON.stringify(dbJson))
  213. // console.log(JSON.parse(GM_getValue(save_key)))
  214. }
  215.  
  216. function MD555(str) {
  217. return CryptoJS.MD5(str).toString()
  218. }
  219.  
  220. // 重复数据过滤
  221. function uniqueByField(array, field) {
  222. const seen = new Set();
  223. return array.filter((item) => {
  224. const key = item[field];
  225. return seen.has(key) ? false : seen.add(key);
  226. });
  227. }
  228.  
  229. // 去除字符串中的空格、换行、制表符
  230. function filterStr(str){
  231. return str.replace(/[\r\n\s<br>]/g, "").trim()
  232. }

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址