// ==UserScript== // @name 💟国家中小学智慧教育平台全能助手|免登录查看|PDF教材下载💯|支持高清阅读✨|其它超多功能+ // @namespace www.zxx.edu.cn/ // @version 1.0 // @description 免登录查看电子课本、与课外书籍、下载PDF,支持批量下载,支持新课标教材。免登录情况下替换默认阅读器、实现高清阅读。 // @match *://basic.smartedu.cn/* // @match *://www.zxx.edu.cn/* // @match *.ykt.cbern.com.cn/* // @icon  // @grant none // ==/UserScript== (function() { 'use strict'; if (window.location.href.includes("-private") && window.location.href.includes("=download")) { window.location.replace(window.location.href.replace("-private", "")); } const toolbarViewerRight = document.querySelector("#toolbarViewerRight"); if (toolbarViewerRight && document.referrer.includes("basic.smartedu.cn")) { // 仅限 basic.smartedu.cn,排除全屏授课页面 const hiddenElements = toolbarViewerRight.querySelectorAll("[hidden][data-l10n-id='download']"); hiddenElements.forEach(function(element) { element.removeAttribute("hidden"); }); } // 样式 const style = document.createElement("style"); style.innerHTML = ` /* iframe 全屏按钮样式 */ .fullscreenMode:before { -webkit-mask-image: url('data:image/svg+xml;utf8,'); mask-image: url('data:image/svg+xml;utf8,'); } /* 视频去水印 */ .vjs-watermark { display: none !important; } `; document.head.append(style); function fullscreenSwitch() { const courseDocument = document.querySelector(".course-document"); const html = document.querySelector("html"); if (courseDocument.classList.contains("full-screen")) { // 当前处于"全屏状态",切换为"非全屏状态" courseDocument.classList.remove("full-screen"); html.style.overflow = ""; } else { // 当前处于"非全屏状态",切换为"全屏状态" courseDocument.classList.add("full-screen"); html.style.overflow = "hidden"; } } // iframe 向主页面发送全屏切换消息 const iframeDownloadBtn = document.querySelector("#download"); if (iframeDownloadBtn && !iframeDownloadBtn.hidden) { const fullscreenBtn = document.createElement("button"); fullscreenBtn.className = "toolbarButton fullscreenMode"; fullscreenBtn.title = "切换全屏模式"; fullscreenBtn.innerHTML = `全屏模式`; iframeDownloadBtn.parentNode.insertBefore(fullscreenBtn, iframeDownloadBtn.nextSibling); fullscreenBtn.addEventListener("click", () => { console.log("点击全屏按钮"); window.parent.postMessage("full-screen mode switches"); // iframe 向主页面发送消息 }); } // 主页面等待全屏指令 window.addEventListener("message", function(event) { if (event.data === "full-screen mode switches") { console.log("Received message from iframe:", event.data); fullscreenSwitch(); } }, false); function AddBtnsToListPg(i, title, pdfUrl) { const container = document.querySelectorAll("li.index-module_item_GfOnF")[i]; container?.querySelector(".PDF-btns")?.remove(); // 去除旧按钮 const PDF_btns = document.createElement("div"); // 创建 PDF_btns PDF_btns.setAttribute("class", "PDF-btns"); PDF_btns.innerHTML = ` 查看PDF 下载PDF `; container.appendChild(PDF_btns); // 将 PDF_btns 添加到父元素中 document.querySelectorAll(".index-module_content_KmLzG")[i].style.width = "550px"; // 统一 content 宽度 PDF_btns.addEventListener("click", function(event) { // 停止 PDF_btns 的点击事件向上一层元素传播 event.stopPropagation(); }); const element2 = PDF_btns.querySelector("a.fish-btn.fish-btn-primary"); element2.addEventListener("click", function(event) { // 点击下载 var fileUrl = pdfUrl; var xhr = new XMLHttpRequest(); xhr.open("GET", fileUrl); xhr.responseType = "blob"; xhr.onloadstart = function() { // 初始化进度 element2.innerText = "下载中 ..."; }; xhr.onprogress = function(event) { if (event.lengthComputable) { var progress = Math.round((event.loaded / event.total) * 100); element2.innerText = "下载中 (" + progress + "%)"; } }; xhr.onload = function() { if (xhr.status === 200) { var blob = xhr.response; var downloadLink = document.createElement("a"); downloadLink.href = URL.createObjectURL(blob); downloadLink.download = title || document.querySelectorAll("div.index-module_line_LgJAC")[i].querySelector("span").getAttribute("title"); // 文件名 downloadLink.dispatchEvent(new MouseEvent("click")); element2.innerText = "下载PDF"; // 下载完成后恢复按钮文字 } }; xhr.send(); // 发送请求 }); } let title, pdfUrl; // 内页 pdfURL let pdfUrls = []; // 声明空数组 // Resources requested via XHR const originalOpen = XMLHttpRequest.prototype.open; XMLHttpRequest.prototype.open = function(method, url) { // console.log("Intercepted URL:", url); if (window.location.pathname === "/tchMaterial" && url.includes("query?res_ids=")) { // 获取列表页各 id const ids = url.match(/res_ids=([^&]+)/)[1].split(","); // 多个 id 构成的数组 // 匹配出 url 中“query?res_ids=”之后的内容,并将匹配出来的内容以其中的“,”隔开,形成多个“id” console.log(ids); const baseUrl = "https://s-file-1.ykt.cbern.com.cn/zxx/ndrv2/resources/tch_material/details/"; const jsonUrls = ids.map(id => `${baseUrl}${id}.json`); // 多个 jsonURL 构成的数组 console.log(jsonUrls); const baseUrlThe2nd = "https://s-file-1.ykt.cbern.com.cn/zxx/ndrs/special_edu/thematic_course/"; const jsonUrlsThe2nd = ids.map(id => `${baseUrlThe2nd}${id}/resources/list.json`); // 多个 jsonURL 构成的数组 console.log(jsonUrlsThe2nd); pdfUrls.length = 0; // 清空 pdfUrls 数组 for (let i = 0; i < jsonUrls.length; i++) { fetch(jsonUrls[i]) .then(response => { return response.text(); }) .then(text => { let title = text.match(/(?<="title":\s*")[^"]+/g)[0]; let pdfUrl = text.match(/\bhttps?:\/\/[^"]*pkg[^"]*\.pdf\b/g).slice(-1)[0].replace("-private", ""); // 不存在则报错 pdfUrls.splice(i, 0, pdfUrl); console.log(`pdf${i + 1}:`, title, pdfUrl); AddBtnsToListPg(i, title, pdfUrl); }) .catch(error => { // console.log(error); fetch(jsonUrlsThe2nd[i]) .then(response => { return response.text(); }) .then(text => { let pdfUrl = text.match(/\bhttps?:\/\/[^"]*pkg[^"]*\.pdf\b/g).slice(-1)[0].replace("-private", ""); // 不存在则报错 pdfUrls.splice(i, 0, pdfUrl); console.log(`pdf${i + 1}:`, pdfUrl); AddBtnsToListPg(i, undefined, pdfUrl); }) .catch(error => { console.log(error); let pdfUrl = `notfound`; pdfUrls.splice(i, 0, pdfUrl); console.log(`pdf${i + 1}:`, pdfUrl); }); }); } console.log("pdfURLs:", pdfUrls); } else if (window.location.pathname === "/tchMaterial/detail" && (/resources\/tch_material\/details|special_edu\/thematic_course.*list\.json/).test(url)) { // 内页 pdfUrl const originalOnReadyStateChange = this.onreadystatechange; this.onreadystatechange = () => { if (this.readyState === 4 && this.status === 200) { try { const text = this.responseText; if (url.includes("resources/tch_material/details")) { title = text.match(/(?<="title":\s*")[^"]+/g)[0]; pdfUrl = text.match(/\bhttps?:\/\/[^"]*pkg[^"]*\.pdf\b/g).slice(-1)[0].replace("-private", ""); // 不存在则报错 console.log(title, pdfUrl); } else if (url.includes("special_edu/thematic_course") && url.endsWith("list.json")) { const urlList = text.match(/\bhttps?:\/\/(?!r1|r2)[^"]*pkg[^"]*\.(pdf|pptx?)\b/g); // 必须包含 pkg 以排除无效链接 console.log(urlList); const errorText = document.querySelector(".index-module_error_IC0KJ > .index-module_text_1noC9"); if (errorText) { errorText.innerHTML = "哎呀,该内容不需要登录也能查看"; const div = document.createElement("div"); div.className = "urls"; div.style = "display: flex;flex-direction: column"; errorText.insertAdjacentElement("afterend", div); for (let i = 0; i < urlList.length; i++) { urlList[i] = urlList[i].replace("-private", ""); div.innerHTML += `${urlList[i]}`; } } } } catch (e) { console.error(e); } } if (originalOnReadyStateChange) { originalOnReadyStateChange.apply(this, arguments); } }; } else if (url.endsWith(".mp3")) { // MP3 链接 let modifiedURL = url.replace("-private", ""); console.log("Requested MP3 (modified URL):", modifiedURL); const element1 = document.querySelector("a.mp3"); // 按钮 element1 if (element1) { element1.href = modifiedURL; // 修改 element1 的 URL } const suggestBtn = document.getElementsByClassName("index-module_suggestion-wrap_s+Ii+")[0]; if (suggestBtn && !document.querySelector(".mp3")) { const div = document.createElement("div"); div.setAttribute("style", "display: flex; align-items: center; margin-left: 30px"); div.innerHTML = `📼 查看MP3`; suggestBtn.insertAdjacentElement("afterend", div); } } return originalOpen.apply(this, arguments); }; const maxTimeToCheck = 15000; // 最多检查 15000 毫秒,即 15 秒 let elapsedTime = 0; // 免登录内页 function checkLoginModal() { const loginModal = document.querySelector(".fish-modal-root"); if (loginModal) { clearInterval(intervalId1); // 清除定时器 // 去除阻碍 loginModal.remove(); // 去除遮罩 const body = document.querySelector("html > body"); // 去除 body 属性 body.removeAttribute("class"); body.removeAttribute("style"); // 添加阅读器 const indexModuleWrapperECeCo = document.querySelector(".index-module_wrapper_ECeCo"); if (indexModuleWrapperECeCo) { indexModuleWrapperECeCo.innerHTML = `

