Greasy Fork

Greasy Fork is available in English.

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

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

在您安装前,Greasy Fork 希望您知道此脚本声明其包含了一些负面功能。这些功能也许会使脚本作者获利,而不能给您带来任何直接的金钱收益。

此脚本会在您访问的网站中插入广告

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==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/[email protected]/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...
}))();