AnswerTip

http://tampermonkey.net/

  1. // ==UserScript==
  2. // @name AnswerTip
  3. // @description http://tampermonkey.net/
  4. // @version 0.32
  5. // @author polygon
  6. // @icon 
  7. // @match https://changjiang.yuketang.cn/*/exercise/*
  8. // @match https://changjiang-exam.yuketang.cn/exam/*
  9. // @match https://changjiang.yuketang.cn/*
  10. // @match https://www.xuetangx.com/*/exercise/*
  11. // @match https://mooc1.chaoxing.com/work/doHomeWorkNew*
  12. // @match https://examination.xuetangx.com/exam/*
  13. // @match https://onlineexamh5new.zhihuishu.com/*
  14. // @grant GM_xmlhttpRequest
  15. // @grant GM_addStyle
  16. // @grant GM_getValue
  17. // @grant GM_setValue
  18. // @grant GM_addElement
  19. // @run-at document-end
  20. // @namespace http://tampermonkey.net/
  21. // ==/UserScript==
  22. (function() {
  23. const order = ['A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I']
  24. const n = 10
  25. const wait = 3 // s
  26. const configs = {
  27. 'onlineexamh5new.zhihuishu.com': {
  28. main: '.examPaper_subject', // 答案要展示的父级元素节点,会append到该Node下
  29. question: '.subject_describe', // 问题,题目所在位置,其innerText属性是问题(不含选项)
  30. options: '.subject_node p span, .node_detail p', // 每个选项的位置,会根据它匹配多个Node
  31. select: 'onChecked', // 点击后,相应属性增加值,因为有的选项需要多次点击,所以增加此用于判断是否点击上
  32. // 一些自定义添加的答案区域style
  33. style: `
  34. background-color: #f5f5f5;
  35. height: 200px;
  36. overflow-y: scroll;
  37. `
  38. },
  39. 'changjiang.yuketang.cn': {
  40. main: '.item-body', // 答案要展示的父级元素节点,会append到该Node下
  41. question: '.problem-body', // 问题,题目所在位置,其innerText属性是问题(不含选项)
  42. options: '.item-body ul li label', // 每个选项的位置,会根据它匹配多个Node
  43. select: 'is-checked', // 点击后,相应属性增加值,因为有的选项需要多次点击,所以增加此用于判断是否点击上
  44. // 一些自定义添加的答案区域style
  45. style: `
  46. background-color: #f5f5f5;
  47. `
  48. },
  49. 'changjiang-exam.yuketang.cn': {
  50. main: '.item-body',
  51. question: 'h4',
  52. options: '.item-body ul li label',
  53. select: 'is-checked',
  54. style: `
  55. background-color: #f5f5f5;
  56. height: 200px;
  57. overflow-y: scroll;
  58. `
  59. },
  60. 'www.xuetangx.com': {
  61. main: '.courseActionLesson',
  62. question: '.leftQuestion',
  63. options: '.leftradio',
  64. select: 'active',
  65. style: `
  66. background-color: #fcfcfc;
  67. border: 1px solid hsla(0,0%,81.2%,.31);
  68. width: 500px;
  69. height: 200px;
  70. overflow-y: scroll;
  71. `
  72. },
  73. 'mooc1.chaoxing.com': {
  74. main: '.TiMu',
  75. question: '.Zy_TItle div p',
  76. options: 'ul li',
  77. select: 'Hover',
  78. style: `
  79. background-color: #fcfcfc;
  80. height: 200px;
  81. overflow-y: scroll;
  82. `
  83. },
  84. 'examination.xuetangx.com': {
  85. main: '.subject-item',
  86. question: '.item-body',
  87. options: 'ul li',
  88. select: 'is-checked',
  89. style: `
  90. background-color: #fcfcfc;
  91. height: 200px;
  92. overflow-y: scroll;
  93. `
  94. }
  95. }
  96. let url = window.location.href
  97. let host = Object.keys(configs).filter((host) => {
  98. if (url.includes(host)) {
  99. return true
  100. }
  101. return false
  102. })
  103. if (host.length == 0) return
  104. const config = configs[host[0]]
  105. GM_addStyle(`
  106. .answer-polygon {
  107. ${config.style};
  108. font-size: 14px;
  109. border-radius: 4px;
  110. padding: 10px 15px;
  111. margin-bottom: 20px;
  112. text-align: left;
  113. font-family: -apple-system,SF UI Text,Arial,PingFang SC,Hiragino Sans GB,Microsoft YaHei,WenQuanYi Micro Hei,"sans-serif";
  114. }
  115. `)
  116. let cookieSet = () => {
  117. GM_xmlhttpRequest({
  118. method: 'POST',
  119. url: 'https://tiku.fenbi.com/android/tourist/enter',
  120. responseType: 'json',
  121. onload: (xhr) => {
  122. const data = xhr.response
  123. let cookie = `userid=${data.userId}; tourist=${data.touristToken};`
  124. console.log(cookie)
  125. }
  126. })
  127. }
  128. let clickOption = (optionNodes, text) => {
  129. optionNodes.forEach((optionNode) => {
  130. let option_text = optionNode.innerText.replaceAll('[p]', '').replaceAll('[/p]', '').split('\n').slice(-1)
  131. console.log(option_text, text)
  132. if (option_text.includes(text)||text.includes(option_text)) {
  133. let id = setInterval(() => {
  134. if (!optionNode.innerHTML.includes(config.select)) {
  135. optionNode.click()
  136. } else {
  137. clearInterval(id)
  138. }
  139. }, 3000)
  140. }
  141. })
  142. }
  143. let answerParser = (data, answerNode, optionNodes, mainNode) => {
  144. let s = ''
  145. let answer_groups = []
  146. for (let i=0;i<n;i++) {
  147. let item = data['questionList'][i]
  148. // 1 题目
  149. s += item['content'].replaceAll('[', '<').replaceAll(']', '>') + '<br/>'
  150. const re = new RegExp("[`~!@#$^&*()=|{}':;',\\[\\].<>/?~!@#¥……&*()——|{}【】‘;:”“'。,、?%+_]", 'g');
  151. let same = item['content'].replaceAll(re, '').includes(mainNode.getAttribute('currentQuestion').replaceAll(re, '').split('\n')[0])
  152. if (same) {
  153. answer_groups.push([])
  154. }
  155. // 2 正确答案
  156. // 2.1 选项形式 ABCD
  157. if (item['correctAnswer'].hasOwnProperty('choice')) {
  158. let choice_indexes = item['correctAnswer']['choice'].split(',')
  159. if (choice_indexes.length > 0) {
  160. let choice_orders = []
  161. for (let j=0;j<choice_indexes.length;j++) {
  162. let choice_order = order[parseInt(choice_indexes[j])]
  163. if (!choice_orders.includes(choice_order)) {
  164. choice_orders.push(choice_order)
  165. }
  166. }
  167. s += '<p>' + choice_orders.join('') + '</p><br/>'
  168. // choice_orders = ['A', 'B', 'C']
  169. // 选项
  170. if (item['accessories'].length > 0){
  171. item['accessories'] = item['accessories'].filter((item) => {
  172. return (item.hasOwnProperty('options'))
  173. })
  174. if (item['accessories'].length > 0 && item['accessories'][0]['options'].length > 0) {
  175. let options = item['accessories'][0]['options']
  176. for (let j=0;j<options.length;j++) {
  177. let option = options[j].replace('[p]', '').replace('[/p]', '')
  178. if (choice_orders.includes(order[j])) {
  179. if (same) {
  180. answer_groups[answer_groups.length-1].push(options[j])
  181. }
  182. // if (same) clickOption(optionNodes, options[j])
  183. s += `<b>[${order[j]}] ` + option + '</b><br/>'
  184. } else {
  185. s += `[${order[j]}] ` + option + '<br/>'
  186. }
  187.  
  188. }
  189.  
  190. }
  191. }
  192. }
  193. // 2.2 文字描述形式
  194. } else if (item['correctAnswer'].hasOwnProperty('answer')) {
  195. let text = item['correctAnswer']['answer']
  196. .replace('', '<br/>')
  197. .replace('\u0001', '<br/>')
  198. .replace('\x01', '<br/>')
  199. .replace(/-+/g, '<br/>')
  200. .replaceAll('[', '<')
  201. .replaceAll(']', '>')
  202. .replace(';;', ';')
  203. if (same) {
  204. text.replace('<p>', '')
  205. .replace('</p>', '')
  206. .split('<br/>')
  207. .forEach((option_text) => {
  208. answer_groups[answer_groups.length-1].push(option_text)
  209. // clickOption(optionNodes, option_text)
  210. })
  211. }
  212. s += '<b>' + text + '</b>'
  213. }
  214. if (i < n-1) {
  215. s += '<br/><hr/><br/>'
  216. }
  217. }
  218. // 判断是否为多选
  219. let multi = false
  220. answer_groups.forEach((group) => {
  221. if (group.length > 1) {
  222. multi = true
  223. }
  224. })
  225. if (multi) {
  226. answer_groups = answer_groups.filter((group) => {
  227. return group.length > 1
  228. })
  229. }
  230. let answers = []
  231. if (answer_groups.length == 1) {
  232. answers = answer_groups[0]
  233. } else {
  234. // 取交集
  235. // 得到所有答案元素
  236. let answer_items = []
  237. answer_groups.forEach((group) => {
  238. group.forEach((item) => {
  239. if (!answer_items.includes(item)) {
  240. answer_items.push(item)
  241. }
  242. })
  243. })
  244. answers = answer_items.filter((item) => {
  245. let b = true
  246. answer_groups.forEach((group) => {
  247. if (!group.includes(item)) {
  248. b = false
  249. }
  250. })
  251. return b
  252. })
  253. }
  254. console.log(answers)
  255. answers.forEach((answer) => {
  256. clickOption(optionNodes, answer)
  257. })
  258. answerNode.innerHTML = s
  259. }
  260. let searchAnswer = (text, answerNode, optionNodes, mainNode) => {
  261. GM_xmlhttpRequest({
  262. method: 'POST',
  263. url: 'https://schoolapi.fenbi.com/college/android/search-item/search?format=ubb&searchType=2&text=' + encodeURIComponent(text),
  264. responseType: 'json',
  265. onload: (xhr) => {
  266. let res = xhr.response
  267. let enc = res['data']
  268. if (enc == null) {
  269. if (res.errMessage.includes('次数')) {
  270. cookieSet()
  271. }
  272. answerNode.innerHTML = xhr.response.errMessage
  273. setTimeout(() => {
  274. mainNode.setAttribute('currentQuestion', '')
  275. }, wait * 1000);
  276. return
  277. }
  278. // 解密
  279. let t = new Date().valueOf()
  280. GM_xmlhttpRequest({
  281. method: 'POST',
  282. url: `http://101.35.131.22:5500/decrypt`,
  283. headers: {
  284. "Content-Type": "application/x-www-form-urlencoded"
  285. },
  286. data: `enc=${enc}`,
  287. responseType: 'text',
  288. onload: (xhr) => {
  289. let data = JSON.parse(xhr.responseText)
  290. answerParser(data, answerNode, optionNodes, mainNode)
  291. }
  292. })
  293. },
  294. ontimeout: function () {
  295. answer.innerHTML = "No Sugesstion"
  296. }
  297. })
  298. }
  299. cookieSet()
  300. let id = setInterval(() => {
  301. let mainNodes = document.querySelectorAll(config.main)
  302. if (mainNodes.length == 0) return
  303. clearInterval(id)
  304. mainNodes.forEach(function(mainNode) {
  305. mainNode.setAttribute('currentQuestion', '')
  306. setInterval(() => {
  307. let questionNode = mainNode.querySelector(config.question)
  308. if (questionNode == null) return
  309. let text = questionNode.innerText.slice(0, 233)
  310. if (text == mainNode.getAttribute('currentQuestion')) return
  311. mainNode.setAttribute('currentQuestion', text)
  312. let answerNode = mainNode.querySelector('.answer-polygon')
  313. if (answerNode == null) {
  314. answerNode = document.createElement('div')
  315. answerNode.setAttribute('class', 'answer-polygon')
  316. mainNode.append(answerNode)
  317. }
  318. try {
  319. answerNode.innerHTML = 'searching...'
  320. } catch {
  321. return
  322. }
  323. let optionNodes = mainNode.querySelectorAll(config.options)
  324. searchAnswer(text, answerNode, optionNodes, mainNode)
  325. }, 6000)
  326. })
  327. }, 233)
  328. })();

QingJ © 2025

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