知到智慧树全自动刷问答平时分助手

通过ai快速地,全自动刷智慧树问答,半自动化可选择地提问,并且自动添加防提问重复机制,添加防止无效问题提问机制,嘎嘎好用。

// ==UserScript==
// @name         知到智慧树全自动刷问答平时分助手
// @version      1.0.8
// @description  通过ai快速地,全自动刷智慧树问答,半自动化可选择地提问,并且自动添加防提问重复机制,添加防止无效问题提问机制,嘎嘎好用。
// @author       fenda
// @match      	 *://*.zhihuishu.com/*
// @grant        GM_xmlhttpRequest
// @grant        GM_setValue
// @grant        GM_getValue
// @icon         
// @grant        GM_notification
// @grant        GM_registerMenuCommand
// @license      MIT
// @namespace http://fenda.github.io/
// ==/UserScript==

(function () {

    'use strict';

    const defaultAPIKey = "sk-zt3dcYmk8D4FECtgDa7427F7E71444B589D6FbD6CcC44b30";
    const defaultAPIUrl = "https://api.gpt.ge/v1/chat/completions";
    const defaultModel = "glm-4-flash";
    const intervalTime = GM_getValue("intervalTime", 2000);
    const initialCountdownTime = GM_getValue("countdownTime", 20);

    let requestLock = false;
    let noAnswerButtonCount = 0;
    let countdownInterval;

    const inputEvent = new Event("input", { bubbles: true, cancelable: true });

    function debounce(fn, delay) {
        let timeout;
        return function () {
            clearTimeout(timeout);
            timeout = setTimeout(() => fn.apply(this, arguments), delay);
        };
    }

    function initMutationObserver() {
        const debouncedUpdate = debounce(() => {
            console.log("DOM 变化检测到,更新问题列表...");
            addQuestionStatus();
            if (GM_getValue("isAutoScrolling", false)) {
                const isAutoScrolling = GM_getValue("isAutoScrolling", false);
                if (isAutoScrolling) {
                    scrollToLastUnansweredQuestion();
                }
            }
            checkHomeRegex();
        }, 500);

        const observer = new MutationObserver((mutations) => {
            debouncedUpdate();

        });

        const config = {
            attributes: true,
            childList: true,
            subtree: true,
            characterData: true
        };

        observer.observe(document.body, config);

        console.log("MutationObserver 已启动,正在监听 DOM 变化...");
    }


    function createQuestionManagementModal() {
        const modal = document.createElement('div');
        modal.className = 'question-management-modal';
        modal.style = `
            position: fixed;
            top: 50px;
            left: 5%;
            width: 90%;
            height: 80%;
            background: linear-gradient(145deg, #ffffff, #e6e6e6);
            box-shadow: 0px 10px 20px rgba(0, 0, 0, 0.1);
            border-radius: 15px;
            z-index: 1000;
            padding: 20px;
            overflow: hidden;
        `;

        let questionsData = GM_getValue("questionsData", {});
        const lastViewedCourseIndex = GM_getValue("lastViewedCourseIndex", 0);
        let questionsPerPage = GM_getValue("questionsPerPage", 10);

        const coursetabs = document.createElement('div');
        coursetabs.style = `
            display: flex;
            justify-content: center;
            margin-bottom: 15px;
            border-bottom: 1px solid #e0e0e0;
        `;

        const contentArea = document.createElement('div');
        contentArea.style.height = 'calc(100% - 100px)';

        Object.keys(questionsData).forEach((courseName, index) => {
            const courseTab = document.createElement('div');
            courseTab.textContent = courseName;
            courseTab.style = `
                padding: 10px 15px;
                cursor: pointer;
                background: ${index === lastViewedCourseIndex ? 'linear-gradient(145deg, #4c88d6, #3b5998)' : 'transparent'};
                color: ${index === lastViewedCourseIndex ? 'white' : 'black'};
                border-radius: 5px 5px 0 0;
                margin-right: 5px;
            `;

            const courseTable = document.createElement('div');
            courseTable.style = `
                display: ${index === lastViewedCourseIndex ? 'block' : 'none'};
                width: 100%;
                height: 100%;
                overflow-y: auto;
            `;

            const tableContainer = document.createElement('div');
            tableContainer.style = `
                width: 100%;
                overflow-x: auto;
            `;

            const table = document.createElement('table');
            table.style = `
                width: 100%;
                border-collapse: collapse;
                table-layout: fixed;
            `;

            const thead = document.createElement('thead');
            thead.style = `
                position: sticky;
                top: 0;
                background: linear-gradient(145deg, #4c88d6, #3b5998);
                color: white;
                z-index: 1;
            `;
            thead.innerHTML = `
                <tr>
                    <th style="padding: 10px; text-align: left; width: 60%;">题目
                        <input type="text" placeholder="搜索..." class="filter-input" style="margin-left: 5px; padding: 3px; border-radius: 5px; border: 1px solid #ccc; color: black;">
                    </th>
                    <th style="padding: 10px; text-align: center; width: 20%;">状态</th>
                    <th style="padding: 10px; text-align: center; width: 20%;">操作</th>
                </tr>
            `;

            const tbody = document.createElement('tbody');

            let questions = Object.entries(questionsData[courseName]);
            let currentPage = GM_getValue(`lastViewedPage_${courseName}`, 1);
            let totalPages = Math.ceil(questions.length / questionsPerPage);
            let isSearchMode = false;

            function displayQuestions(page, filteredQuestions = null) {
                tbody.innerHTML = '';
                const questionsToDisplay = filteredQuestions || questions;
                const startIndex = (page - 1) * questionsPerPage;
                const endIndex = Math.min(startIndex + questionsPerPage, questionsToDisplay.length);

                for (let i = startIndex; i < endIndex; i++) {
                    const [question, status] = questionsToDisplay[i];
                    const questionIndex = questions.findIndex(([q]) => q === question);
                    const questionPage = Math.ceil((questionIndex + 1) / questionsPerPage);

                    const row = document.createElement('tr');
                    row.setAttribute('data-question', question);
                    let questionHTML = question;
                    if (isSearchMode && filteredQuestions) {
                        questionHTML += ` <span style="color: orange;">(第${questionPage}页)</span>`;
                    }
                    row.innerHTML = `
                        <td style="padding: 10px; border-bottom: 1px solid #e0e0e0; white-space: normal; word-break: break-word;">${questionHTML}</td>
                        <td style="padding: 10px; border-bottom: 1px solid #e0e0e0; text-align: center; color: ${status === '已回答' ? 'green' : 'red'}; white-space: nowrap;">
                            ${status}
                        </td>
                        <td style="padding: 10px; border-bottom: 1px solid #e0e0e0; text-align: center; white-space: nowrap;">
                            <button class="set-answered" data-course="${courseName}" data-question="${question}" style="
                                margin-right: 5px;
                                padding: 5px 10px;
                                background: green;
                                color: white;
                                border: none;
                                border-radius: 5px;
                            ">设为已答</button>
                            <button class="set-unanswered" data-course="${courseName}" data-question="${question}" style="
                                padding: 5px 10px;
                                background: red;
                                color: white;
                                border: none;
                                border-radius: 5px;
                            ">设为未答</button>
                        </td>
                    `;
                    tbody.appendChild(row);
                }
            }

            displayQuestions(currentPage);

            thead.querySelector('.filter-input').addEventListener('input', function () {
                const filterValue = this.value.toLowerCase();
                isSearchMode = filterValue.length > 0;
                const filteredQuestions = Object.entries(questionsData[courseName]).filter(([question, status]) => {
                    return question.toLowerCase().includes(filterValue);
                });
                totalPages = Math.ceil(filteredQuestions.length / questionsPerPage);
                currentPage = 1;
                displayQuestions(currentPage, filteredQuestions);
                updatePagination();
            });

            table.addEventListener('click', function (event) {
                const target = event.target;
                const courseName = target.dataset.course;
                const question = target.dataset.question;

                const questionsData = GM_getValue("questionsData", {});

                if (target.classList.contains('set-answered')) {
                    questionsData[courseName][question] = '已回答';
                    GM_setValue("questionsData", questionsData);
                    refreshQuestionManagementModal(courseName, currentPage);
                }

                if (target.classList.contains('set-unanswered')) {
                    questionsData[courseName][question] = '未回答';
                    GM_setValue("questionsData", questionsData);
                    refreshQuestionManagementModal(courseName, currentPage);
                }
            });

            table.appendChild(thead);
            table.appendChild(tbody);
            tableContainer.appendChild(table);
            courseTable.appendChild(tableContainer);

            courseTab.addEventListener('click', () => {
                Array.from(coursetabs.children).forEach((tab, idx) => {
                    tab.style.background = 'transparent';
                    tab.style.color = 'black';
                    contentArea.children[idx].style.display = 'none';
                });

                courseTab.style.background = 'linear-gradient(145deg, #4c88d6, #3b5998)';
                courseTab.style.color = 'white';
                courseTable.style.display = 'block';
                GM_setValue("lastViewedCourseIndex", index);
                currentPage = GM_getValue(`lastViewedPage_${courseName}`, 1);
                displayQuestions(currentPage);
                updatePagination();
                isSearchMode = false;
            });

            courseTab.setAttribute('data-course', courseName);

            coursetabs.appendChild(courseTab);
            contentArea.appendChild(courseTable);

            const paginationContainer = document.createElement('div');
            paginationContainer.style = `
                display: flex;
                justify-content: space-between;
                align-items: center;
                padding: 10px;
                background: #f0f0f0;
                border-radius: 0 0 15px 15px;
                position: absolute;
                bottom: 0;
                left: 0;
                width: 100%;
                box-sizing: border-box;
            `;

            const questionsPerPageContainer = document.createElement('div');
            questionsPerPageContainer.style = `
                display: flex;
                align-items: center;
            `;

            const questionsPerPageLabel = document.createElement('span');
            questionsPerPageLabel.textContent = '每页显示:';
            questionsPerPageLabel.style = `
                margin-right: 5px;
            `;

            const questionsPerPageInput = document.createElement('input');
            questionsPerPageInput.type = 'number';
            questionsPerPageInput.value = questionsPerPage;
            questionsPerPageInput.style = `
                width: 60px;
                padding: 5px;
                border-radius: 5px;
                border: 1px solid #ccc;
                margin-right: 5px;
            `;

            const saveButton = document.createElement('button');
            saveButton.textContent = '保存';
            saveButton.style = `
                padding: 5px 10px;
                border: none;
                border-radius: 5px;
                background: #4CAF50;
                color: white;
                cursor: pointer;
                margin-right: 5px;
            `;

            saveButton.addEventListener('click', () => {
                const newQuestionsPerPage = parseInt(questionsPerPageInput.value);
                if (!isNaN(newQuestionsPerPage) && newQuestionsPerPage > 0) {
                    GM_setValue("questionsPerPage", newQuestionsPerPage);
                    questionsPerPage = newQuestionsPerPage;
                    totalPages = Math.ceil(Object.keys(questionsData[courseName]).length / questionsPerPage);
                    displayQuestions(currentPage);
                    updatePagination();
                } else {
                    showAlert('请输入有效的每页显示数量');
                    questionsPerPageInput.value = questionsPerPage;
                }
            });

            const setPageAnsweredButton = document.createElement('button');
            setPageAnsweredButton.textContent = '一键已答';
            setPageAnsweredButton.style = `
                padding: 5px 10px;
                background: green;
                color: white;
                border: none;
                border-radius: 5px;
                cursor: pointer;
                margin-right: 5px;
            `;
            setPageAnsweredButton.addEventListener('click', () => {
                setQuestionsStatusForCurrentPage(courseName, '已回答', currentPage, questionsPerPage);
                refreshQuestionManagementModal(courseName, currentPage);
            });

            const setPageUnansweredButton = document.createElement('button');
            setPageUnansweredButton.textContent = '一键本页未答';
            setPageUnansweredButton.style = `
                padding: 5px 10px;
                background: red;
                color: white;
                border: none;
                border-radius: 5px;
                cursor: pointer;
            `;
            setPageUnansweredButton.addEventListener('click', () => {
                setQuestionsStatusForCurrentPage(courseName, '未回答', currentPage, questionsPerPage);
                refreshQuestionManagementModal(courseName, currentPage);
            });

            questionsPerPageContainer.appendChild(questionsPerPageLabel);
            questionsPerPageContainer.appendChild(questionsPerPageInput);
            questionsPerPageContainer.appendChild(saveButton);
            questionsPerPageContainer.appendChild(setPageAnsweredButton);
            questionsPerPageContainer.appendChild(setPageUnansweredButton);

            const paginationControls = document.createElement('div');
            paginationControls.style = `
                display: flex;
                align-items: center;
            `;

            const prevButton = document.createElement('button');
            prevButton.textContent = '上一页';
            prevButton.style = `
                padding: 5px 10px;
                margin: 0 5px;
                border: none;
                border-radius: 5px;
                background: #ddd;
                cursor: pointer;
            `;
            prevButton.addEventListener('click', () => {
                if (currentPage > 1) {
                    currentPage--;
                    GM_setValue(`lastViewedPage_${courseName}`, currentPage);
                    displayQuestions(currentPage);
                    updatePagination();
                }
            });

            const nextButton = document.createElement('button');
            nextButton.textContent = '下一页';
            nextButton.style = `
                padding: 5px 10px;
                margin: 0 5px;
                border: none;
                border-radius: 5px;
                background: #ddd;
                cursor: pointer;
            `;
            nextButton.addEventListener('click', () => {
                if (currentPage < totalPages) {
                    currentPage++;
                    GM_setValue(`lastViewedPage_${courseName}`, currentPage);
                    displayQuestions(currentPage);
                    updatePagination();
                }
            });

            const pageInfo = document.createElement('span');
            pageInfo.style = `
                margin: 0 10px;
            `;

            const gotoPageContainer = document.createElement('div');
            gotoPageContainer.style = `
                display: flex;
                align-items: center;
                margin-left: auto;
            `;

            const gotoPageInput = document.createElement('input');
            gotoPageInput.type = 'number';
            gotoPageInput.placeholder = '页码';
            gotoPageInput.style = `
                width: 50px;
                padding: 5px;
                border-radius: 5px;
                border: 1px solid #ccc;
                margin-right: 5px;
            `;

            const gotoPageButton = document.createElement('button');
            gotoPageButton.textContent = '未答';
            gotoPageButton.style = `
                padding: 5px 10px;
                border: none;
                border-radius: 5px;
                background: #4CAF50;
                color: white;
                cursor: pointer;
            `;
            gotoPageButton.addEventListener('click', () => {
                const pageNumber = parseInt(gotoPageInput.value);
                if (!isNaN(pageNumber) && pageNumber >= 1 && pageNumber <= totalPages) {
                    currentPage = pageNumber;
                    GM_setValue(`lastViewedPage_${courseName}`, currentPage);
                    displayQuestions(currentPage);
                    updatePagination();
                } else {
                    alert('请输入有效的页码');
                    gotoPageInput.value = '';
                }
            });

            gotoPageContainer.appendChild(gotoPageInput);
            gotoPageContainer.appendChild(gotoPageButton);

            function updatePagination() {
                pageInfo.textContent = `第 ${currentPage} 页 / 共 ${totalPages} 页`;
            }

            updatePagination();

            paginationControls.appendChild(prevButton);
            paginationControls.appendChild(pageInfo);
            paginationControls.appendChild(nextButton);

            paginationContainer.appendChild(questionsPerPageContainer);
            paginationContainer.appendChild(paginationControls);
            paginationContainer.appendChild(gotoPageContainer);

            courseTable.appendChild(paginationContainer);

            currentPage = GM_getValue(`lastViewedPage_${courseName}`, 1);
            displayQuestions(currentPage);
            updatePagination();
        });

        const closeButton = document.createElement('button');
        closeButton.textContent = '关闭';
        closeButton.style = `
            position: absolute;
            top: 10px;
            right: 10px;
            padding: 5px 10px;
            background: red;
            color: white;
            border: none;
            border-radius: 5px;
        `;
        closeButton.addEventListener('click', () => modal.remove());

        modal.appendChild(coursetabs);
        modal.appendChild(contentArea);
        modal.appendChild(closeButton);

        document.body.appendChild(modal);
    }

    function setQuestionsStatusForCurrentPage(courseName, status, currentPage, questionsPerPage) {
        let questionsData = GM_getValue("questionsData", {});
        if (questionsData[courseName]) {
            const questions = Object.keys(questionsData[courseName]);
            const startIndex = (currentPage - 1) * questionsPerPage;
            const endIndex = Math.min(startIndex + questionsPerPage, questions.length);

            for (let i = startIndex; i < endIndex; i++) {
                const question = questions[i];
                questionsData[courseName][question] = status;
            }
            GM_setValue("questionsData", questionsData);
        }
    }

    function refreshQuestionManagementModal(courseName, currentPage) {
        GM_setValue(`lastViewedPage_${courseName}`, currentPage);
        const scrollTop = document.querySelector('.question-management-modal')?.scrollTop || 0;
        const existingModal = document.querySelector('.question-management-modal');
        if (existingModal) {
            existingModal.remove();
        }
        createQuestionManagementModal();
        document.querySelector('.question-management-modal').scrollTop = scrollTop;
    }


    function displayCountdown(countdown) {
        let participatoryDiv = document.querySelector('.MyParticipatory-div');
        let qrCodeDiv = document.querySelector('.QRcode-div');
        let countdownElement = document.querySelector('.new-neumorphic-card');
        const linkClicked = GM_getValue("linkClicked", false);

        if (participatoryDiv && qrCodeDiv) {
            const { answeredCount, totalCount } = calculateAnsweredQuestions();

            if (!countdownElement) {
                countdownElement = createCountdownElement(countdown, linkClicked, answeredCount, totalCount);
                qrCodeDiv.parentNode.insertBefore(countdownElement, qrCodeDiv);
                setupEventListeners(countdownElement);
            } else {
                updateCountdownElement(countdownElement, countdown, answeredCount, totalCount);
            }
        }
    }

    function calculateAnsweredQuestions() {
        let answeredCount = 0;
        let totalCount = 0;
        const questionItems = document.querySelectorAll('.question-item');
        questionItems.forEach(item => {
            const spanElement = item.querySelector('.question-content span');
            const questionText = spanElement ? spanElement.title.trim() : '';
            if (questionText) {
                const courseNameElement = document.querySelector(".course-name");
                if (courseNameElement) {
                    const courseName = courseNameElement.innerText.trim();
                    const questionsData = GM_getValue("questionsData", {});
                    if (questionsData[courseName] && questionsData[courseName][questionText] === "已回答") {
                        answeredCount++;
                    }
                    totalCount++;
                }
            }
        });
        return { answeredCount, totalCount };
    }

    function createCountdownElement(countdown, linkClicked, answeredCount, totalCount) {
        const element = document.createElement('div');
        element.className = 'new-neumorphic-card';
        element.style = `
            padding: 15px;
            border-radius: 15px;
            background: linear-gradient(145deg, #ffffff, #e6e6e6);
            box-shadow: 6px 6px 12px #aaaaaa,
                        -6px -6px 12px #ffffff;
            display: inline-block;
            max-width: 300px;
        `;

        const modelDropdownHTML = createModelDropdownHTML();

        element.innerHTML = `
            <div class="Participatorytitle-div" style="white-space: normal; overflow-wrap: break-word; display: flex; flex-direction: column;">
                <div style="margin-bottom: 10px; font-weight: bold;">倒计时
                    <i class="el-tooltip item" tabindex="0"
                    style="margin-left: 4px; cursor: pointer; position: relative;" id="infoIcon">
                        ℹ️
                    </i>
                    <span style="margin-left: 10px; white-space: nowrap;">${countdown} 秒</span>
                    <button id="toggleCountdownButton" style="
                        margin-left: 10px;
                        padding: 10px 20px;
                        border: none;
                        border-radius: 10px;
                        background: linear-gradient(145deg, #4c88d6, #3b5998);
                        box-shadow: 2px 2px 4px #2f4686, -2px -2px 4px #507bcf;
                        color: white;
                        font-weight: bold;
                        cursor: pointer;
                    ">${countdownInterval ? '暂停' : '继续'}</button>
                </div>
                ${!linkClicked ? `<a href="#" id="purchaseLink" target="_blank" style="margin-top: 10px; text-decoration: none; color: #0066cc; position: relative;">注册(不可用)购买API</a>` : ''}
                <div style="display: flex; flex-direction: column; align-items: start; margin-top: 10px;">
                    <label style="margin-bottom: 15px; display: flex; align-items: center; white-space: nowrap;">API_KEY:
                        <input type="text" id="apiKeyInput" style="
                            margin-left: 10px;
                            width: 100%;
                            padding: 10px;
                            border-radius: 10px;
                            border: none;
                            outline: none;
                            background: linear-gradient(145deg, #e6e6e6, #ffffff);
                            box-shadow: inset 2px 2px 4px #cccccc,
                                        inset -2px -2px 4px #ffffff;
                        ">
                    </label>
                    <label style="margin-bottom: 15px; display: flex; align-items: center; white-space: nowrap;">API_URL:
                        <input type="text" id="apiUrlInput" style="
                            margin-left: 10px;
                            width: 100%;
                            padding: 10px;
                            border-radius: 10px;
                            border: none;
                            outline: none;
                            background: linear-gradient(145deg, #e6e6e6, #ffffff);
                            box-shadow: inset 2px 2px 4px #cccccc,
                                        inset -2px -2px 4px #ffffff;
                        ">
                    </label>
                    <label style="margin-bottom: 15px; display: flex; align-items: center; white-space: nowrap;">倒计时范围:
                        <input type="number" id="countdownMinInput" style="
                            margin-left: 10px;
                            width: 48%;
                            padding: 10px;
                            border-radius: 10px;
                            border: none;
                            outline: none;
                            background: linear-gradient(145deg, #e6e6e6, #ffffff);
                            box-shadow: inset 2px 2px 4px #cccccc,
                                        inset -2px -2px 4px #ffffff;"
                            placeholder="最小值">
                        <input type="number" id="countdownMaxInput" style="
                            margin-left: 10px;
                            width: 48%;
                            padding: 10px;
                            border-radius: 10px;
                            border: none;
                            outline: none;
                            background: linear-gradient(145deg, #e6e6e6, #ffffff);
                            box-shadow: inset 2px 2px 4px #cccccc,
                                        inset -2px -2px 4px #ffffff;"
                            placeholder="最大值">
                    </label>
                    ${modelDropdownHTML}
                </div>
                <div style="margin-top: 15px; text-align: right;">
                    <button id="manageButton" style="
                        margin-right: 10px;
                        padding: 10px 20px;
                        border: none;
                        border-radius: 10px;
                        background: linear-gradient(145deg, #4c88d6, #3b5998);
                        box-shadow: 2px 2px 4px #2f4686, -2px -2px 4px #507bcf;
                        color: white;
                        font-weight: bold;
                        cursor: pointer;
                    ">管理答题列表</button>
                    <button id="saveButton" style="
                        padding: 10px 20px;
                        border: none;
                        border-radius: 10px;
                        background: linear-gradient(145deg, #4c88d6, #3b5998);
                        box-shadow: 2px 2px 4px #2f4686, -2px -2px 4px #507bcf;
                        color: white;
                        font-weight: bold;
                        cursor: pointer;
                    ">保存</button>
                </div>
                <div style="margin-top: 20px; text-align: center; font-size: 12px; color: #666;">
                    <span>当前版本:1.0.8(最新:</span><span id="latestVersion">加载中...</span><span>)</span>
                </div>
            </div>
        `;

        return element;
    }

    function createModelDropdownHTML() {
        const defaultModels = ["gemini-1.5-flash", "glm-4-flash"];
        let customModels = GM_getValue("customModels", []);
        let allModels = [...defaultModels, ...customModels];
        let selectedModel = GM_getValue("aimodel", "glm-4-flash");

        return `
            <label style="display: flex; align-items: center; white-space: nowrap; margin-bottom: 15px;">模型:
                <div class="custom-dropdown" style="position: relative; margin-left: 10px; width: 100%;">
                    <div class="dropdown-selected" style="
                        padding: 10px;
                        border-radius: 10px;
                        background: linear-gradient(145deg, #e6e6e6, #ffffff);
                        box-shadow: inset 4px 4px 8px #cccccc,
                                    inset -4px -4px 8px #ffffff;
                        cursor: pointer;
                        width: 100%;
                        box-sizing: border-box;
                        text-align: left;
                    ">${selectedModel}</div>
                    <div class="dropdown-options" style="
                        position: absolute;
                        top: 100%;
                        left: 0;
                        background: linear-gradient(145deg, #ffffff, #e6e6e6);
                        border-radius: 10px;
                        box-shadow: 4px 4px 8px #aaaaaa,
                                    -4px -4px 8px #ffffff;
                        z-index: 1;
                        display: none;
                        overflow: hidden;
                        min-width: 200px;
                    ">
                        ${allModels.map(model => `
                            <div class="dropdown-item" data-model="${model}" style="
                                padding: 10px;
                                cursor: pointer;
                                ${defaultModels.includes(model) ? '' : 'display: flex; justify-content: space-between; align-items: center;'}
                                background: linear-gradient(145deg, #ffffff, #e6e6e6);
                                border-bottom: 1px solid #e0e0e0;
                            ">
                                ${model}
                                ${defaultModels.includes(model) ? '' : `<span class="remove-model" data-model="${model}" style="cursor: pointer; color: red;">×</span>`}
                            </div>
                        `).join('')}
                        <div class="add-model-container" style="padding: 10px; display: flex; align-items: center;">
                            <input type="text" id="newModelInput" placeholder="输入模型名称" style="
                                padding: 8px;
                                border-radius: 5px;
                                border: none;
                                background: linear-gradient(145deg, #e6e6e6, #ffffff);
                                box-shadow: inset 2px 2px 4px #cccccc,
                                            inset -2px -2px 4px #ffffff;
                                width: 70%;
                                box-sizing: border-box;
                                height: 32px;
                            ">
                            <button id="saveModelButton" style="
                                border: none;
                                border-radius: 8px;
                                background: linear-gradient(145deg, #4c88d6, #3b5998);
                                box-shadow: 2px 2px 4px #2f4686,
                                            -2px -2px 4px #507bcf;
                                color: white;
                                cursor: pointer;
                                width: 25%;
                                text-align: center;
                                margin-left: 5px;
                                height: 32px;
                                display: flex;
                                justify-content: center;
                                align-items: center;
                            ">保存</button>
                        </div>
                    </div>
                </div>
            </label>
            <div class="answer-counter" style="
                margin-top: 10px;
                padding: 10px;
                border-radius: 10px;
                background: linear-gradient(145deg, #e6e6e6, #ffffff);
                box-shadow: inset 4px 4px 8px #cccccc,
                            inset -4px -4px 8px #ffffff;
                text-align: center;
                font-weight: bold;
                color: #4B5563;
                display: flex;
                flex-direction: column;
                align-items: center;
                justify-content: center;
            ">
                <div style="display: flex; align-items: center; justify-content: space-between;">
                    <div style="display: flex; flex-direction: column; border: 1px solid #ccc; padding: 5px; border-radius: 5px;">
                        <div>已答: ${0}</div>
                        <div>总数: ${0}</div>
                    </div>
                    <div style="display: flex; flex-direction: row; align-items: center;">
                        <button id="jumpToUnansweredButton" style="
                            margin-left: 10px;
                            margin-bottom: 5px;
                            padding: 5px 10px;
                            border: none;
                            border-radius: 8px;
                            background: linear-gradient(145deg, #4c88d6, #3b5998);
                            box-shadow: 2px 2px 4px #2f4686,
                                        -2px -2px 4px #507bcf;
                            color: white;
                            cursor: pointer;
                            white-space: normal;
                            line-height: 1.2;
                            text-align: center;
                            width: 60px;
                            height: 100%;
                        ">
                            跳转<br>未答
                        </button>
                        <button id="autoScrollButton" style="
                            margin-left: 10px;
                            padding: 5px 10px;
                            border: none;
                            border-radius: 8px;
                            background: linear-gradient(145deg, #4c88d6, #3b5998);
                            box-shadow: 2px 2px 4px #2f4686,
                                        -2px -2px 4px #507bcf;
                            color: white;
                            cursor: pointer;
                            white-space: normal;
                            line-height: 1.2;
                            text-align: center;
                            width: 60px;
                            height: 100%;
                        ">
                            自动<br>滚动
                        </button>
                    </div>
                </div>
            </div>
        `;
    }



    function updateCountdownElement(countdownElement, countdown, answeredCount, totalCount) {
        countdownElement.querySelector('.Participatorytitle-div div span').innerText = `${countdown} 秒`;
        const toggleButton = countdownElement.querySelector('#toggleCountdownButton');
        toggleButton.textContent = countdownInterval ? "暂停" : "继续";
        updateAnswerCounter(countdownElement, answeredCount, totalCount);
    }

    function updateAnswerCounter(countdownElement, answeredCount, totalCount) {
        const counterElement = countdownElement.querySelector('.answer-counter');
        const isAutoScrolling = GM_getValue("isAutoScrolling", false);
        if (counterElement) {
            counterElement.innerHTML = `
                <div style="display: flex; align-items: center; justify-content: space-between;">
                    <div style="display: flex; flex-direction: column; border: 1px solid #ccc; padding: 5px; border-radius: 5px;">
                        <div>已答: ${answeredCount}</div>
                        <div>总数: ${totalCount}</div>
                    </div>
                    <div style="display: flex; flex-direction: row; align-items: center;">
                        <button id="jumpToUnansweredButton" style="
                            margin-left: 10px;
                            margin-bottom: 5px;
                            padding: 5px 10px;
                            border: none;
                            border-radius: 8px;
                            background: linear-gradient(145deg, #4c88d6, #3b5998);
                            box-shadow: 2px 2px 4px #2f4686,
                                        -2px -2px 4px #507bcf;
                            color: white;
                            cursor: pointer;
                            white-space: normal;
                            line-height: 1.2;
                            text-align: center;
                            width: 60px;
                            height: 100%;
                        ">
                            跳转<br>未答
                        </button>
                        <button id="autoScrollButton" style="
                            margin-left: 10px;
                            padding: 5px 10px;
                            border: none;
                            border-radius: 8px;
                            background: linear-gradient(145deg, #4c88d6, #3b5998);
                            box-shadow: 2px 2px 4px #2f4686,
                                        -2px -2px 4px #507bcf;
                            color: white;
                            cursor: pointer;
                            white-space: normal;
                            line-height: 1.2;
                            text-align: center;
                            width: 60px;
                            height: 100%;
                        ">
                            ${isAutoScrolling ? '关闭<br>滚动' : '自动<br>滚动'}
                        </button>
                    </div>
                </div>
            `;
            const jumpButton = counterElement.querySelector('#jumpToUnansweredButton');
            setupJumpButton(jumpButton);
            const autoScrollButton = counterElement.querySelector('#autoScrollButton');
            setupAutoScrollButton(autoScrollButton);
        }
    }

    function setupAutoScrollButton(autoScrollButton) {
        autoScrollButton.addEventListener('click', () => {
            const isAutoScrolling = GM_getValue("isAutoScrolling", false);
            if (isAutoScrolling) {
                GM_setValue("isAutoScrolling", false);
                autoScrollButton.innerHTML = '自动<br>滚动';
                showAlert("自动滚动已关闭");
            } else {
                GM_setValue("isAutoScrolling", true);
                autoScrollButton.innerHTML = '关闭<br>滚动';
                showAlert("自动滚动已开启,将持续滚动未答题目到最上方");
                scrollToLastUnansweredQuestion();
            }
        });
        setupButtonShadowEffects(autoScrollButton);
    }




    function setupEventListeners(countdownElement) {
        const dropdown = countdownElement.querySelector('.custom-dropdown');
        const dropdownSelected = dropdown.querySelector('.dropdown-selected');
        const dropdownOptions = dropdown.querySelector('.dropdown-options');

        dropdownSelected.addEventListener('click', () => {
            dropdownOptions.style.display = dropdownOptions.style.display === 'block' ? 'none' : 'block';
        });

        dropdownOptions.addEventListener('click', (event) => {
            if (event.target.classList.contains('dropdown-item')) {
                const model = event.target.dataset.model;
                GM_setValue("aimodel", model);
                const defaultModels = ["gemini-1.5-flash", "glm-4-flash"];
                let customModels = GM_getValue("customModels", []);
                let allModels = [...defaultModels, ...customModels];
                let selectedModel = model;
                dropdownSelected.textContent = model;
                dropdownOptions.style.display = 'none';
            } else if (event.target.classList.contains('remove-model')) {
                const modelToRemove = event.target.dataset.model;
                let customModels = GM_getValue("customModels", []);
                customModels = customModels.filter(model => model !== modelToRemove);
                GM_setValue("customModels", customModels);
                const defaultModels = ["gemini-1.5-flash", "glm-4-flash"];
                let allModels = [...defaultModels, ...customModels];
                renderDropdownItems(dropdownOptions, allModels, defaultModels, dropdownSelected);
            }
        });


        const defaultModels = ["gemini-1.5-flash", "glm-4-flash"];
        let customModels = GM_getValue("customModels", []);
        let allModels = [...defaultModels, ...customModels];
        renderDropdownItems(dropdownOptions, allModels, defaultModels, dropdownSelected);

        document.addEventListener('click', (event) => {
            if (!dropdown.contains(event.target)) {
                dropdownOptions.style.display = 'none';
            }
        });

        document.getElementById("manageButton").addEventListener('click', createQuestionManagementModal);
        document.getElementById("apiKeyInput").value = GM_getValue("API_KEY", defaultAPIKey);
        document.getElementById("apiUrlInput").value = GM_getValue("API_URL", defaultAPIUrl);
        document.getElementById("countdownMinInput").value = GM_getValue("countdownMin", 20);
        document.getElementById("countdownMaxInput").value = GM_getValue("countdownMax", 30);

        document.getElementById("saveButton").addEventListener('click', function () {
            showAlert("保存参数成功");
            GM_setValue("API_KEY", document.getElementById("apiKeyInput").value);
            GM_setValue("API_URL", document.getElementById("apiUrlInput").value);
            GM_setValue("countdownMin", parseInt(document.getElementById("countdownMinInput").value));
            GM_setValue("countdownMax", parseInt(document.getElementById("countdownMaxInput").value));

            const newModel = dropdownOptions.querySelector('#newModelInput').value.trim();
            if (newModel && !getAllModels().includes(newModel)) {
                let customModels = GM_getValue("customModels", []);
                customModels.push(newModel);
                GM_setValue("customModels", customModels);
                const defaultModels = ["gemini-1.5-flash", "glm-4-flash"];
                let allModels = [...defaultModels, ...customModels];
                renderDropdownItems(dropdownOptions, allModels, defaultModels, dropdownSelected);
            }
            const selectedModel = dropdownSelected.textContent;
            GM_setValue("aimodel", selectedModel);
            alert('设置已保存');
        });

        const toggleButton = document.getElementById("toggleCountdownButton");
        setupButtonShadowEffects(toggleButton);
        toggleButton.addEventListener('click', () => {
            if (countdownInterval) {
                clearInterval(countdownInterval);
                countdownInterval = null;
                toggleButton.textContent = "继续";
            } else {
                const savedCountdown = GM_getValue("currentCountdown", GM_getValue("countdownTime", initialCountdownTime));
                startCountdown(savedCountdown);
                toggleButton.textContent = "暂停";
            }
        });

        const saveButton = document.getElementById("saveButton");
        setupButtonShadowEffects(saveButton);

        const manageButton = document.getElementById("manageButton");
        setupButtonShadowEffects(manageButton);

        const jumpButton = document.getElementById("jumpToUnansweredButton");
        setupJumpButton(jumpButton);

        checkForLatestVersion();

        setupPurchaseLink(countdownElement);

        setupInfoIcon(countdownElement);
    }


    function setupPurchaseLink(countdownElement) {
        const linkClicked = GM_getValue("linkClicked", false);
        if (!linkClicked) {
            const purchaseLink = countdownElement.querySelector("#purchaseLink");
            purchaseLink.addEventListener('mouseenter', function (event) {
                const tooltipText = "非作者运营勿找,点击后刷新不再显示";
                const tooltip = document.createElement('div');
                tooltip.id = 'purchaseLinkTooltip';
                tooltip.style = `
                    position: absolute;
                    top: 100%;
                    left: 0;
                    background: #333;
                    color: #fff;
                    padding: 8px;
                    border-radius: 5px;
                    font-size: 12px;
                    white-space: pre-wrap;
                    width: 200px;
                    height: auto;
                    z-index: 1000;
                    box-sizing: border-box;
                    overflow: hidden;
                `;
                tooltip.textContent = tooltipText;
                purchaseLink.appendChild(tooltip);
            });

            purchaseLink.addEventListener('mouseleave', function (event) {
                const tooltip = document.getElementById('purchaseLinkTooltip');
                if (tooltip) {
                    purchaseLink.removeChild(tooltip);
                }
            });

            purchaseLink.addEventListener('click', function (event) {
                event.preventDefault();
                GM_setValue("linkClicked", true);
                window.open("https://api.v3.cm/register?aff=25L7", "_blank");
            });
        }
    }

    function setupInfoIcon(countdownElement) {
        const infoIcon = countdownElement.querySelector("#infoIcon");
        infoIcon.addEventListener("mouseenter", function () {
            const tooltip = document.createElement('div');
            tooltip.id = 'tooltip';
            tooltip.style = `
                position: absolute;
                top: 25px;
                left: 0;
                background: #333;
                color: #fff;
                padding: 8px;
                border-radius: 5px;
                font-size: 12px;
                white-space: pre-wrap;
                width: 200px;
                height: auto;
                z-index: 1000;
                box-sizing: border-box;
                overflow: hidden;
            `;
            tooltip.innerText = `1.由于答题太快会导致被禁言,故倒计时不要太低。\n2.作者提供的API仅供测试,不要用于其他用途。\n3.被屏蔽的问题会显示下划线,未回答问题显示红色,回答过显示黑色。\n4.大部分课程回答50条,问30条问题即可满分。不要一天刷几百条。`;
            infoIcon.appendChild(tooltip);
        });

        infoIcon.addEventListener("mouseleave", function () {
            const tooltip = document.getElementById("tooltip");
            if (tooltip) tooltip.remove();
        });
    }

    function setupJumpButton(jumpButton) {
        jumpButton.addEventListener('click', () => {
            scrollToLastUnansweredQuestion();
        });
        setupButtonShadowEffects(jumpButton);
    }

    function setupButtonShadowEffects(button) {
        button.addEventListener('mousedown', () => {
            button.style.boxShadow = 'inset 2px 2px 4px #2f4686, inset -2px -2px 4px #507bcf';
        });
        button.addEventListener('mouseup', () => {
            button.style.boxShadow = '2px 2px 4px #2f4686, -2px -2px 4px #507bcf';
        });
        button.addEventListener('mouseleave', () => {
            button.style.boxShadow = '2px 2px 4px #2f4686, -2px -2px 4px #507bcf';
        });
    }

    function renderDropdownItems(dropdownOptions, allModels, defaultModels, dropdownSelected) {
        dropdownOptions.innerHTML = `
            ${allModels.map(model => `
                <div class="dropdown-item" data-model="${model}" style="
                    padding: 10px;
                    cursor: pointer;
                    ${defaultModels.includes(model) ? '' : 'display: flex; justify-content: space-between; align-items: center;'}
                    background: linear-gradient(145deg, #ffffff, #e6e6e6);
                    border-bottom: 1px solid #e0e0e0;
                ">
                    ${model}
                    ${defaultModels.includes(model) ? '' : `<span class="remove-model" data-model="${model}" style="cursor: pointer; color: red;">×</span>`}
                </div>
            `).join('')}
             <div class="add-model-container" style="padding: 10px; display: flex; align-items: center;">
                <input type="text" id="newModelInput" placeholder="输入模型名称" style="
                    padding: 8px;
                    border-radius: 5px;
                    border: none;
                    background: linear-gradient(145deg, #e6e6e6, #ffffff);
                    box-shadow: inset 2px 2px 4px #cccccc,
                                inset -2px -2px 4px #ffffff;
                    width: 70%;
                    box-sizing: border-box;
                    height: 32px;
                ">
                <button id="saveModelButton" style="
                    border: none;
                    border-radius: 8px;
                    background: linear-gradient(145deg, #4c88d6, #3b5998);
                    box-shadow: 2px 2px 4px #2f4686,
                                -2px -2px 4px #507bcf;
                    color: white;
                    cursor: pointer;
                    width: 25%;
                    text-align: center;
                    margin-left: 5px;
                    height: 32px;
                ">保存</button>
            </div>
        `;

        const newModelInput = dropdownOptions.querySelector('#newModelInput');
        const saveModelButton = dropdownOptions.querySelector('#saveModelButton');

        saveModelButton.addEventListener('click', () => {
            const newModel = newModelInput.value.trim();
            if (newModel && !getAllModels().includes(newModel)) {
                let customModels = GM_getValue("customModels", []);
                customModels.push(newModel);
                GM_setValue("customModels", customModels);
                const defaultModels = ["gemini-1.5-flash", "glm-4-flash"];
                let allModels = [...defaultModels, ...customModels];
                renderDropdownItems(dropdownOptions, allModels, defaultModels, dropdownSelected);
            }
        });

        dropdownOptions.addEventListener('click', (event) => {
            if (event.target.classList.contains('dropdown-item')) {
                const model = event.target.dataset.model;
                GM_setValue("aimodel", model);
                const defaultModels = ["gemini-1.5-flash", "glm-4-flash"];
                let customModels = GM_getValue("customModels", []);
                let allModels = [...defaultModels, ...customModels];
                let selectedModel = model;
                dropdownSelected.textContent = model;
                dropdownOptions.style.display = 'none';
            } else if (event.target.classList.contains('remove-model')) {
                const modelToRemove = event.target.dataset.model;
                let customModels = GM_getValue("customModels", []);
                customModels = customModels.filter(model => model !== modelToRemove);
                GM_setValue("customModels", customModels);
                const defaultModels = ["gemini-1.5-flash", "glm-4-flash"];
                let allModels = [...defaultModels, ...customModels];
                renderDropdownItems(dropdownOptions, allModels, defaultModels, dropdownSelected);
            }
        });
    }



    function getAllModels() {
        const defaultModels = ["gemini-1.5-flash", "glm-4-flash"];
        let customModels = GM_getValue("customModels", []);
        return [...defaultModels, ...customModels];
    }



    let scrolling = false;

    function scrollToLastUnansweredQuestion() {
        if (scrolling) {
            console.log("正在滚动,请稍后...");
            showAlert("正在滚动,请稍后...");
            return;
        }

        scrolling = true;

        let scrollContainer = document.querySelector('.el-scrollbar__wrap');
        if (!scrollContainer) {
            scrolling = false;
            return;
        }

        const questionItems = document.querySelectorAll('.question-item');
        const courseNameElement = document.querySelector(".course-name");
        const courseName = courseNameElement ? courseNameElement.innerText.trim() : '';
        const questionsData = GM_getValue("questionsData", {});

        function performScroll(targetElement) {
            const targetOffsetTop = targetElement.offsetTop;
            scrollContainer.scrollTo({
                top: targetOffsetTop,
                behavior: 'smooth'
            });

            const checkScrollInterval = setInterval(() => {
                const currentScrollTop = scrollContainer.scrollTop;
                const offsetDifference = Math.abs(currentScrollTop - targetOffsetTop);

                if (offsetDifference <= 5) {
                    console.log("滚动完成,位于目标位置!");
                    clearInterval(checkScrollInterval);
                    scrolling = false;
                } else {
                    console.log("滚动未完全到达目标位置,继续等待...");
                }
            }, 200);

            setTimeout(() => {
                clearInterval(checkScrollInterval);
                scrolling = false;
                console.log("滚动超时,停止检查。");
            }, 5000);
        }

        for (let i = 0; i < questionItems.length; i++) {
            const item = questionItems[i];
            const spanElement = item.querySelector('.question-content span');
            const questionText = spanElement ? spanElement.title.trim() : '';

            if (questionText && courseName && questionsData[courseName]) {
                const status = questionsData[courseName][questionText] || "未回答";

                const isExcluded = spanElement.style.textDecoration === 'line-through';

                if (status === "未回答" && !isExcluded) {
                    console.log(`找到第一个未回答且未排除的问题,索引: ${i},问题内容: ${questionText}`);
                    performScroll(item);
                    return;
                }
            }
        }


        console.log("没有找到未回答且未排除的问题,滚动到最底部...");

        scrollContainer.scrollTo({
            top: scrollContainer.scrollHeight,
            behavior: 'smooth'
        });

        const scrollEvent = new Event('scroll', { bubbles: true });
        scrollContainer.dispatchEvent(scrollEvent);

        let lastQuestionCount = questionItems.length;
        const checkNewQuestions = setInterval(() => {
            const newQuestionItems = document.querySelectorAll('.question-item');
            if (newQuestionItems.length > lastQuestionCount) {
                console.log("检测到新问题,重新查找未回答问题...");
                clearInterval(checkNewQuestions);
                scrolling = false;
                scrollToLastUnansweredQuestion();
            }
        }, 1000);

        setTimeout(() => {
            clearInterval(checkNewQuestions);
            scrolling = false;
            console.log("未检测到新问题,停止监控。");
            showAlert("未检测到新问题,请稍后重试。");
        }, 10000);
    }




    function showAlert(message) {

        console.log('Alert:', message);

        if (document.hidden) {
            return;
        }

        const stackId = 'alert-stack';
        let alertStack = document.getElementById(stackId);

        if (!alertStack) {
            alertStack = document.createElement('div');
            alertStack.id = stackId;
            Object.assign(alertStack.style, {
                position: 'fixed',
                bottom: '10px',
                left: '50%',
                transform: 'translateX(-50%)',
                zIndex: '9999999',
                display: 'flex',
                flexDirection: 'column-reverse',
                gap: '10px'
            });
            document.body.appendChild(alertStack);
        }

        const alertBox = document.createElement('div');
        Object.assign(alertBox.style, {
            backgroundColor: '#E0E5EC',
            padding: '20px',
            borderRadius: '12px',
            boxShadow: '10px 10px 15px #AEBEC7, -5px -5px 10px #FFFFFF',
            opacity: '0',
            transition: 'opacity 0.3s',
            position: 'relative',
            width: 'fit-content'
        });

        const alertText = document.createElement('div');
        Object.assign(alertText.style, {
            margin: '0',
            fontSize: '16px',
            fontWeight: 'bold',
            textAlign: 'center',
            textShadow: '2px 2px 3px rgba(0, 0, 0, 0.2)',
            color: '#4B5563'
        });
        alertText.innerText = message;

        alertBox.appendChild(alertText);
        alertStack.appendChild(alertBox);

        requestAnimationFrame(() => {
            alertBox.style.opacity = '1';
        });

        setTimeout(() => {
            alertBox.style.opacity = '0';
            setTimeout(() => {
                if (alertStack.contains(alertBox)) {
                    alertStack.removeChild(alertBox);
                }
                if (alertStack.childNodes.length === 0 && document.body.contains(alertStack)) {
                    document.body.removeChild(alertStack);
                }
            }, 300);
        }, 2500);
    }

    function checkForLatestVersion() {
        const lastChecked = GM_getValue("lastVersionCheck", 0);
        const currentTime = new Date().getTime();
        const oneHourMillis = 60 * 60 * 1000;
        let retryAttempts = 0;
        const maxRetries = 5;
        const retryDelay = 1000;

        function requestVersion() {
            GM_xmlhttpRequest({
                method: "GET",
                url: "https://update.greasyfork.dpdns.org/scripts/519662/知到智慧树全自动刷问答平时分助手.meta.js",
                onload: function (response) {
                    const match = response.responseText.match(/@version\s+([\d.]+)/);
                    if (match) {
                        const latestVersion = match[1];
                        GM_setValue("lastVersionCheck", currentTime);
                        GM_setValue("lastCheckedVersion", latestVersion);
                        document.getElementById("latestVersion").innerText = latestVersion;
                    } else {
                        if (retryAttempts < maxRetries) {
                            retryAttempts++;
                            setTimeout(requestVersion, retryDelay);
                        } else {
                            document.getElementById("latestVersion").innerText = "获取失败";
                        }
                    }
                }
            });
        }

        if (currentTime - lastChecked > oneHourMillis) {
            requestVersion();
        } else {
            const lastCheckedVersion = GM_getValue("lastCheckedVersion", "未检测");
            document.getElementById("latestVersion").innerText = lastCheckedVersion;
        }
    }


    function startCountdown(restartCount = null) {
        if (countdownInterval) {
            clearInterval(countdownInterval);
        }

        const minTime = GM_getValue("countdownMin", 20);
        const maxTime = GM_getValue("countdownMax", 30);
        let countdown = restartCount !== null
            ? restartCount
            : Math.floor(Math.random() * ((maxTime + 1) - minTime)) + minTime;
        GM_setValue("countdownTime", countdown);
        displayCountdown(countdown);

        countdownInterval = setInterval(() => {
            countdown--;
            displayCountdown(countdown);
            GM_setValue("currentCountdown", countdown);
            if (countdown <= 0) {
                clearInterval(countdownInterval);
                countdownInterval = null;
                clickTheUnansweredLink();
                console.log(`${GM_getValue("countdownTime", initialCountdownTime)}秒已过,点击动作触发。`);
                startCountdown();
            }
        }, 1000);
    }

    function clickTheUnansweredLink() {
        const questionItems = document.querySelectorAll('.question-item');
        const unansweredQuestions = Array.from(questionItems).filter(item => {
            const spanElement = item.querySelector('.question-content span');
            const questionText = spanElement ? spanElement.title.trim() : '';
            const courseNameElement = document.querySelector(".course-name");
            const courseName = courseNameElement ? courseNameElement.innerText.trim() : '';
            const status = GM_getValue("questionsData", {})[courseName]?.[questionText];
            return status === "未回答";
        });

        if (unansweredQuestions.length > 0) {
            const firstUnansweredQuestionSpan = unansweredQuestions[0].querySelector('.question-content span');

            if (firstUnansweredQuestionSpan) {
                firstUnansweredQuestionSpan.click();
                console.log(`已点击第一个未回答的问题: ${firstUnansweredQuestionSpan.title}`);
            } else {
                console.log("未找到未回答问题的链接元素");
            }
        } else {
            console.log("目前无未回答问题.");
        }
    }

    function checkHomeRegex() {
        const homePattern = /^https:\/\/qah5\.zhihuishu\.com\/qa\.html#\/web\/home/;
        const currentURL = window.location.href;

        if (homePattern.test(currentURL)) {
            console.log("正在获取问题列表...");

            const items = document.querySelectorAll(".question-item .question-content span");
            const courseNameElement = document.querySelector(".course-name");

            if (courseNameElement && items.length) {
                const courseName = courseNameElement.innerText.trim();
                let questionsData = GM_getValue("questionsData", {});

                if (!questionsData[courseName]) {
                    questionsData[courseName] = {};
                }

                items.forEach(item => {
                    const questionTitle = item.getAttribute("title").trim();
                    if (!questionsData[courseName][questionTitle]) {
                        questionsData[courseName][questionTitle] = "未回答";
                    }
                });

                GM_setValue("questionsData", questionsData);
                console.log("问题列表已更新:", questionsData);
            }
        }
    }

    function markQuestionAsAnswered(questionText) {
        const courseNameElement = document.querySelector(".course-name");
        if (courseNameElement) {
            const courseName = courseNameElement.innerText.trim();
            let questionsData = GM_getValue("questionsData", {});

            if (questionsData[courseName] && questionsData[courseName][questionText] === "未回答") {
                questionsData[courseName][questionText] = "已回答";
                GM_setValue("questionsData", questionsData);
                console.log(`问题“${questionText}”状态已更新为“已回答”。`);
            }
        }
    }

    function waitForAnswerButton() {

        const answerButton = document.querySelector("div.my-answer-btn.ZHIHUISHU_QZMD.tool-show span");

        if (answerButton && answerButton.textContent.trim() === "我来回答") {
            console.log("找到‘我来回答’按钮,点击...");
            answerButton.click();
            noAnswerButtonCount = 0;

            setTimeout(() => {
                const inputElement = document.querySelector("textarea");
                if (inputElement) {
                    console.log("文本框已出现,准备回答问题...");
                    startChecking();
                } else {
                    console.log("文本框未出现,继续检测‘我来回答’按钮...");
                    waitForAnswerButton();
                }
            }, intervalTime);
        } else {
            checkHomeRegex();

            console.log("未找到‘我来回答’按钮或按钮状态不符合条件,稍后重试...");
            noAnswerButtonCount++;

            const currentURL = window.location.href;
            const pattern = /^https:\/\/qah5\.zhihuishu\.com\/qa\.html#\/web\/questionDetail\/\d+/;

            if (noAnswerButtonCount > 1) {
                if (pattern.test(currentURL)) {
                    const questionElement = document.querySelector(".question-content span");
                    const questionText = questionElement ? questionElement.innerText.trim() : '';
                    if (questionText) {
                        markQuestionAsAnswered(questionText);
                    }
                    console.log("当前页面符合特定模式,关闭网页...");
                    showAlert("答题完毕,准备关闭网页")
                    window.close();
                } else {
                    console.log("当前页面不符合关闭条件,继续检测...");
                    noAnswerButtonCount = 0;
                }
            } else {
                setTimeout(waitForAnswerButton, intervalTime);
            }
        }
    }

    function addQuestionStatus() {
        const questionItems = document.querySelectorAll('.question-item');
        questionItems.forEach(item => {
            const spanElement = item.querySelector('.question-content span');
            const questionText = spanElement ? spanElement.title.trim() : '';
            if (questionText) {
                const courseNameElement = document.querySelector(".course-name");
                if (courseNameElement) {
                    const courseName = courseNameElement.innerText.trim();
                    let status = "未获取";
                    const questionsData = GM_getValue("questionsData", {});
                    if (questionsData[courseName] && questionsData[courseName][questionText]) {
                        status = questionsData[courseName][questionText];
                    }

                    const invalidPatterns = [
                        /什么是.+/, /.+是什么/, /.+是什么意思/, /.+的含义/, /指什么/, /是不是/, /对不对/, /好不好/, /好吗/, /近吗/, /远吗/, /对吗/,
                        /可以吗/, /可以.+吗/, /是否可以/, /有没有/, /能不能/, /应该.+吗/,
                        /是否应该/, /必须.+吗/, /是否必须/, /需要.+吗/, /是否需要/, /一定.+吗/,
                        /是否一定/, /是否合理/, /是否相同/, /是否不同/,
                        /有意义吗/, /意义大吗/, /喜欢.+吗/, /如何翻译.+/
                    ];

                    const isInvalidQuestion = invalidPatterns.some(pattern => pattern.test(questionText));

                    if (isInvalidQuestion) {
                        spanElement.style.textDecoration = 'line-through';
                        status = "已回答";
                        questionsData[courseName] = questionsData[courseName] || {};
                        questionsData[courseName][questionText] = status;
                        GM_setValue("questionsData", questionsData);
                    } else if (status === "未回答") {
                        spanElement.style.color = 'red';
                    } else if (status === "已回答") {
                        spanElement.style.color = 'green';
                    }
                }
            }
        });
    }


    function startChecking() {
        setInterval(() => {
            console.log("检查问题和输入框是否存在...");

            const questionElement = document.querySelector(".question-content span");
            if (!questionElement) {
                console.log("未找到问题元素。");
                return;
            }

            const questionText = questionElement.innerText.trim();
            if (!questionText) {
                console.log("问题内容为空。");
                return;
            }

            const inputElement = document.querySelector("textarea");
            if (!inputElement) {
                console.log("未找到输入框元素。");
                return;
            }

            addQuestionStatus();

            const answerButton = document.querySelector("div.my-answer-btn.ZHIHUISHU_QZMD.tool-show span");
            const modalExists = document.querySelector(".yidun_modal");
            if (modalExists) {
                console.log("检测到二维码,需要手动验证。10秒后关闭窗口。");
                GM_notification("检测到二维码,请手动验证。10秒后关闭窗口。", "安全验证");
                setTimeout(() => {
                    window.close();
                }, 10000);
                return;
            }

            if (!answerButton || answerButton.textContent.trim() !== "我来回答") {
                const currentURL = window.location.href;
                const pattern = /^https:\/\/qah5\.zhihuishu\.com\/qa\.html#\/web\/questionDetail/;
                if (pattern.test(currentURL)) {
                    console.log("未找到'我来回答'按钮且当前页面符合特定模式,关闭当前窗口。");
                    markQuestionAsAnswered(questionText);
                    window.close();
                } else {
                    console.log("未找到'我来回答'按钮且当前页面不符合特定模式,停止回答逻辑。");
                }
                return;
            }

            if (inputElement.value.trim() !== "") {
                console.log("输入框中已有数据,检查是否需要点击立即发布...");
                startPublishChecking();
                return;
            }

            console.log("找到问题并准备回答:", questionText);

            if (!requestLock) {
                requestLock = true;
                answerQuestion(questionText, inputElement);
                showAlert("正在回答问题")
            } else {
                console.log("请求进行中,跳过此次回答逻辑。");
                showAlert("正在回答问题");
            }


        }, intervalTime);
    }
    function checkAndAddGenerateQuestionButton() {
        const dialog = document.querySelector('div[role="dialog"][aria-modal="true"][aria-label="提问"].el-dialog');
        if (dialog) {
            const publishButton = dialog.querySelector('.dialog-bottom .up-btn');
            if (publishButton) {
                if (!dialog.querySelector('#generateQuestionButton')) {
                    const generateQuestionButton = document.createElement('div');
                    generateQuestionButton.id = 'generateQuestionButton';
                    generateQuestionButton.textContent = '生成问题';

                    generateQuestionButton.style.cssText = `
                        display: inline-block;
                        margin-right: 40px; /* 添加 40px 的右侧外边距 */
                        padding: 10px 20px;
                        border: none;
                        border-radius: 10px;
                        background: linear-gradient(145deg, #4c88d6, #3b5998);
                        box-shadow: 2px 2px 4px #2f4686, -2px -2px 4px #507bcf;
                        color: white;
                        font-weight: bold;
                        cursor: pointer;
                    `;

                    generateQuestionButton.addEventListener('click', () => {
                        generateQuestion(dialog);
                    });


                    generateQuestionButton.addEventListener('mousedown', () => {
                        generateQuestionButton.style.boxShadow = 'inset 2px 2px 4px #2f4686, inset -2px -2px 4px #507bcf';
                    });
                    generateQuestionButton.addEventListener('mouseup', () => {
                        generateQuestionButton.style.boxShadow = '2px 2px 4px #2f4686, -2px -2px 4px #507bcf';
                    });
                    generateQuestionButton.addEventListener('mouseleave', () => {
                        generateQuestionButton.style.boxShadow = '2px 2px 4px #2f4686, -2px -2px 4px #507bcf';
                    });


                    const dialogBottom = publishButton.parentNode;
                    dialogBottom.style.display = 'flex';
                    dialogBottom.style.justifyContent = 'flex-end';


                    dialogBottom.insertBefore(generateQuestionButton, publishButton);
                }
            }
        }
    }



    const observer = new MutationObserver(checkAndAddGenerateQuestionButton);


    observer.observe(document.body, {
        childList: true,
        subtree: true,
        attributes: false,
        characterData: false
    });
    function generateQuestion(dialog) {
        const courseNameElement = document.querySelector(".course-name");
        const courseName = courseNameElement ? courseNameElement.innerText.trim() : '当前课程';

        const inputElement = dialog.querySelector('textarea');

        if (!inputElement) {
            showAlert('未找到提问输入框!');
            return;
        }

        const questionElements = document.querySelectorAll(".question-content.ZHIHUISHU_QZMD span");
        const recentQuestions = Array.from(questionElements)
            .map((element, index) => `${index + 1}. ${element.textContent.trim()}`)
            .slice(0, 20)
            .join("\n");

        const prompt = `针对${courseName},请生成一个与以下30个问题**完全不同**的开放式问题,字数限制在50字以内。务必不同句式

                                    **以下是最近别人发送的1o个问题,请**务必避免句式重复或者句式相似**:
                                    ${recentQuestions} \n不要一直用"如何怎么样"的句式,你需要随便地提问,只要关于这个主题就行,发散思维。只回复你想到的问题即可,只需要回复一个问题即可,不要换行。`;

        console.log(prompt);

        showAlert(`正在生成问题...`)
        GM_xmlhttpRequest({
            method: "POST",
            url: GM_getValue("API_URL", defaultAPIUrl),
            headers: {
                "Content-Type": "application/json",
                "Authorization": `Bearer ${GM_getValue("API_KEY", defaultAPIKey)}`
            },
            data: JSON.stringify({
                model: "gemini-1.5-flash",
                messages: [
                    {
                        role: "user",
                        content: prompt
                    }
                ],
                max_tokens: 10240,
                temperature: 1,
                top_p: 1,
                stream: false
            }),
            onload: function (response) {
                console.log("请求成功:", response.responseText);
                if (response.status === 200) {
                    const jsonResponse = JSON.parse(response.responseText);
                    const generatedQuestion = jsonResponse.choices[0].message.content.trim();

                    inputElement.value = generatedQuestion;
                    inputElement.dispatchEvent(new Event('input', { 'bubbles': true }));

                    showAlert('问题已生成并填充到提问框!');
                } else {
                    console.error("API 请求失败,状态码:", response.status);
                    showAlert("问题生成失败,请检查API设置或稍后重试。");
                }
            },
            onerror: function () {
                console.error("API 请求出错。");
                showAlert("问题生成出错,请检查网络连接。");
            }
        });
    }


    function startPublishChecking() {
        console.log("检查是否需要点击立即发布按钮...");
        const publishButton = document.querySelector('.dialog-bottom .up-btn.ZHIHUISHU_QZMD.set-btn');

        const inputElement = document.querySelector('textarea');
        if (
            inputElement &&
            inputElement.value.trim() !== "" &&
            publishButton &&
            publishButton.textContent.trim() === "立即发布"
        ) {
            console.log("找到立即发布按钮,自动点击...");
            publishButton.click();
        }
    }

    function answerQuestion(questionText, inputElement) {
        GM_xmlhttpRequest({
            method: "POST",
            url: GM_getValue("API_URL", defaultAPIUrl),
            headers: {
                "Content-Type": "application/json",
                "Authorization": `Bearer ${GM_getValue("API_KEY", defaultAPIKey)}`
            },
            data: JSON.stringify({
                model: GM_getValue("aimodel", defaultModel),
                messages: [
                    {
                        role: "user",
                        content: `智慧树问题回答,你比如我提问:音乐剧与民歌在曲风、表现形式和情感表达上有何不同?如何在声乐演唱中体现这些差异?你只需要给出简介30字以内或者左右的回答即可。只需要输出回答,不需要其他,也不需要markdown格式,回答中严禁出现下面三个词连在一起:“学习通”。且回答至少超过10个字。问题如下:${questionText}`
                    }
                ],
                max_tokens: 1688,
                temperature: 0.5,
                stream: false
            }),
            onload: function (response) {
                requestLock = false;
                console.log("请求成功:", response.responseText);
                if (response.status === 200) {
                    const jsonResponse = JSON.parse(response.responseText);
                    const answer = jsonResponse.choices[0].message.content.trim();
                    console.log("获取到的回答:", answer);

                    inputElement.value = answer;
                    inputElement.dispatchEvent(inputEvent);
                    showAlert("成功填入问题答案");
                } else {
                    console.error("API 请求失败,状态码:", response.status);
                    showAlert("申请答题链接失败,请刷新网页或稍后再试。")
                }
            },
            onerror: function () {
                requestLock = false;
                console.error("API 请求出错。");
            }
        });
    }

    window.onload = () => {
        console.log("启动智慧树自动回答器...");
        waitForAnswerButton();
        initMutationObserver();
        startCountdown();
    };

})();

QingJ © 2025

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