济南专业技术人员继续教育—公需课

公需科目辅助|自动答题|自动切换下一小节

Instalar o script?
Script sugerido do autor

Você também pode gostar de 暴力猴控制台美化.

Instalar o script
  1. // ==UserScript==
  2. // @name 济南专业技术人员继续教育—公需课
  3. // @namespace Violentmonkey Scripts
  4. // @match *://*.ghlearning.com/*
  5. // @match http://221.214.69.254:9091/*
  6. // @grant none
  7. // @version 0.1.10
  8. // @author aliha
  9. // @description 公需科目辅助|自动答题|自动切换下一小节
  10. // @license GPL-3.0
  11. // @run-at document-end
  12. // @icon 
  13. // ==/UserScript==
  14. (function () {
  15.  
  16. // 定义一些设置变量
  17. const NEXT_SUBMIT_DEALY = 6; // 尝试错误后下一次提交间隔秒,目前测试6秒刚好,不然会在弹窗出现之前往下执行,跳过一些答案选项组合
  18. const SUBMIT_DELAY = 1; // 填写完成答案后,提交间隔
  19. const FIRST_RUN_DELAY = 15; // 首次运行延迟15秒,防止在player没加载前运行出错
  20. const RUN_INTERVAL = 180; // 脚本运行间隔,默认180秒
  21. const PLAY_TYPE = 1; // 播放方式,1-方式一;2-方式二
  22.  
  23.  
  24. // 延迟s秒
  25. function delay(s) {
  26. return new Promise(resolve => setTimeout(resolve, s * 1000));
  27. }
  28.  
  29. // 检测答题元素,获取选项
  30. function getItems() {
  31. if (document.querySelector(".pv-ask-modal")) {
  32. let qusCard = document.querySelector(".pv-ask-modal")
  33. let inputs = qusCard.querySelectorAll("input")
  34. return inputs
  35. }
  36.  
  37. return null
  38. }
  39.  
  40. // 生成数组所有的穷举组合并剔除空数组
  41. function generateCombinations(arr) {
  42. const combinations = [
  43. []
  44. ];
  45.  
  46. // 遍历数组元素
  47. for (let i = 0; i < arr.length; i++) {
  48. const currentLength = combinations.length;
  49.  
  50. // 遍历当前已生成的组合
  51. for (let j = 0; j < currentLength; j++) {
  52. const currentCombination = combinations[j];
  53.  
  54. // 生成新的组合,包含当前数组元素
  55. const newCombination = currentCombination.concat(arr[i]);
  56.  
  57. // 如果组合不为空,则将新组合添加到二维数组中
  58. if (newCombination.length > 0) {
  59. combinations.push(newCombination);
  60. }
  61. }
  62. }
  63.  
  64. // 剔除空数组
  65. return combinations.filter(combination => combination.length > 1);
  66. }
  67.  
  68. // 判断是否暂停
  69. function isPuased() {
  70. var elements = document.getElementsByClassName("pv-icon-btn-play");
  71. for (var i = 0; i < elements.length; i++) {
  72. var element = elements[i];
  73. var computedStyle = window.getComputedStyle(element);
  74. if (computedStyle.getPropertyValue("display") === "block") {
  75. // console.log(element);
  76. return true;
  77. } else {
  78. return false;
  79. }
  80. }
  81. }
  82.  
  83. // 获取进度,自动播放
  84. function getProcess() {
  85. let jindu = document.querySelector("#a span[du-html=sumschedule]"); // 获取总进度
  86. if (jindu) {
  87.  
  88. if (jindu.innerText === "100.00") {
  89. console.info("本课程已完成");
  90. return 0;
  91. } else {
  92. // 获取当前进度,如果播放完毕或者未开始,点击播放按钮开始播放,(因为系统播放完一节会自动切换下一节,所以光点击播放按钮就可以了,不用手动切换小节)
  93. let dangqian = document.querySelector(".videoLi.active");
  94. let ispaused = isPuased();
  95. if (dangqian.innerText.match(/[0-9]+%/)[0] == "100%" || dangqian.innerText.match(/[0-9]+%/)[0] == "0%" || ispaused) {
  96. console.info("本课程未完成,继续播放");
  97.  
  98. if (PLAY_TYPE === 1) {
  99. console.info("使用方式一自动播放")
  100. const pauseBtn = document.querySelector('button[type="button"].pv-playpause.pv-iconfont.pv-icon-btn-play');
  101. if (pauseBtn) {
  102. pauseBtn.click();
  103. }
  104.  
  105. } else if (PLAY_TYPE === 2) {
  106. console.info("使用方式二自动播放")
  107. const video = document.getElementById('video');
  108. video.muted = true;
  109. video.play()
  110. .then(() => {
  111. console.info("自动播放成功");
  112. })
  113. .catch(error => {
  114. console.error('自动播放失败,请手动点击播放:', error);
  115. });
  116. }
  117. }
  118. }
  119. }
  120. }
  121.  
  122. // 挨个尝试,检测到回答错误继续,检测到回答正确跳出,同时清空答案
  123. async function answer(res_ls) {
  124.  
  125. for (let i = res_ls.length - 1; i != 0; i--) {
  126. await delay(NEXT_SUBMIT_DEALY)
  127.  
  128. let inputs = getItems()
  129. if (inputs) {
  130. console.log(`尝试第${res_ls.length - i}次作答`)
  131. // console.log(...res_ls[i])
  132. for (let j = 0; j < res_ls[i].length; j++) {
  133. inputs[res_ls[i][j]].checked = true;
  134. }
  135.  
  136. // 提交答案
  137. let button = document.querySelector('button.pv-ask-submit[data-type="pvSubmit"]');
  138. await delay(SUBMIT_DELAY)
  139. if (button) {
  140. button.click()
  141. }
  142. }else{
  143. console.log("回答正确");
  144. break;
  145. }
  146. }
  147. }
  148.  
  149. // 主函数
  150. async function main() {
  151. console.info("开始答题")
  152.  
  153. await getProcess()
  154.  
  155. // 自动答题
  156. if (getItems()) {
  157. let inputs = getItems()
  158. const array = Array.from({
  159. length: inputs.length
  160. }, (_, index) => index);
  161. let num = generateCombinations(array)
  162. await answer(num)
  163. console.info("答题脚本执行完毕")
  164. } else {
  165. console.info(`未检测到答题卡,${RUN_INTERVAL}秒后再次运行`)
  166. }
  167. }
  168.  
  169.  
  170. setTimeout(main, FIRST_RUN_DELAY*1000);
  171. setInterval(main, RUN_INTERVAL*1000);
  172. })();