智慧树 - 网络请求中有答案时,自动答选择题

全自动,无界面;应该可以答多选题和对错题;只有在网络请求中有答案时才能自动答题,否则这个脚本什么也不会干

// ==UserScript==
// @name        智慧树 - 网络请求中有答案时,自动答选择题
// @description 全自动,无界面;应该可以答多选题和对错题;只有在网络请求中有答案时才能自动答题,否则这个脚本什么也不会干
// @namespace   UnKnown
// @author      UnKnown
// @license     AGPL-3.0-or-later
// @version     1.2
// @icon        
// @match       https://hiexam.zhihuishu.com/atHomeworkExam/stu/*
// @grant       none
// @inject-into page
// @run-at      document-start
// ==/UserScript==

"use strict";

const isNonEmptyArray = arr => Array.isArray(arr) && arr.length > 0;
const isAnswerURL = urlStr => new URL(urlStr).pathname.endsWith("/getDoQuestSingle");

const msgPrefix = "[智慧树 - 网络请求中有答案时,自动答选择题]\n";
const consoleConfirm = (type, msg) => {
	console[type](msgPrefix + msg);
	confirm(
		msgPrefix + msg +
		"\n\n按下 F12 打开开发者工具查看和复制日志,\n",
		"点击 “确定” 打开 GreaseFork 进行反馈,\n",
		"点击 “取消” 关闭本提示。"
	);
};

const parseQuestionOption = option => (
	option?.isCorrect === true &&
	typeof option?.id === "number"
	? document.querySelector(`.questionBox .optionUl input[value="${ option.id }"]`).click()
	: false
);

/* 已知的 questionTypeId:
    1: 单选题
    2: 多选题
    3: 填空题
    4: 问答题
    5: 分析题/解答题/计算题/证明题
    9: 阅读理解(选择)/完型填空
   14: 判断题 */

const parseRt = rt => {
	const questionTypeId = rt?.questionTypeId;
	if ([1, 2, 14].some(id => id === questionTypeId)) {
		const questionOptionList = rt?.questionOptionList;
		if (isNonEmptyArray(questionOptionList)) {
			questionOptionList.map(parseQuestionOption).some(result => result !== false) &&
			console.log(msgPrefix + "网络请求中没有答案。");
		}
	} else {
		[3, 4, 5, 9, 14].every(id => id !== questionTypeId) ||
		consoleConfirm("warn", `发现新的题目类型,ID:${questionTypeId} 类型:${rt?.questionName},请反馈!`);
	}
};

// https://hiexam-server.zhihuishu.com/zhsathome/homeworkUserPaper/getDoQuestSingle?homeworkId=22737&questionId=98337

const onXHRLoad = ({target: xhr}) => {
	if (xhr.status === 200 && isAnswerURL(xhr.responseURL)) {

		let responseJSON;
		try { responseJSON = JSON.parse(xhr.responseText); }
		catch (e) { consoleConfirm("error", "无法解析响应 JSON。错误信息:" + e); }

		const rt = responseJSON?.rt;
		rt && typeof rt === "object"
		? parseRt(rt)
		: consoleConfirm("error", "响应 JSON 中的 rt 属性无效。具体值为:" + JSON.stringify(rt));

	}
};

const xhrProto = unsafeWindow.XMLHttpRequest.prototype;
xhrProto.send = new Proxy(xhrProto.send, {
	apply: (target, thisArg, args) => {
		thisArg.addEventListener("load", onXHRLoad);
		return Reflect.apply(target, thisArg, args);
	}
});