${title}

智慧中小学
5.0分
100万+
建议
`; } // 全屏切换 const fullscreenBtn = document.querySelector(".tool-btn.fullscreen"); fullscreenBtn.addEventListener("click", fullscreenSwitch); // 重定向至登录页 const toLogin = [ document.querySelector(".fish-dropdown-trigger.index-module_assessment-btn_6imdF"), document.querySelector(".index-module_like-wrap_NbyLe"), document.querySelector(".index-module_suggestion-wrap_s\\+Ii\\+") ]; toLogin.forEach(element => { element.addEventListener("click", () => { window.open("https://auth.smartedu.cn/uias/login/"); }); }); } else { elapsedTime += 100; if (elapsedTime >= maxTimeToCheck) { clearInterval(intervalId1); // 清除定时器,停止检查 } } } const intervalId1 = setInterval(checkLoginModal, 100); // 每隔 100 毫秒检查一次 .fish-modal-root 元素是否存在 // 内页 function checkIndexModule() { const container = document.querySelector(".index-module_extra_tUQog"); // 找到要添加按钮的容器元素 if (container && !document.querySelector(".Btns")) { clearInterval(intervalId2); // 清除定时器 const div = document.createElement("div"); div.className = "Btns"; div.innerHTML = `📓 查看PDF📓 下载PDF`; container.appendChild(div); // 将按钮添加到网页中 const element1 = div.querySelector("a.link"); if (element1) { element1.addEventListener("mouseover", function() { this.innerHTML = "📘 查看PDF"; this.style.color = "#1e62ec"; // 鼠标移入时修改元素的样式 }); element1.addEventListener("mouseout", function() { this.innerHTML = "📓 查看PDF"; this.style.color = "#888"; // 鼠标移出时恢复原来的样式 }); } const element2 = div.querySelector("a.download"); if (element2) { element2.addEventListener("click", function(event) { // 点击下载 var fileUrl = pdfUrl; var xhr = new XMLHttpRequest(); xhr.open("GET", fileUrl); xhr.responseType = "blob"; xhr.onloadstart = function() { // 初始化进度 element2.innerText = "📓 下载中 ..."; }; xhr.onprogress = function(event) { if (event.lengthComputable) { var progress = Math.round((event.loaded / event.total) * 100); element2.innerText = "📓 下载中 (" + progress + "%)"; } }; xhr.onload = function() { if (xhr.status === 200) { var blob = xhr.response; var downloadLink = document.createElement("a"); downloadLink.href = URL.createObjectURL(blob); downloadLink.download = title || document.querySelector(".index-module_title_bnE9V").innerText; // 文件名 downloadLink.dispatchEvent(new MouseEvent("click")); element2.innerText = "📓 下载PDF"; // 下载完成后恢复按钮文字 } }; xhr.send(); // 发送请求 }); } setTimeout(function() { // 等待 iframe 加载;教材内页主文档通过 iframe 获取 URL 以更新链接 const iframe = document.querySelector("iframe"); if (iframe) { const iframeSrc = iframe.src; const iframeSrcExtracted = iframeSrc.match(/file=([^&#]+)/)?.[1]?.replace("-private", ""); console.log("iframeSrcExtracted:", iframeSrcExtracted); const element1 = div.querySelector("a.link"); // 按钮 element1 if (element1) { element1.href = iframeSrcExtracted; // 更新 element1 的 URL } pdfUrl = iframeSrcExtracted; // 更新 element2 的 URL } }, 3000); } else { elapsedTime += 100; if (elapsedTime >= maxTimeToCheck) { clearInterval(intervalId2); // 清除定时器,停止检查 } } } const intervalId2 = setInterval(checkIndexModule, 100); // 每隔 100 毫秒检查一次 index-module_extra_tUQog 元素是否存在 // PDF fetched const originalFetch = window.fetch; window.fetch = function (url, options) { return originalFetch(url, options) // 调用原始 fetch 函数 .then(response => { if (response.status === 200 && url.endsWith(".pdf")) { let modifiedURL = url.replace("-private", ""); console.log("Fetched PDF (modified valid URL):", modifiedURL); const element1 = document.querySelector("a.link"); // 按钮 element1 if (element1) { // iframe 内部不存在此元素;可在主文档中通过 iframe 获取文件 URL element1.href = modifiedURL; // 修改 element1 的 URL } pdfUrl = modifiedURL; // 修改 element2 的 URL const syncClassSuggestion = document.getElementsByClassName("index-module_suggestion-wrap_s+Ii+")[0]; if (syncClassSuggestion && !document.querySelector("div > .link")) { const div = document.createElement("div"); div.setAttribute("style", "display: flex; align-items: center; margin-left: 30px"); div.innerHTML = `📓 查看PDF`; syncClassSuggestion.insertAdjacentElement("afterend", div); } } return response; }) .catch(error => { console.error(error); }); }; })();