Greasy Fork is available in English.

🔥知到/智慧树-网课【答题小助手】-- 持续更新 | 支持作业、测验、考试答题 | 题库持续更新 | 题库永久免费 |✍💯

🔥知到/智慧树-网课答题小助手。支持以下功能:1、智慧树课堂测验答题【支持自动搜索相关题目、自动做题和保存】;2、智慧树课程考试答题【支持自动搜索相关题目、自动做题和保存】

// ==UserScript==
// @name         🔥知到/智慧树-网课【答题小助手】-- 持续更新 | 支持作业、测验、考试答题 | 题库持续更新 | 题库永久免费 |✍💯
// @namespace    starlism
// @version      1.3
// @description  🔥知到/智慧树-网课答题小助手。支持以下功能:1、智慧树课堂测验答题【支持自动搜索相关题目、自动做题和保存】;2、智慧树课程考试答题【支持自动搜索相关题目、自动做题和保存】
// @author       starlism
// @match        *://*.zhihuishu.com/stuExamWeb*
// @connect      free.tikuhai.com
// @run-at       document-start
// @grant        unsafeWindow
// @grant        GM_xmlhttpRequest
// @grant        GM_addStyle
// @grant        GM_getResourceText
// @grant        GM_setClipboard
// @grant        GM_setValue
// @grant        GM_getValue
// @resource css https://unpkg.com/bootstrap@3.3.7/dist/css/bootstrap.min.css
// @license      MIT
// @antifeature  ads
// ==/UserScript==

const qrCode = `

`

enableWebpackHook();

const sleep = (ms) => {
    return new Promise(resolve => setTimeout(resolve, ms));
};

const config = {
    awaitTime: 5000,
    stopTimer: false,
    questionCount: 0,
    finishCount: 0,
    questionType: {
        '判断题': 10,
        '单选题': 20,
        '多选题': 25,
        '填空题': 30,
        '问答题': 40,
    }
};

// answer question and click to next question
function answerQuestion(questionBody, questionIndex) {
    const questionTitle = questionBody.querySelector('.subject_describe div,.smallStem_describe p').__Ivue__._data.shadowDom.textContent;
    appendToTable(questionTitle,"", questionIndex);
    // 题目类型处理
    const questionType = questionBody.querySelector(".subject_type").innerText.match(/【(.+)】|$/)[1]
    let type = config.questionType[questionType]
    if (type === undefined) {
        type = -1;
    }
    // console.log(type);
    // get answer
    GM_xmlhttpRequest({
        method: "GET",
        url: "https://free.tikuhai.com/q?q=" + encodeURIComponent(questionTitle),
        //         headers: {
        //             "Content-type": "application/x-www-form-urlencoded"
        //         },
        //         data: 'question=' + encodeURIComponent(questionTitle),
        onload: function(xhr) {
            const res = JSON.parse(xhr.responseText);
            const msg = res.msg;
            let answerString = res.data;
            if (msg === "暂无答案") {
                answerString = "暂无答案"
                changeAnswerInTable(answerString, questionIndex, true)
            } else {
                let isSelect = chooseAnswer(type, questionBody, answerString);
                changeAnswerInTable(answerString, questionIndex, isSelect)
            }
            // switch to next question
            document.querySelectorAll('.switch-btn-box > button')[1].click()
        },
        onerror: function(err) {
            console.log(err)
        }
    });
}

// choose the correct option of question
function chooseAnswer(questionType, questionBody, answerString) {
    let isSelect = false;
    // 判断题
    if (questionType === 10) {
        // 获取选项
        const firstOptionText = questionBody.querySelector(".node_detail")
        const firstOption = questionBody.querySelector(".nodeLab")
        // 选项表示正确
        if (firstOptionText.innerText.match(/(^|,)(正确|是|对|√|T|ri|right|true)(,|$)/)) {
            if (answerString.match(/(^|,)(正确|是|对|√|T|ri|right|true)(,|$)/)) {
                firstOption.click();
                isSelect = true;
            } else {
                // const secondOption = questionBody.querySelectorAll(".nodeLab")[1];
                // secondOption.cilck();
                questionBody.querySelectorAll(".nodeLab")[1].click();
                isSelect = true;
            }
        } else {
            if (answerString.match(/(^|,)(正确|是|对|√|T|ri|right|true)(,|$)/)) {
                // const secondOption = questionBody.querySelectorAll(".nodeLab")[1];
                // secondOption.cilck();
                questionBody.querySelectorAll(".nodeLab")[1].click();
                isSelect = true;
            } else {
                firstOption.click();
                isSelect = true;
            }
        }
    }
    // 单选题
    if (questionType === 20) {
        const regexPattern = new RegExp("^(" + answerString + ")$");
        // 获取所有选项
        const optionsTextAll = questionBody.querySelectorAll(".node_detail")
        const optionsAll = questionBody.querySelectorAll(".nodeLab")
        for (let i = 0; i < optionsAll.length; i++) {
            if (regexPattern.test(optionsTextAll[i].innerText)) {
                optionsAll[i].click();
                isSelect = true;
                break;
            }
        }
    }
    // 多选题
    if (questionType === 25) {
        const regexPattern = new RegExp("^(" + answerString + ")$");
        // 获取所有选项
        const optionsTextAll = questionBody.querySelectorAll(".node_detail")
        const optionsAll = questionBody.querySelectorAll(".nodeLab")
        for (let i = 0; i < optionsAll.length; i++) {
            if (regexPattern.test(optionsTextAll[i].innerText)) {
                optionsAll[i].click();
                isSelect = true;
            }
        }
    }
    // 填空题

    // 问答题

    return isSelect;
}

