Greasy Fork

Greasy Fork is available in English.

国家中小学智慧教育平台教材PDF电子课本链接与下载工具

教材列表页与预览页添加了PDF按钮,免登录查看或下载电子课本与课外书籍,可批量下载,支持新课标教材

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name         国家中小学智慧教育平台教材PDF电子课本链接与下载工具
// @namespace    http://greasyfork.icu/zh-CN/scripts/466598
// @version      3.1.1
// @description  教材列表页与预览页添加了PDF按钮,免登录查看或下载电子课本与课外书籍,可批量下载,支持新课标教材
// @match        *://basic.smartedu.cn/*
// @match        *://www.zxx.edu.cn/*
// @match        *.ykt.cbern.com.cn/*
// @match        *://x-edu-pdfjs.ykt.eduyun.cn/*
// @icon         
// @grant        GM_getValue
// @grant        GM_setValue
// ==/UserScript==

(function() {
    'use strict';

    const idFromMac = window.location.href.match(/id=%5C%22(.*?)%5C%22,/)?.[1];
    if (idFromMac && idFromMac.length >= 50) {
        GM_setValue("macId", idFromMac);
        console.log("idFromMac:", idFromMac);
    }
    const macId = GM_getValue("macId", "0");
    console.log("macId:", macId);

    if (window.location.pathname === "/pdfjs/2.15/web/viewer.html" && macId.length < 50) {
        alert("登录账号并以网站原有方式查看教材后,重新打开网站即可使用脚本。");
    }

    const errorMessage = document.querySelector("#errorMessage");
    if (errorMessage) {
        const newElement = document.createElement("span");
        newElement.textContent = `登录账号并以网站原有方式查看教材后,重新打开网站或浏览器即可使用脚本功能。`;
        errorMessage.insertAdjacentElement("afterend", newElement);
    }

    const toolbarViewerRight = document.querySelector("#toolbarViewerRight");
    if (toolbarViewerRight) {
        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,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16"><g><path d="M13,13h-2.7v2H14c0.6,0,1-0.4,1-1v-3.7h-2L13,13z"/><path d="M3,10.3H1V14c0,0.6,0.4,1,1,1h3.7v-2H3V10.3z"/><path d="M1,2v3.7h2V3h2.7V1H2C1.4,1,1,1.4,1,2z"/><path d="M14,1h-3.7v2H13v2.7h2V2C15,1.4,14.6,1,14,1z"/></g></svg>');
            mask-image: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16"><g><path d="M13,13h-2.7v2H14c0.6,0,1-0.4,1-1v-3.7h-2L13,13z"/><path d="M3,10.3H1V14c0,0.6,0.4,1,1,1h3.7v-2H3V10.3z"/><path d="M1,2v3.7h2V3h2.7V1H2C1.4,1,1,1.4,1,2z"/><path d="M14,1h-3.7v2H13v2.7h2V2C15,1.4,14.6,1,14,1z"/></g></svg>');
        }
        /* 视频去水印 */
        .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 = "";
            document.querySelector("iframe")?.scrollIntoView({
                block: "nearest"
            });
        } else { // 当前处于"非全屏状态",切换为"全屏状态"
            courseDocument.classList.add("full-screen");
            html.style.overflow = "hidden";
        }
    }

    // iframe 向主页面发送全屏切换消息
    if (window.self !== window.top) {
        const verticalToolbarSeparator = document.querySelector("#toolbarViewerRight > .verticalToolbarSeparator");
        if (verticalToolbarSeparator) {
            const fullscreenBtn = document.createElement("button");
            fullscreenBtn.className = "toolbarButton fullscreenMode";
            fullscreenBtn.title = "切换全屏模式";
            fullscreenBtn.innerHTML = `<span>全屏模式</span>`;
            verticalToolbarSeparator.insertAdjacentElement("beforebegin", fullscreenBtn);
            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();
        }
    });

    async function downloadFile(fileUrl, element2, progressText, fileName, completeText) {
        try {
            element2.innerText = `${progressText} ...`;
            const response = await fetch(fileUrl, {
                headers: {
                    "x-nd-auth": `MAC id=\"${macId}\",nonce=\"0\",mac=\"0\"`
                },
                // cache: "no-store"
            });
            if (!response.ok) {
                throw new Error("请求失败: " + response.status);
            }
            const contentLength = response.headers.get("Content-Length");
            const total = contentLength ? parseInt(contentLength, 10) : 0;
            const reader = response.body.getReader();
            let received = 0;
            const chunks = [];
            while (true) {
                const { done, value } = await reader.read();
                if (done) break;
                chunks.push(value);
                received += value.length;
                if (total) {
                    const progress = Math.round((received / total) * 100);
                    element2.innerText = `${progressText} (${progress}%)`;
                } else {
                    element2.innerText = `${progressText} (${(received / 1024 / 1024).toFixed(2)} MB)`;
                }
            }
            const blob = new Blob(chunks);
            const downloadLink = document.createElement("a");
            downloadLink.href = URL.createObjectURL(blob);
            downloadLink.download = fileName;
            downloadLink.click();
            element2.innerText = completeText;
        } catch (err) {
            console.error(err);
            element2.innerText = "下载失败";
        }
    }

    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 = `
            <a type="button" class="fish-btn" style="margin-left: 24px; margin-bottom: 5px;" href="${pdfUrl}?accessToken=${macId}" target="_blank">查看PDF</a>
            <a type="button" class="fish-btn fish-btn-primary" style="margin-left: 24px; margin-bottom: 5px;">下载PDF</a>
        `;
        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");
        const progressText = "下载中";
        const fileName = (title || document.querySelectorAll("div.index-module_line_LgJAC")[i].querySelector("span").getAttribute("title") || "PDF") + ".pdf";
        const completeText = "下载完成";
        element2.addEventListener("click", function() {
            downloadFile(pdfUrl, element2, progressText, fileName, completeText);
        });
    }

    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 jsonUrls = ids.map(id => `https://s-file-1.ykt.cbern.com.cn/zxx/ndrv2/resources/tch_material/details/${id}.json`); // 多个 jsonURL 构成的数组
            console.log(jsonUrls);
            const jsonUrlsThe2nd = ids.map(id => `https://s-file-1.ykt.cbern.com.cn/zxx/ndrs/special_edu/thematic_course/${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]; // 不存在则报错
                        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]; // 不存在则报错
                                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]; // 不存在则报错
                            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("._error_18kvb_1.index-module_error_310Am > ._text_18kvb_16");
                            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++) {
                                    if (urlList[i].endsWith("pdf")) {
                                        div.innerHTML += `<a href="${urlList[i]}?accessToken=${macId}" target="_blank" class="fish-btn">${urlList[i]}</a>`;
                                    } else {
                                        div.innerHTML += `<a href="${urlList[i]}" target="_blank" class="fish-btn">${urlList[i]}</a>`;
                                    }
                                }
                            }
                        }
                    } catch (e) {
                        console.error(e);
                    }
                }
                if (originalOnReadyStateChange) {
                    originalOnReadyStateChange.apply(this, arguments);
                }
            };
        } else if (url.endsWith(".mp3") || url.includes(".mp3?")) { // MP3 链接
            console.log("Requested MP3:", url);
            const element1 = document.querySelector("a.mp3"); // 按钮 element1
            if (element1) {
                element1.parentElement.remove();
            }
            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 = `<a class="mp3" style="color: #888;">📼 下载MP3</a>`;
                suggestBtn.insertAdjacentElement("afterend", div);
                if (!document.querySelector(".index-module_wrapper_ECeCo > .imageList-module_special-edu-image_A7C2c")) { // 非常规界面空间不够
                    suggestBtn.previousElementSibling?.remove();
                }
                const element1 = div.querySelector("a.mp3");
                element1.addEventListener("click", function(event) { // 点击下载
                    var fileUrl = url;
                    var xhr = new XMLHttpRequest();
                    xhr.open("GET", fileUrl);
                    xhr.setRequestHeader("x-nd-auth", `MAC id=\"${macId}\",nonce=\"0\",mac=\"0\"`);
                    xhr.responseType = "blob";
                    xhr.onloadstart = function() { // 初始化进度
                        element1.innerText = "📼 下载中 ...";
                    };
                    xhr.onprogress = function(event) {
                        if (event.lengthComputable) {
                            var progress = Math.round((event.loaded / event.total) * 100);
                            element1.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 = document.querySelector(".audioList-module_audio-item_GGkA9.audioList-module_audio-item-active_Xx7f- .audioList-module_center_MjbID").innerText || "MP3"; // 文件名
                            downloadLink.dispatchEvent(new MouseEvent("click"));
                            element1.innerText = "📼 下载MP3"; // 下载完成后恢复按钮文字
                        } else {
                            console.error("请求失败:", xhr.status);
                            element1.innerText = "下载失败";
                        }
                    };
                    xhr.onerror = function() {
                        console.error("请求出错");
                        element1.innerText = "下载出错";
                    };
                    xhr.send(); // 发送请求
                });
            }
        }
        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 = `
<div class="index-module_wrapper_ECeCo">
<div class="imageList-module_special-edu-image_A7C2c">
  <div class="index-module_header_tG-zz">
    <h3 class="index-module_title_bnE9V">${title}</h3>
    <div class="index-module_info_evO1d">
      <span class="index-module_origin_nuihE">
        <svg class="index-module_icon_dwVZ4">
          <use xlink:href="#icon_hotel_fill">
          </use>
        </svg>
        <span class="index-module_department_ewVZW">智慧中小学</span>
      </span>
      <div class="index-module_assesment-detail_jhGLz">
        <div class="index-module_assessment-content_06v6Z">
          <div class="fish-dropdown-trigger index-module_assessment-rate_euPTQ">
            <ul class="fish-rate fish-rate-disabled" tabindex="-1" role="radiogroup">
              <li class="fish-rate-star fish-rate-star-full">
                <div role="radio" aria-checked="true" aria-posinset="1" aria-setsize="5" tabindex="-1">
                  <div class="fish-rate-star-first">
                    <div class="custom-star-wrap" style="width: 100%;">
                      <div class="custom-star">
                      </div>
                    </div>
                  </div>
                  <div class="fish-rate-star-second">
                    <div class="custom-star-wrap" style="width: 100%;">
                      <div class="custom-star">
                      </div>
                    </div>
                  </div>
                </div>
              </li>
              <li class="fish-rate-star fish-rate-star-full">
                <div role="radio" aria-checked="true" aria-posinset="2" aria-setsize="5" tabindex="-1">
                  <div class="fish-rate-star-first">
                    <div class="custom-star-wrap" style="width: 100%;">
                      <div class="custom-star">
                      </div>
                    </div>
                  </div>
                  <div class="fish-rate-star-second">
                    <div class="custom-star-wrap" style="width: 100%;">
                      <div class="custom-star">
                      </div>
                    </div>
                  </div>
                </div>
              </li>
              <li class="fish-rate-star fish-rate-star-full">
                <div role="radio" aria-checked="true" aria-posinset="3" aria-setsize="5" tabindex="-1">
                  <div class="fish-rate-star-first">
                    <div class="custom-star-wrap" style="width: 100%;">
                      <div class="custom-star">
                      </div>
                    </div>
                  </div>
                  <div class="fish-rate-star-second">
                    <div class="custom-star-wrap" style="width: 100%;">
                      <div class="custom-star">
                      </div>
                    </div>
                  </div>
                </div>
              </li>
              <li class="fish-rate-star fish-rate-star-full">
                <div role="radio" aria-checked="true" aria-posinset="4" aria-setsize="5" tabindex="-1">
                  <div class="fish-rate-star-first">
                    <div class="custom-star-wrap" style="width: 100%;">
                      <div class="custom-star">
                      </div>
                    </div>
                  </div>
                  <div class="fish-rate-star-second">
                    <div class="custom-star-wrap" style="width: 100%;">
                      <div class="custom-star">
                      </div>
                    </div>
                  </div>
                </div>
              </li>
              <li class="fish-rate-star fish-rate-star-zero">
                <div role="radio" aria-checked="true" aria-posinset="5" aria-setsize="5" tabindex="-1">
                  <div class="fish-rate-star-first">
                    <div class="custom-star-wrap" style="width: 100%;">
                      <div class="custom-star">
                      </div>
                    </div>
                  </div>
                  <div class="fish-rate-star-second">
                    <div class="custom-star-wrap" style="width: 100%;">
                      <div class="custom-star">
                      </div>
                    </div>
                  </div>
                </div>
              </li>
            </ul>
            <span class="index-module_text_rukZW">5.0分</span>
          </div>
        </div>
        <span class="fish-dropdown-trigger index-module_assessment-btn_6imdF ">
          <button type="button" class="fish-btn fish-btn-round">
            <svg class="index-module_rate-icon_YM1Lc">
              <use xlink:href="#icon_evaluate_fill">
              </use>
            </svg>
            <span class="index-module_assessment-btn-text_fw-+Z">评分</span>
          </button>
        </span>
      </div>
      <div class="index-module_extra_tUQog">
        <div class="index-module_like-wrap_NbyLe  ">
          <svg class="index-module_like_qOb9K">
            <use xlink:href="#web_icon_dianzan_fill">
            </use>
          </svg>
          <div class="index-module_like-count_GXOGd">100万+</div>
        </div>
        <div class="index-module_suggestion-wrap_s+Ii+ ">
          <svg class="index-module_suggestion-icon_IrRxU">
            <use xlink:href="#icon_feedback_fill">
            </use>
          </svg>
          <div>建议</div>
        </div>
      </div>
    </div>
  </div>
  <div class="index-module_divider_rI-lg">
  </div>
  <div class="imageList-module_special-edu-image-list-wrapper_18zfs">
    <div class="imageList-module_special-edu-image-list_+ywag" style="max-width: unset;">
      <div class="course-document">
        <div class="document-context" style="overflow: hidden; height: 100%;">
          <iframe id="pdfPlayerFirefox" src='/pdfjs/2.13/web/viewer.html?file=${pdfUrl}&headers=%7B"X-ND-AUTH":"MAC%20id=%5C"${macId}%5C",nonce=%5C"0%5C",mac=%5C"0%5C""%7D#disablestream=true&disableAutoFetch=true&page=1' allowfullscreen="" frameborder="0" height="100%" width="100%"></iframe>
        </div>
      </div>
    </div>
  </div>
</div>
</div>
                `;
            } else if (!document.querySelector(".main-wrapper")) {
                // 直接跳转
                window.location.href = `${pdfUrl}?accessToken=${macId}`;
            }
            // 重定向至登录页
            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.style = "display: flex";
            div.innerHTML = `<a class="link" href="${pdfUrl}?accessToken=${macId}" target="_blank" style="margin-left: 24px; color: #888;">📓 查看PDF</a><a class="download" style="margin-left: 24px; color: #888;">📓 下载PDF</a>`;
            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) {
                const progressText = "📓 下载中";
                const fileName = (title || document.querySelector(".index-module_title_bnE9V").innerText || "PDF") + ".pdf";
                const completeText = "📓 下载完成";
                element2.addEventListener("click", function() {
                    downloadFile(pdfUrl, element2, progressText, fileName, completeText);
                });
            }
            setTimeout(function() { // 等待 iframe 加载;教材内页主文档通过 iframe 获取 URL 以更新链接
                const iframe = document.querySelector("iframe");
                if (iframe) {
                    const iframeSrc = iframe.src;
                    console.log("iframeSrc:", iframeSrc);
                    const iframePDF = new URL(iframeSrc).searchParams.get("file");
                    console.log("iframePDF:", iframePDF);
                    const element1 = div.querySelector("a.link"); // 按钮 element1
                    if (element1) {
                        element1.href = `${iframePDF}?accessToken=${macId}`; // 更新 element1 的 URL
                    }
                    pdfUrl = iframePDF; // 更新 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 = unsafeWindow.fetch;
    unsafeWindow.fetch = function (url, options) {
        return originalFetch(url, options) // 调用原始 fetch 函数
            .then(response => {
                if (response.status === 200 && url.endsWith(".pdf")) {
                    const headers = Object.fromEntries(response.headers.entries()); // 响应头
                    console.log("PDF response headers:", headers);
                    console.log("Fetched PDF:", url);
                    if (!url.includes("-private")) { // 课后阅读
                        const element1 = document.querySelector("a.link"); // 按钮 element1
                        if (element1) { // iframe 内部不存在此元素;可在主文档中通过 iframe 获取文件 URL
                            element1.href = url; // 修改 element1 的 URL
                        }
                        pdfUrl = url; // 修改 element2 的 URL
                    } else { // 课程教学
                        const syncClassSuggestion = document.getElementsByClassName("index-module_suggestion-wrap_s+Ii+")[0];
                        if (syncClassSuggestion) {
                            document.querySelector("div > .link")?.parentElement.remove();
                            const div = document.createElement("div");
                            div.setAttribute("style", "display: flex; align-items: center; margin-left: 30px");
                            div.innerHTML = `<a class="link" href="${url}?accessToken=${macId}" target="_blank" style="color: #888;">📓 查看PDF</a>`;
                            syncClassSuggestion.insertAdjacentElement("afterend", div);
                        }
                    }
                }
                return response;
            })
            .catch(error => {
                console.error(error);
            });
    };

})();