您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
通过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或关注我们的公众号极客氢云获取最新地址