function appendToTable(questionTitle, answerString, questionIndex) {
    const table = document.querySelector("#record-table");
    table.innerHTML += `<tr><td>` + questionIndex + `</td><td>` + questionTitle + `</td><td id=answer${questionIndex}>正在搜索...</td></tr>`
    //     const tr = document.createElement("tr");
    //     tr.appendChild(1);
    //     tr.appendChild(questionTitle);
    //     tr.appendChild(answerString);
    //     table.appendChild(tr)
}

function changeAnswerInTable(answerString, questionIndex, isSelect) {
    document.querySelector(`#answer${questionIndex}`).innerText = answerString;
    if (answerString === "暂无答案") {
        // document.querySelector(`#answer${questionIndex}`).style.color = 'red';
        document.querySelector(`#answer${questionIndex}`).insertAdjacentHTML('beforeend',`<p style="color:red">扫码下载APP从完整版题库中获取答案</p>`)
    }
    if (!isSelect) {
        document.querySelector(`#answer${questionIndex}`).insertAdjacentHTML('beforeend',`<p style="color:green">未匹配答案,请根据搜索结果手动选择答案</p>`)
    }
}

function enableWebpackHook() {
    let originCall = Function.prototype.call
    Function.prototype.call = function (...args) {
        const result = originCall.apply(this, args)
        if (args[2]?.default?.version === '2.5.2') {
            args[2]?.default?.mixin({
                mounted: function () {
                    this.$el['__Ivue__'] = this
                }
            })
        }
        return result
    }
}


unsafeWindow.onload = (() => (async () => {
    // css style
    GM_addStyle(GM_getResourceText("css"));
    const cssStyle = `
    <div class="panel panel-info" style="z-index:99999; position:fixed; left:0; top:10%; width:22vw">
      <div class="panel-heading">
        <h3 class="panel-title" style="display:flex; align-items:center; justify-content:center">✍智慧树网课作业考试小助手✍</h3>
      </div>
      <div class="panel-body" id="qr-code" style="display:flex; align-items:center; justify-content:center; margin-top:-10px"></div>
      <div class="panel-body" style="display:flex; align-items:center; justify-content:center; margin-top:-30px">扫码下载APP,获得完整版题库答案</div>
      <div style="display:flex; align-items:center; justify-content:center">读取题目中...</div>
      <div class="panel-body" style="max-height:35vh; overflow-y:scroll; margin-top:0px">
        <div class="bs-example" data-example-id="bordered-table">
          <table class="table table-bordered" id="record-table">
            <thead>
              <tr>
                <th>#</th>
                <th>题目</th>
                <th>答案</th>
              </tr>
            </thead>
            <tbody>
            </tbody>
          </table>
        </div>
      </div>
    </div>
    `
    document.body.insertAdjacentHTML('beforeend', cssStyle);

    // QR Code
    const img = new Image();
    img.src = qrCode;
    img.width = 200;
    img.height = 200;
    document.querySelector("#qr-code").appendChild(img);

    // wait load
    await sleep(config.awaitTime);

    // get all question body
    var questionBodyAll = document.querySelectorAll(".examPaper_subject.mt20")
    if (questionBodyAll.length === 0) {
        return;
    }
    config.questionCount = questionBodyAll.length;

    // generate interval
    answerQuestion(questionBodyAll[0], 1)
    let finishCount = 1;
    var interval = setInterval(()=>{
        answerQuestion(questionBodyAll[finishCount], finishCount+1);
        finishCount += 1;
        if (finishCount === questionBodyAll.length) {
            clearInterval(interval);
            return;
        }
    }, 3000);

    //     for (let i = 0; i < questionBodyAll.length; i++) {
    //         setTimeout(answerQuestion(questionBodyAll[i]), 1000*i);
    //     }


    // setInterval(()=>{answerQuestion()}, 1000);

    // document.querySelector(".asdfgh").innerHTML = questionTitle
    // console.log(questionTitle)
    // Your code here...
}))();