超星考试助手【MCD】

自动挂机看尔雅MOOC,支持视频、音频、文档、图书自动完成,章节测验自动答题提交,支持自动切换任务点、挂机阅读时长、自动登录(不可用)等,解除各类功能限制,开放自定义参数。集成各个作者大大的接口,便捷切换

目前为 2020-11-26 提交的版本。查看 最新版本

  1. // ==UserScript==
  2. // @name 超星考试助手【MCD】
  3. // @namespace wyn665817@163.com
  4. // @version 4.0.4.6
  5. // @description 自动挂机看尔雅MOOC,支持视频、音频、文档、图书自动完成,章节测验自动答题提交,支持自动切换任务点、挂机阅读时长、自动登录(不可用)等,解除各类功能限制,开放自定义参数。集成各个作者大大的接口,便捷切换
  6. // @author wyn665817 & MCDream
  7. // @match *://*.chaoxing.com/exam/test/reVersionTestStartNew*
  8. // @match *://*.edu.cn/exam/test/reVersionTestStartNew*
  9. // @connect api.902000.xyz
  10. // @connect api.xmlm8.com
  11. // @connect 47.112.247.80
  12. // @connect 106.52.197.16
  13. // @run-at document-end
  14. // @grant unsafeWindow
  15. // @grant GM_xmlhttpRequest
  16. // @grant GM_setClipboard
  17. // @license MIT
  18. // ==/UserScript==
  19.  
  20. // 设置修改后,需要刷新或重新打开网课页面才会生效
  21. var queryapi = [
  22. //接口可以定义,参数可以自行理解
  23. {
  24. "url": "http://106.52.197.16:8080/chaoxing_war/topicServlet",
  25. "getIssueParam": "action=query&q=",
  26. "keyParam": "data",
  27. "method": "get"
  28. },
  29. {
  30. "url": "http://imnu.52king.cn/api/wk/index.php",
  31. "getIssueParam": "c=",
  32. "keyParam": "answer",
  33. "method": "get"
  34. },
  35. {
  36. "url": "http://api.902000.xyz:88/wkapi.php",
  37. "postIssueParam": "q",
  38. "keyParam": "answer",
  39. "method": "post"
  40. },
  41.  
  42. ];
  43.  
  44. var setting = {
  45. api: 2 //默认接口序号(0开始)由于李大佬经常抽风,换一个。。。
  46. // 8E3 == 8000,科学记数法,表示毫秒数
  47. ,
  48. time: 8E3 // 默认响应速度为8秒,不建议小于5秒
  49. // 1代表开启,0代表关闭
  50. ,
  51. none: 0 // 未找到答案或无匹配答案时执行默认操作,默认关闭
  52. ,
  53. jump: 1 // 答题完成后自动切换,默认关闭
  54. ,
  55. copy: 0 // 自动复制答案到剪贴板,也可以通过手动点击按钮或答案进行复制,默认关闭
  56.  
  57. // 非自动化操作
  58. ,
  59. hide: 0 // 不加载答案搜索提示框,键盘↑和↓可以临时移除和加载,默认关闭
  60. ,
  61. scale: 0 // 富文本编辑器高度自动拉伸,用于文本类题目,答题框根据内容自动调整大小,默认关闭
  62. },
  63. _self = unsafeWindow,
  64. $ = _self.jQuery,
  65. UE = _self.UE;
  66.  
  67. String.prototype.toCDB = function () {
  68. return this.replace(/\s/g, '').replace(/[\uff01-\uff5e]/g, function (str) {
  69. return String.fromCharCode(str.charCodeAt(0) - 65248);
  70. }).replace(/[“”]/g, '"').replace(/[‘’]/g, "'").replace(/。/g, '.');
  71. };
  72.  
  73. // setting.time += Math.ceil(setting.time * Math.random()) - setting.time / 2;
  74. setting.TiMu = [
  75. filterImg('.Cy_TItle .clearfix').replace(/\s*(\d+\.\d+分)$/, ''),
  76. $('[name^=type]:not([id])').val() || '-1',
  77. $('.cur a').text().trim() || '无',
  78. $('li .clearfix').map(function () {
  79. return filterImg(this);
  80. })
  81. ];
  82.  
  83. setting.div = $(
  84. '<div style="border: 2px dashed rgba(255,12,0,0.7); width: 330px; position: fixed; top: 0; right: 0; z-index: 99999; background-color: rgba(255,250,110,0.6); overflow-x: auto;">' +
  85. '<span style="font-size: medium;"></span>' +
  86. '<div style="font-size: medium;">正在搜索答案...</div>' +
  87. '<button style="margin-right: 10px;">暂停答题</button>' +
  88. '<button style="margin-right: 10px;' + (setting.jump ? '' : ' display: none;') + '">点击停止本次切换</button>' +
  89. '<button style="margin-right: 10px;">重新查询</button>' +
  90. '<button style="margin-right: 10px; display: none;">复制答案</button>' +
  91. '<button style="margin-right: 10px;">切换接口</button>' +
  92. '<button >答题详情</button>' +
  93. '<div style="max-height: 200px; overflow-y: auto;">' +
  94. '<table border="1" style="font-size: 12px;">' +
  95. '<thead>' +
  96. '<tr>' +
  97. '<th colspan="2">' + ($('#randomOptions').val() == 'false' ? '' : '<font color="red">本次考试的选项为乱序 脚本会选择正确的选项</font>') + '</th>' +
  98. '</tr>' +
  99. '<tr>' +
  100. '<th style="width: 60%; min-width: 130px;">题目(点击可复制)</th>' +
  101. '<th style="min-width: 130px;">答案(点击可复制)</th>' +
  102. '</tr>' +
  103. '</thead>' +
  104. '<tfoot style="' + (setting.jump ? ' display: none;' : '') + '">' +
  105. '<tr>' +
  106. '<th colspan="2">已关闭 本次自动切换</th>' +
  107. '</tr>' +
  108. '</tfoot>' +
  109. '<tbody>' +
  110. '<tr>' +
  111. '<td colspan="2" style="display: none;"></td>' +
  112. '</tr>' +
  113. '</tbody>' +
  114. '</table>' +
  115. '</div>' +
  116. '</div>'
  117. ).appendTo('body').on('click', 'button, td', function () {
  118. var num = setting.$btn.index(this);
  119. if (num == -1) {
  120. GM_setClipboard($(this).text());
  121. }
  122. else if (num === 0) {
  123. if (setting.loop) {
  124. clearInterval(setting.loop);
  125. delete setting.loop;
  126. num = ['已暂停搜索', '继续答题'];
  127. }
  128. else {
  129. setting.loop = setInterval(findTiMu, setting.time);
  130. num = ['正在搜索答案...', '暂停答题'];
  131. }
  132. setting.$div.html(function () {
  133. return $(this).data('html') || num[0];
  134. }).removeData('html');
  135. $(this).html(num[1]);
  136. }
  137. else if (num == 1) {
  138. setting.jump = 0;
  139. setting.$div.html(function () {
  140. return arguments[1].replace('即将切换下一题', '未开启自动切换');
  141. });
  142. setting.div.find('tfoot').add(this).toggle();
  143. }
  144. else if (num == 2) {
  145. location.reload();
  146. }
  147. else if (num == 3) {
  148. GM_setClipboard(setting.div.find('td:last').text());
  149. }
  150. else if (num == 4) {
  151. setting.api < queryapi.length - 1 ? setting.api += 1 : setting.api = 0;
  152. $(this).html('切换接口,当前:' + setting.api);
  153. //重新请求
  154. clearInterval(setting.loop);
  155. delete setting.loop;
  156. setting.loop = setInterval(findTiMu, setting.time);
  157. //num = ['正在搜索答案...', '暂停答题'];
  158. }
  159. else if (num == 5) {
  160. ($('.leftCard .saveYl')[0] || $()).click();
  161. }
  162. }).detach(setting.hide ? '*' : 'html');
  163. setting.$btn = setting.div.children('button');
  164. setting.$div = setting.div.children('div:eq(0)');
  165.  
  166. $(document).keydown(function (event) {
  167. if (event.keyCode == 38) {
  168. setting.div.detach();
  169. }
  170. else if (event.keyCode == 40) {
  171. setting.div.appendTo('body');
  172. }
  173. });
  174.  
  175. if (setting.scale) _self.UEDITOR_CONFIG.scaleEnabled = false;
  176. $.each(UE.instants, function () {
  177. var key = this.key;
  178. this.ready(function () {
  179. this.destroy();
  180. UE.getEditor(key);
  181. });
  182. });
  183. setting.loop = setInterval(findTiMu, setting.time);
  184.  
  185. function findTiMu() {
  186. GM_xmlhttpRequest({
  187. method: queryapi[setting.api].method,
  188. url: queryapi[setting.api].url + "?" + queryapi[setting.api].getIssueParam + encodeURIComponent(setting.TiMu[0]),
  189. headers: {
  190. 'Content-type': 'application/x-www-form-urlencoded'
  191. },
  192. data: queryapi[setting.api].postIssueParam + '=' + encodeURIComponent(setting.TiMu[0]) + '&type=' + setting.TiMu[1],
  193. timeout: setting.time,
  194. onload: function (xhr) {
  195. if (!setting.loop) {}
  196. else if (xhr.status == 200) {
  197. var obj = $.parseJSON(xhr.responseText) || {};
  198. if (obj.answer) {}
  199. else obj.answer = obj.data || obj.da;
  200. obj.code = obj.code == null ? 1 : obj.code;
  201. if (obj.code == 1 || obj.tm) {
  202. var data = String(obj.answer).replace(/&/g, '&amp;').replace(/<(?!img)/g, '&lt;'),
  203. que = setting.TiMu[0].match('<img') ? setting.TiMu[0] : setting.TiMu[0].replace(/&/g, '&amp;').replace(/</g, '&lt');
  204. obj.answer = /^http/.test(data) ? '<img src="' + obj.answer + '">' : obj.answer;
  205. setting.div.find('tbody').append(
  206. '<tr>' +
  207. '<td title="点击可复制">' + que + '</td>' +
  208. '<td title="点击可复制">' + (/^http/.test(data) ? obj.answer : '') + data + '</td>' +
  209. '</tr>'
  210. );
  211. setting.copy && GM_setClipboard(obj.answer);
  212. setting.$btn.eq(3).show();
  213. fillAnswer(obj);
  214. }
  215. else {
  216. setting.$div.html(obj.data || '服务器繁忙,正在重试...');
  217. }
  218. setting.div.children('span').html(obj.msg || '');
  219. }
  220. else if (xhr.status == 403) {
  221. var html = xhr.responseText.indexOf('{') ? '请求过于频繁,建议稍后再试' : $.parseJSON(xhr.responseText).data;
  222. setting.$div.data('html', html).siblings('button:eq(0)').click();
  223. }
  224. else {
  225. setting.$div.text('服务器异常,正在重试...');
  226. }
  227. },
  228. ontimeout: function () {
  229. setting.loop && setting.$div.text('服务器超时,正在重试...');
  230. }
  231. });
  232. }
  233.  
  234. function fillAnswer(obj, tip) {
  235. var $input = $(':radio, :checkbox', '.Cy_ulBottom'),
  236. str = String(obj.answer).toCDB() || new Date().toString(),
  237. data = str.split(/#|\x01|\|/),
  238. opt = obj.opt || str,
  239. btn = $('.saveYl:contains(下一题)').offset();
  240. // $input.filter(':radio:checked').prop('checked', false);
  241. obj.code > 0 && $input.each(function (index) {
  242. if (this.value == 'true') {
  243. data.join().match(/(^|,)(正确|是|对|√|T|ri)(,|$)/) && this.click();
  244. }
  245. else if (this.value == 'false') {
  246. data.join().match(/(^|,)(错误|否|错|×|F|wr)(,|$)/) && this.click();
  247. }
  248. else {
  249. index = setting.TiMu[3][index].toCDB() || new Date().toString();
  250. index = $.inArray(index, data) + 1 || (setting.TiMu[1] == '1' && str.indexOf(index) + 1);
  251. Boolean(index) == this.checked || this.click();
  252. }
  253. }).each(function () {
  254. if (!/^A?B?C?D?E?F?G?$/.test(opt)) return false;
  255. Boolean(opt.match(this.value)) == this.checked || this.click();
  256. });
  257. if (setting.TiMu[1].match(/^[013]$/)) {
  258. tip = $input.is(':checked') || setting.none && (($input[Math.floor(Math.random() * $input.length)] || $()).click(), ' ');
  259. }
  260. else if (setting.TiMu[1].match(/^(2|[4-9]|1[08])$/)) {
  261. data = String(obj.data).split(/#|\x01|\|/);
  262. tip = $('.Cy_ulTk textarea').each(function (index) {
  263. index = (obj.code > 0 && data[index]) || '';
  264. UE.getEditor(this.name).setContent(index.trim());
  265. }).length;
  266. tip = (obj.code > 0 && data.length == tip) || setting.none && ' ';
  267. setting.len = str.length * setting.time / 10;
  268. }
  269. if (tip == ' ') {
  270. tip = '已执行默认操作';
  271. }
  272. else if (tip) {
  273. tip = '自动答题已完成';
  274. }
  275. else if (tip === undefined) {
  276. tip = '该题型不支持自动答题';
  277. }
  278. else {
  279. tip = '未找到有效答案';
  280. }
  281. if (btn) {
  282. tip += setting.jump ? ',即将切换下一题' : ',未开启自动切换';
  283. setInterval(function () {
  284. if (!setting.jump) return;
  285. var mouse = document.createEvent('MouseEvents'),
  286. arr = [btn.left + Math.ceil(Math.random() * 80), btn.top + Math.ceil(Math.random() * 26)];
  287. mouse.initMouseEvent('click', true, true, document.defaultView, 0, 0, 0, arr[0], arr[1], false, false, false, false, 0, null);
  288. _self.event = $.extend(true, {}, mouse);
  289. delete _self.event.isTrusted;
  290. _self.getTheNextQuestion(1);
  291. }, setting.len || Math.ceil(setting.time * Math.random()) * 2);
  292. }
  293. else {
  294. setting.$btn.eq(1).hide();
  295. tip = '答题已完成,请自行查看答题详情';
  296. }
  297. setting.$div.data('html', tip).siblings('button:eq(0)').hide().click();
  298. }
  299.  
  300. function filterImg(dom) {
  301. return $(dom).clone().find('img[src]').replaceWith(function () {
  302. return $('<p></p>').text('<img src="' + $(this).attr('src') + '">');
  303. }).end().find('iframe[src]').replaceWith(function () {
  304. return $('<p></p>').text('<iframe src="' + $(this).attr('src') + '"></irame>');
  305. }).end().text().trim();
  306. }

QingJ © 2025

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