// ==UserScript== // @name 国家智慧教育公共服务平台助手 // @namespace github.com/hmjz100 // @version 1.0.8.1 // @description 《也许同类型中最好用?》系列 - 中小学智教平台免登录查看文档(教材 / 书籍 / 政策 / 课件,下载原件或预览)、视频(宣传 / 经验 / 影视,播放器更多功能)、音频(听力,倍速播放)、专题(前两者的综合列表)、课程(自主学习 / 备课授课),还可以查看、切换和下载 新版(2024版) / 中版(常见版) / 先版(旧版) 教材;美化整个国家智慧教育公共服务平台,顶栏毛玻璃,页脚一言诗;增强原 PDF 预览页,可网页全屏,可下载,确保舒适阅读;脚本界面融合原生网页风格,简洁且美观 // @author hmjz100 // @match *://*.smartedu.cn/* // @match *://*.edu.cn/* // @match *://*.eduyun.cn/* // @icon  // @license MIT // @run-at document-body // @grant GM_getresourceText // @grant GM_openInTab // @grant GM_xmlHttpRequest // @grant GM.xmlHttpRequest // @grant unsafeWindow // @connect smartedu.cn // @connect edu.cn // @connect eduyun.cn // @connect ykt.cbern.com.cn // @connect hitokoto.cn // @require https://unpkg.com/jquery@3.6.3/dist/jquery.min.js // @require https://unpkg.com/hls.js@1.5.10/dist/hls.min.js // @require https://unpkg.com/artplayer@5.1.6/dist/artplayer.legacy.js // @require https://unpkg.com/pinyin-pro@3.24.2/dist/index.js // ==/UserScript== (async function SmartEduDownload() { 'use strict'; /* 防止代码因其他原因被执行多次 这段代码出自 Via轻插件,作者谷花泰 */ const key = encodeURIComponent('国家中小学智慧教育平台助手:主代码'); if (unsafeWindow[key]) return; unsafeWindow[key] = true; var { pinyin } = pinyinPro; Artplayer.LOG_VERSION = false console.log("【国家中小学智慧教育平台助手】即时\n运行中…") // Hook 窗口打开 let originOpen = window.open unsafeWindow.oopen = originOpen unsafeWindow.open = function (url, name, specs) { const cUrl = new URL(location.href); if (cUrl.searchParams.get('x-edu-theme') === 'dark') { const xUrl = new URL(url, location.href); if (xUrl.host === location.host) { xUrl.searchParams.set('x-edu-theme', 'dark'); return originOpen.call(unsafeWindow, xUrl.href, name, specs); } } else { return originOpen.call(unsafeWindow, url, name, specs); } }; addDownStyle(); let activityId = new URL(location.href).searchParams.get("activityId") || ''; let contentType = new URL(location.href).searchParams.get("contentType") || ''; let contentId = new URL(location.href).searchParams.get("contentId") || ''; let isNight = new URL(location.href).searchParams.get("x-edu-theme") === 'dark' || false; let contentFirst = new URL(location.href).searchParams.get("contentFirst") || ''; let contentCommon = new URL(location.href).searchParams.get("contentCommon") || ''; let isLogin = false // 根据头像检测是否已经登录账号 waitForKeyElements('[class*="index-module_avatar"]', function () { isLogin = true }) // 给顶栏加个时钟 let img_question = "" waitForKeyElements('[class^="index-module_menu-container_"]', (element) => { var timebar = $(`
,现在是感谢您使用本脚本~
`); timebar.append(``); timebar.find("#smartEduGood").on('click', function () { GM_openInTab('https://greasyfork.org/scripts/459404/feedback', { active: true }); }) if (/((en.|)basic.smartedu.cn|(xue\-test|web\-bd|en\-test|test\d).ykt.eduyun.cn)/.test(location.host)) { timebar.append(``); timebar.find("#smartEduTheme").on('click', function () { let url = new URL(location); if (url.searchParams.get("x-edu-theme") === 'dark') { url.searchParams.delete('x-edu-theme'); } else { url.searchParams.set("x-edu-theme", "dark"); } $(this).remove(); location.href = url.toString(); }); // 默认收起侧栏 waitForKeyElements('[class^="index-module_collapse_"]', (element) => { element.click(); }, true) } if (element && element.length > 0) { element.after(timebar); window.setInterval(function () { timebar.find("#smartEduTime").text(Time()); timebar.find("#smartEduGreeting").text(Greeting()); }, 500); } }, true); // 移除顶部客户端相关内容 if (location.host !== 'www.smartedu.cn') waitForKeyElements('[class^="index-module_top-bar"], [class^="index-module_download"], [class^="index-module_float-download"], #app div.bg-no-repeat div[class^="bg"], #app header.el-header.plang div.topLine1, div.header div.headbar', (element) => { element.remove() }, true) waitForKeyElements('[class*="index-module_modal"].theme-modal', (element) => { element.hide() }) // 给底栏增加个一言 waitForKeyElements('.main-wrapper #main-content div.content', (element) => { var poembar = $(`
`) if (element && element.length > 0) { let text = poembar.find(".todayPoem") element.after(poembar) text.on('click', function () { if (text.text() === "加载中……") return; text.text("加载中……") text.css({ "cursor": "default" }); Poem(text) }) text.text("加载中……") text.css({ "cursor": "default" }); Poem(text) } }, true) const originalOpen = XMLHttpRequest.prototype.open; unsafeWindow.XMLHttpRequest.prototype.open = function (method, url) { if ((location.pathname.includes('/tchMaterial') || location.pathname.includes('/elecEdu')) && url.includes("query?res_ids=") && url.match(/res_ids=([^&]*)/)[1]) { const ids = url.match(/res_ids=([^&]*)/)[1].split(','); $('.index-module_toolbar').remove() try { let idIndex = 0; document.querySelectorAll('li[class*="index-module_item"]').forEach(async element => { element = $(element) if (idIndex < ids.length) { if ( !element.find('[class^="index-module_cover"] img[src]') && !element.find('[class^="index-module_cover"] img[src]').attr('src') && !element.find('[class^="index-module_cover"] img[src]').attr('src').match(/r(\d)-ndr/)[1] ) return; let base = element.find('[class^="index-module_cover"] img[src]').attr('src').match(/r(\d)-ndr/)[1]; let contentId = ids[idIndex] idIndex++; // 教材信息 let content = await request({ url: `https://s-file-${base}.ykt.cbern.com.cn/zxx/ndrv2/resources/tch_material/details/${contentId}.json`, headers: { "referer": "https://basic.smartedu.cn/", 'Cache-Control': 'no-cache' } }); if (content.toString().startsWith('
阅读专题内容(文件列表)
`) button.find(".index-module_item.read.new").on('click', async function () { GM_openInTab(`//${location.host}/tchMaterial/detail?contentType=thematic_course&contentId=${contentId}&catalogType=tchMaterial&subCatalog=tchMaterial${isNight ? '&x-edu-theme=dark' : ''}`, { active: true }); }) element.css({ "border-bottom": "none" }) return element.after(button); } let data = { id: content.id, title: content.global_title['zh-CN'], cover: content.custom_properties?.thumbnails[0], author: content.provider_list[0]?.name, url: content.ti_items?.find(item => item.ti_format === "pdf")?.ti_storages?.[base - 1].replace(/(r\d)-ndr-\w+/, '$1-ndr'), titleCommon: element.find('[class^="index-module_line"] span[title]')[0].title + " 中版", urlCommon: `https://r${base}-ndr.ykt.cbern.com.cn/edu_product/esp/assets/${contentId}.pkg/pdf.pdf`, titleFirst: element.find('[class^="index-module_line"] span[title]')[0].title + " 先版", urlFirst: `https://r${base}-ndr.ykt.cbern.com.cn/edu_product/esp/assets_document/${contentId}.pkg/pdf.pdf` }; console.log("【国家中小学智慧教育平台助手】即时\n教材列表:书籍数据加载成功\n", data) element.css({ "border-bottom": "none" }) let button = $(`
  • 阅读
    下载
    阅读中版
    下载中版
    阅读先版
    下载先版
  • `); if (isLogin) button.find('.index-module_item.read:not(.index-module_item.new)').html("登录下无法使用").fadeOut() let fileName = decodeURIComponent(data.urlCommon.substring(data.urlCommon.lastIndexOf('/') + 1)) let newFileName = decodeURIComponent(data.url.substring(data.url.lastIndexOf('/') + 1)) // 根据文件链接情况显示按钮 let commonAndNewAreSame = (fileName === newFileName); let commonAndFirstAreSame = (data.url === data.urlFirst); // 获取文件大小并显示 getFileSize(data.url).then(size => { button.find('.index-module_item.download.new .size').text(`(${formatFileSize(size)})`); }).catch(() => { button.find('.index-module_item.new').html("找不到新版").fadeOut(); }); button.find(".index-module_item.read.new").on('click', async function () { GM_openInTab(`//${location.host}/tchMaterial/detail?contentType=assets_document&contentId=${contentId}&catalogType=tchMaterial&subCatalog=tchMaterial${isNight ? '&x-edu-theme=dark' : ''}`, { active: true }); }) if (!commonAndNewAreSame) { getFileSize(data.urlCommon).then(size => { button.find('.index-module_item.download.common .size').text(`(${formatFileSize(size)})`); }).catch(() => { button.find('.index-module_item.common').html("找不到中版").fadeOut(); }); button.find(".index-module_item.read.common").on('click', async function () { GM_openInTab(`//${location.host}/tchMaterial/detail?contentType=assets_document&contentId=${contentId}&catalogType=tchMaterial&subCatalog=tchMaterial&contentCommon=true${isNight ? '&x-edu-theme=dark' : ''}`, { active: true }); }) } else { button.find('.index-module_item.common').html("找不到中版").fadeOut(); } if (!commonAndFirstAreSame) { getFileSize(data.urlFirst).then(size => { button.find('.index-module_item.download.first .size').text(`(${formatFileSize(size)})`); }).catch(() => { button.find('.index-module_item.first').html("找不到先版").fadeOut(); }); button.find(".index-module_item.read.first").on('click', async function () { GM_openInTab(`//${location.host}/tchMaterial/detail?contentType=assets_document&contentId=${contentId}&catalogType=tchMaterial&subCatalog=tchMaterial&contentFirst=true${isNight ? '&x-edu-theme=dark' : ''}`, { active: true }); }) } else { button.find('.index-module_item.first').html("找不到先版").fadeOut(); } element.after(button); button.find(".index-module_item.download").on('click', async function () { let element = $(this) let ins = {}; let progress = {}; let index = element.data('index'); let orgtext = element.find('.text').text() element.prop('disabled', true); ins[index] = setInterval(function () { let prog = +progress[index] || 0; element.find('.text').text(prog + "%"); }, 10); try { let response = await fetch(element.hasClass('first') ? data.urlFirst : element.hasClass('common') ? data.urlCommon : data.url); if (!response.ok) throw new Error('网络响应失败'); let totalSize = response.headers.get('Content-Length') || 0; let reader = response.body.getReader(); let receivedLength = 0; let chunks = []; while (true) { const { done, value } = await reader.read(); if (done) break; chunks.push(value); receivedLength += value.length; progress[index] = Math.floor((receivedLength / totalSize) * 100); } let blob = new Blob(chunks); clearInterval(ins[index]); progress[index] = 100; element.find('.text').text("完成~"); // 下载文件 const downloadUrl = URL.createObjectURL(blob); const a = document.createElement('a'); a.href = downloadUrl; a.download = `${element.hasClass('first') ? data.titleFirst : element.hasClass('common') ? data.titleCommon : data.title}.pdf`; a.click(); URL.revokeObjectURL(downloadUrl); setTimeout(function () { element.find('.text').text(orgtext); $(this).prop('disabled', false); }.bind(this), 3000); } catch (error) { clearInterval(ins[index]); element.find('.text').text("失败,重试?"); $(this).prop('disabled', false).one('click', async function () { $(this).trigger('click'); }); } }); } }) } catch (error) { console.error('【国家中小学智慧教育平台助手】\n获取数据时发生错误:', error); } } return originalOpen.apply(this, arguments); }; // 若是教材阅读页 if (/^\/tchMaterial\/detail/.test(location.pathname) && (contentType === 'assets_document' || contentType === 'x_document') && contentId && !isLogin) { try { // 教材信息 let data = await Promise.any(["1", "2", "3"].map(async base => { try { let file = await request({ url: `https://s-file-${base}.ykt.cbern.com.cn/zxx/ndrv2/resources/tch_material/details/${contentId}.json`, headers: { 'Cache-Control': 'no-cache' } }); let audio = await request({ url: `https://s-file-${base}.ykt.cbern.com.cn/zxx/ndrs/resources/${contentId}/relation_audios.json`, headers: { 'Cache-Control': 'no-cache' } }); let item = { id: file.id, title: file.global_title['zh-CN'], cover: file.custom_properties?.thumbnails[0], author: file.provider_list[0]?.name, url: file.ti_items?.find(item => item.ti_format === "pdf")?.ti_storages?.[base - 1].replace(/(r\d)-ndr-\w+/, '$1-ndr'), audio: (() => { if (audio.length !== 0) { let list = [] audio.forEach(item => { list.push({ id: item.id, title: item.global_title['zh-CN'], url: item.ti_items?.find(file => file?.ti_format === 'mp3')?.ti_storages[base - 1].replace(/(r\d)-ndr-\w+/, '$1-ndr'), format: item.ti_items?.find(file => file?.ti_format === 'mp3')?.ti_format, duration: (item.ti_items?.find(file => file?.ti_format === 'mp3')?.custom_properties?.requirements?.find(require => require?.name === 'Duration')?.value.match(/PT(\d+)M(\d+)S/)?.slice(1).reduce((acc, val) => acc * 60 + +val, 0)) || 0, durationText: (() => { const isoDuration = item.ti_items?.find(file => file?.ti_format === 'mp3')?.custom_properties?.requirements?.find(require => require?.name === 'Duration')?.value; const duration = isoDuration?.match(/PT(?:(\d+)H)?(?:(\d+)M)?(?:(\d+)S)/); if (duration) { const hours = String(duration[1] || 0).padStart(2, '0'); // 小时 const minutes = String(duration[2] || 0).padStart(2, '0'); // 分钟 const seconds = String(duration[3] || 0).padStart(2, '0'); // 秒 return hours !== '00' ? `${hours}:${minutes}:${seconds}` : minutes !== '00' ? `${minutes}:${seconds}` : seconds !== '00' ? `${minutes}:${seconds}` : `/`; } return null; })() }) }) return list } return null; })() }; if (contentFirst === "true") { item.title = file.global_title['zh-CN'].replace(/(.*?课程标准.*?)/, '').replace(/•.*?•/, '·') + ' 先版' item.url = `https://r${base}-ndr.ykt.cbern.com.cn/edu_product/esp/assets_document/${contentId}.pkg/pdf.pdf` } if (contentCommon === "true") { item.title = file.global_title['zh-CN'].replace(/(.*?课程标准.*?)/, '').replace(/•.*?•/, '·') + ' 中版' item.url = `https://r${base}-ndr.ykt.cbern.com.cn/edu_product/esp/assets/${contentId}.pkg/pdf.pdf` } return item } catch (error) { console.error('【国家中小学智慧教育平台助手】\n获取数据时发生错误:', error); } })); console.log("【国家中小学智慧教育平台助手】即时\n教材预览:书籍数据加载成功\n", data) $('title').html(data.title + " - 国家中小学智慧教育平台"); // 遇到默认弹窗 waitForKeyElements('div.fish-modal-content div.fish-modal-body div.fish-modal-confirm-body-wrapper', (elemlogin) => { if (!elemlogin.find('div.fish-modal-confirm-content').text().match("需要登录才可以查看")) return; waitForKeyElements('div[class^="index-module_special-edu-detail_"] div[class^="index-module_content-wrap_"] div[class^="index-module_wrapper_"]', (element) => { // 点击登录提示中隐藏的取消按钮 elemlogin.find('button[type="button"].fish-btn').filter(function () { return $(this).text().includes("取 消"); }).click(); $('.fish-breadcrumb.breadcrumb-container:last-child span.fish-breadcrumb-link').html(data.title); let fullDom = $(`

    ${data.title}

    ${data.author} 所有平台资源均免费使用,任何单位及个人不得用于商业行为
    感谢您使用本脚本~
    感谢所有为资源建设提供资料的单位和个人
    未经允许不得转载或引用
    `) if (data.audio) { fullDom = $(`

    ${data.title}

    ${data.author} 所有平台资源均免费使用,任何单位及个人不得用于商业行为
    感谢您使用本脚本~
    感谢所有为资源建设提供资料的单位和个人
    未经允许不得转载或引用
    音频资源
    请选择媒体以开始播放
    X1
    `) let svg_play = ``; let svg_pause = ``; let svg_playing = $(`
    `) let svg_pauseing = $(` `) fullDom.find('#media').on('loadstart', function () { const media = $(this); let button = fullDom.find("#playButton") if (media.paused) { button.html(svg_pause); } else { button.html(svg_play); } fullDom.find("#progress").css('width', '0%'); fullDom.find("#progpoint").css('left', '0%'); }); fullDom.find('#media').on('play', function () { let button = fullDom.find("#playButton") button.html(svg_play); fullDom.find('[class*="audioList-module_audio-item-active"] #playing #status').replaceWith(svg_playing) }); fullDom.find('#media').on('pause', function () { let button = fullDom.find("#playButton") button.html(svg_pause); fullDom.find('[class*="audioList-module_audio-item-active"] #playing #status').replaceWith(svg_pauseing) }); fullDom.find('#media').on('ended', function () { let button = fullDom.find("#playButton") button.html(svg_pause); fullDom.find("#progress").css('width', '100%'); fullDom.find("#progpoint").css('left', '100%'); }); fullDom.find('#media').on('timeupdate', function () { const media = $(this)[0]; const originTime = media.currentTime; // 当前播放时间 const currentTime = Math.floor(originTime); // 当前播放时间(取整) const duration = media.duration; // 总时长(秒) if (isNaN(duration)) return; const hours = String(Math.floor(currentTime / 3600)).padStart(2, '0'); const minutes = String(Math.floor((currentTime % 3600) / 60)).padStart(2, '0'); const seconds = String(currentTime % 60).padStart(2, '0'); const dSeconds = `.${(originTime - currentTime).toFixed(2).split('.')[1]}`.padStart(3, '0'); // 处理总时长 const totalHours = String(Math.floor(duration / 3600)).padStart(2, '0'); const totalMinutes = String(Math.floor((duration % 3600) / 60)).padStart(2, '0'); const totalSeconds = String(Math.floor(duration % 60)).padStart(2, '0'); const totalDSeconds = `.${(duration - Math.floor(duration)).toFixed(2).split('.')[1]}`.padStart(3, '0'); fullDom.find('#audioTime').text(hours !== '00' ? `${hours}:${minutes}:${seconds}${dSeconds}` : minutes !== '00' ? `${minutes}:${seconds}${dSeconds}` : `00:${seconds}${dSeconds}`); fullDom.find('#audioDuration').text(totalHours !== '00' ? `${totalHours}:${totalMinutes}:${totalSeconds}${totalDSeconds}` : totalMinutes !== '00' ? `${totalMinutes}:${totalSeconds}${totalDSeconds}` : `00:${totalSeconds}${totalDSeconds}`); const progress = (originTime / duration) * 100; fullDom.find("#progress").css('width', progress + '%'); fullDom.find("#progpoint").css('left', progress + '%'); }); fullDom.find("#playButton").on('click', function () { let media = fullDom.find('#media')[0] if (isNaN(media.duration)) return; if (media.paused) { media.play(); } else { media.pause(); } }); fullDom.find('#speedButton').on('click', function (event) { const media = fullDom.find('#media')[0]; const element = $(this) if (isNaN(media.duration)) return; const speeds = [0.5, 0.75, 1.0, 1.25, 1.5, 1.75, 2.0]; // 倍速数组 let currentSpeedIndex = (speeds.indexOf(media.playbackRate) + 1) % speeds.length; // 计算下一个速度索引 media.playbackRate = speeds[currentSpeedIndex]; // 更新倍速 element.find('span').text(`X${speeds[currentSpeedIndex]}`); // 更新显示 }); fullDom.find('#progressBar').on('mousedown', function (event) { event.preventDefault(); function update(event) { const media = fullDom.find('#media')[0]; const progressBar = fullDom.find('#progressBar'); const progressBarWidth = progressBar.width(); let offsetX = event ? (event.pageX - progressBar.offset().left) : 0; // 限制拖动范围 if (offsetX < 0) offsetX = 0; if (offsetX > progressBarWidth) offsetX = progressBarWidth; // 更新进度条和点的位置 const percentage = offsetX / progressBarWidth; // 更新视频播放进度 media.currentTime = percentage * media.duration; } if (isNaN(media.duration)) return; update(event) $(document).on('mousemove', function (event) { update(event) }); $(document).on('mouseup', function (event) { event = event || window.event; document.onmousemove = null; document.onmouseup = null; $(document).off('mousemove'); $(document).off('mouseup'); }); }); let lastItem function update(item, itemDom) { if (item.id === lastItem) return; lastItem = item.id if (!fullDom.find('#media')[0].paused) fullDom.find('#media')[0].pause() fullDom.find('#media').attr('src', item.url) fullDom.find('#audioTitle').text(item.title) fullDom.find('#audioTitle').attr('title', item.title) fullDom.find('#audioTime').text(`00:00`) fullDom.find('#audioDuration').text(item.durationText) try { if (fullDom.find('#media')[0].paused) fullDom.find('#media')[0].play() } catch (e) { } fullDom.find('[class*="audioList-module_audio-item-active"] #playing').css('color', '') fullDom.find('[class*="audioList-module_audio-item-active"] #playing #duration').show() fullDom.find('[class*="audioList-module_audio-item-active"] #playing #status').replaceWith('') fullDom.find('[class*="audioList-module_audio-item-active"]').removeClass('audioList-module_audio-item-active_Xx7f-') fullDom.find('[class*="audioList-module_audio-item-active"]').removeClass('audioList-module_audio-item-active_Xx7f-') itemDom.addClass('audioList-module_audio-item-active_Xx7f-') itemDom.find('#playing #duration').hide() itemDom.find('#playing #status').replaceWith(svg_playing) } Object.values(data.audio).forEach((item) => { let itemDom = $(`
    ${item.title}
    ${item.durationText}
    `) fullDom.find('#list').append(itemDom) itemDom.on('click', function () { update(item, itemDom) }) let data = { name: `${item.title}.${item.format}`, url: item.url } let downloadDom = $(``) downloadDom.on('click', async function (event) { event.stopPropagation(); let element = $(this) let ins = {}; let progress = {}; let index = element.data('index'); element.prop('disabled', true); ins[index] = setInterval(function () { let prog = +progress[index] || 0; element.find('.text').text(`${prog} %`); }, 10); try { let response = await fetch(data.url); if (!response.ok) throw new Error('网络响应失败'); let totalSize = response.headers.get('Content-Length') || 0; let reader = response.body.getReader(); let receivedLength = 0; let chunks = []; while (true) { const { done, value } = await reader.read(); if (done) break; chunks.push(value); receivedLength += value.length; progress[index] = Math.floor((receivedLength / totalSize) * 100); } let blob = new Blob(chunks); clearInterval(ins[index]); progress[index] = 100; element.find('.text').text("完成~"); // 下载文件 const downloadUrl = URL.createObjectURL(blob); const a = document.createElement('a'); a.href = downloadUrl; a.download = `${data.name}`; a.click(); URL.revokeObjectURL(downloadUrl); setTimeout(function () { element.find('.text').text("下载"); $(this).prop('disabled', false); }.bind(this), 3000); } catch (error) { clearInterval(ins[index]); console.error("【国家中小学智慧教育平台助手】即时\n下载教材时出错", error) element.find('.text').text("失败,重试?"); $(this).prop('disabled', false).one('click', async function () { $(this).trigger('click'); }); } }) itemDom.find('#info').after(downloadDom) }) } // 既然不给页面内容,那就抄一个已登录的放上去吧 element.html(fullDom) }, true) }) } catch (error) { console.error('【国家中小学智慧教育平台助手】\n获取数据时发生错误:', error); } } // 若是文档阅读页 if ((contentType === 'assets_document' || contentType === 'x_document') && contentId && !/^\/tchMaterial\/detail/.test(location.pathname)) { try { // 文档信息 let data = await Promise.any([1, 2, 3].map(async base => { try { let data = await request({ url: `https://s-file-${base}.ykt.cbern.com.cn/zxx/ndrs/special_edu/resources/details/${contentId}.json`, headers: { 'Cache-Control': 'no-cache' } }); let item = { id: data.id, title: data.global_title['zh-CN'], cover: data.custom_properties?.thumbnails[0], author: data.provider_list[0]?.name, url: data.ti_items?.find(file => file?.ti_format === "pdf")?.ti_storages[base - 1].replace(/(r\d)-ndr-\w+/, '$1-ndr') }; return item } catch (error) { console.error('【国家中小学智慧教育平台助手】\n获取数据时发生错误:', error); } })); console.log("【国家中小学智慧教育平台助手】即时\n文档预览:文档数据加载成功\n", data) $('title').html(data.title + " - 国家中小学智慧教育平台"); // 遇到默认弹窗 waitForKeyElements('div.fish-modal-content div.fish-modal-body div.fish-modal-confirm-body-wrapper', (elemlogin) => { if (!elemlogin.find('div.fish-modal-confirm-content').text().match("需要登录才可以查看")) return; waitForKeyElements('div[class^="index-module_special-edu-detail_"] div[class^="index-module_content-wrap_"] div[class^="index-module_wrapper_"]', (element) => { // 点击登录提示中隐藏的取消按钮 elemlogin.find('button[type="button"].fish-btn').filter(function () { return $(this).text().includes("取 消"); }).click(); $('.fish-breadcrumb.breadcrumb-container:last-child span.fish-breadcrumb-link').html(data.title); let fullDom = $(`

    ${data.title}

    ${data.author} 所有平台资源均免费使用,任何单位及个人不得用于商业行为
    感谢您使用本脚本~
    感谢所有为资源建设提供资料的单位和个人
    未经允许不得转载或引用
    `) // 既然不给页面内容,那就抄一个已登录的放上去吧 element.html(fullDom) }, true) }) waitForKeyElements("div.course-document div#viewerContainer", (element) => { element.html(`
    `) element.css({ "border-radius": "8px" }) element.parent().css({ "border-radius": "8px" }) element.parent().parent().css({ "border-radius": "8px" }) waitForKeyElements('div.course-document div[class^="doc-"]', (element) => { element.css({ "display": "none" }) }) }) } catch (error) { console.error('【国家中小学智慧教育平台助手】\n获取数据时发生错误:', error); } } // 若是视频播放页 if ((contentType === 'assets_video' || contentType === 'x_video') && contentId) { try { // 视频信息 let data = await Promise.any([1, 2, 3].map(async base => { try { let data = await request({ url: `https://s-file-${base}.ykt.cbern.com.cn/zxx/ndrs/special_edu/resources/details/${contentId}.json`, headers: { 'Cache-Control': 'no-cache' } }); let item = { id: data.id, title: data.global_title?.['zh-CN'], cover: data.custom_properties?.thumbnails?.[0], author: data.provider_list?.[0]?.name, url: null, urls: [] }; let seenMd5 = new Set(); let maxQuality = null; data.un_ti_items.forEach(video => { let md5 = video.ti_md5; let encryption = video?.custom_properties?.encryption if (!seenMd5.has(md5) && video?.custom_properties?.requirements && (encryption !== 'drm')) { let quality = null; video.custom_properties.requirements.forEach(req => { if (req.name === 'Height') { quality = Number(req.value); if (maxQuality === null || quality > maxQuality) { maxQuality = quality; } } }); let url = { "url": video.ti_storages[base - 1].replace(/(r\d)-ndr-\w+/, '$1-ndr'), "html": quality <= 360 ? `流畅 ${quality}P` : quality <= 480 ? `标清 ${quality}P` : quality <= 720 ? `高清 ${quality}P` : quality <= 1080 ? `全高清 ${quality}P` : quality <= 1440 ? `极高清 2K ${quality}P` : quality <= 2160 ? `超高清 4K ${quality}P` : `${quality}P`, "quality": quality }; item.urls.push(url); seenMd5.add(md5); } }); item.urls.sort((a, b) => (b.quality || 0) - (a.quality || 0)); item.urls.forEach(url => { if (url.quality === maxQuality) { url.default = true; item.url = url.url } }); return item } catch (error) { console.error('【国家中小学智慧教育平台助手】\n获取数据时发生错误:', error); } })); console.log("【国家中小学智慧教育平台助手】即时\n视频预览:视频数据加载成功\n", data) $('title').html(data.title + " - 国家中小学智慧教育平台"); waitForKeyElements("div.course-video div.fish-video", (element) => { element.html(`
    `) play(data.id, data.url, data.urls, data.cover, (art) => { art.controls.add({ name: 'hideList', index: 50, position: 'right', html: '', tooltip: '显隐推荐', click: function () { $('[class^="index-module_comp-related-recommend"]').fadeToggle() }, }) }) element.css({ "height": "auto", "width": "auto", "aspect-ratio": "16/9" }) element.parent().css({ "border-radius": "8px", "height": "auto", "width": "auto" }) element.parent().parent().css({ "border-radius": "8px", "height": "auto", "width": "auto" }) }, true) } catch (error) { console.error('【国家中小学智慧教育平台助手】\n获取数据时发生错误:', error); } } /* 若是专题页 */ if (contentType === 'thematic_course' && contentId && !isLogin) { try { // 专题信息 let data = await Promise.any(["1", "2", "3"].map(async base => { try { let data = await request({ url: `https://s-file-${base}.ykt.cbern.com.cn/zxx/ndrs/special_edu/resources/details/${contentId}.json`, headers: { 'Cache-Control': 'no-cache' } }); let listParent = await request({ url: `https://s-file-${base}.ykt.cbern.com.cn/zxx/ndrs/special_edu/thematic_course/trees/${contentId}.json`, headers: { 'Cache-Control': 'no-cache' } }); let listChild = await request({ url: `https://s-file-${base}.ykt.cbern.com.cn/zxx/ndrs/special_edu/thematic_course/${contentId}/resources/list.json`, headers: { 'Cache-Control': 'no-cache' } }); let list = {}; listParent.forEach((tree, index) => { list[index] = { id: tree.id, title: tree.title, richTitle: tree.rich_title, item: [] }; listChild.forEach((item) => { if (item.chapter_ids[0] === tree.id) { list[index].item.push({ id: item.id, title: item.global_title['zh-CN'].replace(" ", " "), cover: item.custom_properties?.thumbnails[0], format: item.ti_items?.find(file => (file?.ti_format === "pdf" || file?.ti_format === "m3u8"))?.ti_format, source: (() => { const sourceFile = item.ti_items?.find(file => file?.ti_is_source_file === true); if (sourceFile && sourceFile.ti_storages) { if (sourceFile.ti_storages.length >= base) { return sourceFile.ti_storages[base - 1].replace(/(r\d)-ndr-\w+/, '$1-ndr') } else { return sourceFile.ti_storages[0].replace(/(r\d)-ndr-\w+/, '$1-ndr') } } return null; })(), url: (() => { const previewFile = item.ti_items?.find(file => file?.ti_format === "pdf"); if (previewFile && previewFile.ti_storages) { if (previewFile.ti_storages.length >= base) { return previewFile.ti_storages[base - 1].replace(/(r\d)-ndr-\w+/, '$1-ndr') } else { return previewFile.ti_storages[0].replace(/(r\d)-ndr-\w+/, '$1-ndr') } } return null; })(), urls: (() => { if (item.ti_items?.find(file => (file?.ti_format === "m3u8"))?.ti_format !== "m3u8") return null; let urls = []; let maxQuality = null; let seenMd5 = new Set(); item.un_ti_items.forEach(video => { let md5 = video.ti_md5; let encryption = video?.custom_properties?.encryption if (!seenMd5.has(md5) && video?.custom_properties?.requirements && (encryption !== 'drm')) { let quality = null; video.custom_properties.requirements.forEach(req => { if (req.name === 'Height') { quality = Number(req.value); if (maxQuality === null || quality > maxQuality) { maxQuality = quality; } } }); let url = { "url": video.ti_storages[base - 1].replace(/(r\d)-ndr-\w+/, '$1-ndr'), "html": quality <= 360 ? `流畅 ${quality}P` : quality <= 480 ? `标清 ${quality}P` : quality <= 720 ? `高清 ${quality}P` : quality <= 1080 ? `全高清 ${quality}P` : quality <= 1440 ? `极高清 2K ${quality}P` : quality <= 2160 ? `超高清 4K ${quality}P` : `${quality}P`, "quality": quality, "item": video }; urls.push(url); seenMd5.add(md5); } }); urls.sort((a, b) => (b.quality || 0) - (a.quality || 0)); urls.forEach(url => { if (url.quality === maxQuality) { url.default = true; } }); return urls; })(), }); } }); list[index].item.sort((a, b) => { // 使用 pinyin 库将中文转换为拼音进行排序 let pinyinA = pinyin(a.title, { toneType: 'none' }); let pinyinB = pinyin(b.title, { toneType: 'none' }); // 正则表达式提取字符串中的数字 let numA = a.title.match(/\d+/); let numB = b.title.match(/\d+/); // 如果两者都有数字,按数字排序 if (numA && numB) { return parseInt(numA[0], 10) - parseInt(numB[0], 10); } // 如果只有一个有数字,优先按数字排序 if (numA) return -1; if (numB) return 1; // 否则按拼音排序 return pinyinA.localeCompare(pinyinB); }); }); let item = { id: data.id, title: data.global_title['zh-CN'], cover: data.custom_properties?.thumbnails[0], author: data.provider_list[0]?.name, list: list, }; return item } catch (error) { console.error('【国家中小学智慧教育平台助手】\n获取数据时发生错误:', error); } })); console.log("【国家中小学智慧教育平台助手】即时\n教材专题:项目数据加载成功\n", data) $('title').html(data.title + " - 国家中小学智慧教育平台"); waitForKeyElements('div[class*="index-module_detail-top"]', (element) => { let fullDom = $(`
    ${data.title}
    ${data.author} 所有平台资源均免费使用,任何单位及个人不得用于商业行为
    感谢您使用本脚本~

    启始

    请从右侧目录选择文件以查看预览…

    感谢所有为资源建设提供资料的单位和个人
    未经允许不得转载或引用
    目录
    `) // 既然不给页面内容,那就抄一个已登录的放上去吧 element.replaceWith(fullDom) let lastItem function update(list, item, listDom, itemDom) { if (item.id === lastItem) return; lastItem = item.id fullDom.find('div.course-document div#viewerContainer').html("") fullDom.find('[class*="index-module_study-list-icon-play"]').remove() fullDom.find('[class*="index-module_resource-item-active"]').removeClass("index-module_resource-item-active_7sImn") itemDom.find('#info').append(`
    `) itemDom.addClass("index-module_resource-item-active_7sImn") if (item.format === "pdf") { fullDom.find('div.course-document div#viewerContainer').html(`
    `) } else if (item.format === "m3u8") { fullDom.find('div.course-document div#viewerContainer').html(`
    `) play(item?.id, (item?.urls[0]?.url || ''), (item?.urls || ''), item?.cover, (art) => { art.controls.add({ name: 'hideList', index: 50, position: 'right', html: '', tooltip: '显隐推荐', click: function () { $('[class^="index-module_comp-related-recommend"]').fadeToggle() }, }) }) } } Object.values(data.list).forEach((list, listIndex) => { let listDom = $(`
    ${list.title}
    `) fullDom.find('#list').append(listDom) list.item.forEach((item, itemIndex) => { let itemDom = $(`
    ${item.title}
    `) if (item.title === list.title) { itemDom = $(`
    ${item.title}
    `) } if (item.title === list.title) { listDom.find("*").remove() fullDom.find('#list').append(itemDom) } else { listDom.find('#res').append(itemDom) } itemDom.on('click', function () { update(list, item, listDom, itemDom) }) if (item.source) { let src = new URL(item.source) let data = { name: decodeURIComponent(src.pathname.substring(src.pathname.lastIndexOf('/') + 1)), url: src.toString() } let downloadDom = $(``) downloadDom.on('click', async function (event) { event.stopPropagation(); // 阻止事件冒泡 let element = $(this) let ins = {}; let progress = {}; let index = element.data('index'); element.prop('disabled', true); ins[index] = setInterval(function () { let prog = +progress[index] || 0; element.find('.text').text(`${prog} %`); }, 10); try { let response = await fetch(data.url); if (!response.ok) throw new Error('网络响应失败'); let totalSize = response.headers.get('Content-Length') || 0; let reader = response.body.getReader(); let receivedLength = 0; let chunks = []; while (true) { const { done, value } = await reader.read(); if (done) break; chunks.push(value); receivedLength += value.length; progress[index] = Math.floor((receivedLength / totalSize) * 100); } let blob = new Blob(chunks); clearInterval(ins[index]); progress[index] = 100; element.find('.text').text("完成~"); // 下载文件 const downloadUrl = URL.createObjectURL(blob); const a = document.createElement('a'); a.href = downloadUrl; a.download = `${data.name}`; a.click(); URL.revokeObjectURL(downloadUrl); setTimeout(function () { element.find('.text').text("下载"); $(this).prop('disabled', false); }.bind(this), 3000); } catch (error) { clearInterval(ins[index]); console.error("【国家中小学智慧教育平台助手】即时\n下载教材时出错", error) element.find('.text').text("失败,重试?"); $(this).prop('disabled', false).one('click', async function () { $(this).trigger('click'); }); } }) itemDom.find('#info').append(downloadDom) } //if (listIndex === 0 && itemIndex === 0) itemDom.click() }) }) }, true) // 遇到默认弹窗 waitForKeyElements('div.fish-modal-content div.fish-modal-body div.fish-modal-confirm-body-wrapper', (elemlogin) => { if (!elemlogin.find('div.fish-modal-confirm-content').text().match("需要登录才可以查看")) return; waitForKeyElements('div[class^="index-module_error_"]', (element) => { // 点击登录提示中隐藏的取消按钮 elemlogin.find('button[type="button"].fish-btn').filter(function () { return $(this).text().includes("取 消"); }).click(); $('title').html(data.title + " - 国家中小学智慧教育平台"); let fullDom = $(`
    `) // 既然不给页面内容,那就抄一个已登录的放上去吧 element.replaceWith(fullDom) }, true) }, true) } catch (error) { console.error('【国家中小学智慧教育平台助手】\n获取数据时发生错误:', error); } } /* 若是课程页 */ if (/^\/syncClassroom\/classActivity/.test(location.pathname) && activityId && !isLogin) { try { // 课程信息 let data = await Promise.any(["1", "2", "3"].map(async base => { try { let data = await request({ url: `https://s-file-${base}.ykt.cbern.com.cn/zxx/ndrv2/national_lesson/resources/details/${activityId}.json`, headers: { 'Cache-Control': 'no-cache' } }); let res = await request({ url: `https://s-file-${base}.ykt.cbern.com.cn/zxx/ndrs/national_lesson/resources/${activityId}/relation_resource.json`, headers: { 'Cache-Control': 'no-cache' } }); let item = { id: data.id, title: data.global_title['zh-CN'], cover: data.custom_properties?.thumbnails[0], author: data.provider_list[0]?.name, book: res?.tch_materials?.[0]?.id, list: (() => { if (data?.relations?.national_course_resource) { let list = [] data?.relations?.national_course_resource?.forEach(item => { list.push({ id: item.id, title: item.title, format: item.ti_items?.find(file => (file?.ti_format === "pdf" || file?.ti_format === "m3u8"))?.ti_format, url: (() => { const previewFile = item.ti_items?.find(file => file?.ti_format === "pdf"); if (previewFile && previewFile.ti_storages) { if (previewFile.ti_storages.length >= base) { return previewFile.ti_storages[base - 1].replace(/(r\d)-ndr-\w+/, '$1-ndr') } else { return previewFile.ti_storages[0].replace(/(r\d)-ndr-\w+/, '$1-ndr') } } return null; })(), urls: (() => { if (item.ti_items?.find(file => (file?.ti_format === "m3u8"))?.ti_format !== "m3u8") return null; let urls = []; let maxQuality = null; let seenMd5 = new Set(); item.un_ti_items.forEach(video => { let md5 = video.ti_md5; let encryption = video?.custom_properties?.encryption if (!seenMd5.has(md5) && video?.custom_properties?.requirements && (encryption !== 'drm')) { let quality = null; video.custom_properties.requirements.forEach(req => { if (req.name === 'Height') { quality = Number(req.value); if (maxQuality === null || quality > maxQuality) { maxQuality = quality; } } }); let url = { "url": video.ti_storages[base - 1].replace(/(r\d)-ndr-\w+/, '$1-ndr'), "html": quality <= 360 ? `流畅 ${quality}P` : quality <= 480 ? `标清 ${quality}P` : quality <= 720 ? `高清 ${quality}P` : quality <= 1080 ? `全高清 ${quality}P` : quality <= 1440 ? `极高清 2K ${quality}P` : quality <= 2160 ? `超高清 4K ${quality}P` : `${quality}P`, "quality": quality, "item": video }; urls.push(url); seenMd5.add(md5); } }); urls.sort((a, b) => (b.quality || 0) - (a.quality || 0)); urls.forEach(url => { if (url.quality === maxQuality) { url.default = true; } }); return urls; })() }) }) return list } return null; })(), zorigin: data }; return item } catch (error) { console.error('【国家中小学智慧教育平台助手】\n获取数据时发生错误:', error); } })); console.log("【国家中小学智慧教育平台助手】即时\n教材课程:项目数据加载成功\n", data) $('title').html(data.title + " - 国家中小学智慧教育平台"); // 遇到默认弹窗 waitForKeyElements('div.fish-modal-content div.fish-modal-body div.fish-modal-confirm-body-wrapper', (elemlogin) => { if (!elemlogin.find('div.fish-modal-confirm-content').text().match("需要登录才可以查看")) return; // 点击登录提示中隐藏的取消按钮 elemlogin.find('button[type="button"].fish-btn').filter(function () { return $(this).text().includes("取 消"); }).click(); }, true) if (data.book) { waitForKeyElements('[class*="index-module_tch-material"]', (element) => { let clone = element.clone(true) clone.on('click', () => { GM_openInTab(`//${location.host}/tchMaterial/detail?contentType=assets_document&contentId=${data.book}&catalogType=tchMaterial&subCatalog=tchMaterial${isNight ? '&x-edu-theme=dark' : ''}`, { active: true }); }) element.replaceWith(clone); }, true) } waitForKeyElements('.course-detail-container', (element) => { let fullDom = $(`

    启始

    请从右侧目录选择文件以查看预览…

    感谢所有为资源建设提供资料的单位和个人
    未经允许不得转载或引用
    课时资源
    `) // 既然不给页面内容,那就抄一个已登录的放上去吧 element.html(fullDom) let lastItem function update(item, itemDom) { if (item.id === lastItem) return; lastItem = item.id fullDom.find(".study-list-item-active").removeClass('study-list-item-active') fullDom.find(".study-list-icon-play").removeClass('study-list-icon-play') itemDom.addClass('study-list-item-active') itemDom.find('.study-list-icon-r').addClass('study-list-icon-play') if (item.format === "pdf") { fullDom.find('div.course-document div#viewerContainer').html(`
    `) } else if (item.format === "m3u8") { fullDom.find('div.course-document div#viewerContainer').html(`
    `) play(item?.id, (item?.urls[0]?.url || ''), (item?.urls || ''), item?.cover, (art) => { art.controls.add({ name: 'hideList', index: 50, position: 'right', html: '', tooltip: '显隐推荐', click: function () { $('[class^="index-module_comp-related-recommend"]').fadeToggle() }, }) }) } } data.list.forEach((item, itemIndex) => { let itemDom = $(`
    ${item.title}
    `) fullDom.find('#list').append(itemDom) itemDom.on('click', function () { update(item, itemDom) }) if (item.format === 'pdf') { let src = new URL(item.url) let data = { name: item.title, url: src.toString() } let downloadDom = $(``) downloadDom.on('click', async function (event) { event.stopPropagation(); // 阻止事件冒泡 let element = $(this) let ins = {}; let progress = {}; let index = element.data('index'); element.prop('disabled', true); ins[index] = setInterval(function () { let prog = +progress[index] || 0; element.find('.text').text(`${prog} %`); }, 10); try { let response = await fetch(item.url); if (!response.ok) throw new Error('网络响应失败'); let totalSize = response.headers.get('Content-Length') || 0; let reader = response.body.getReader(); let receivedLength = 0; let chunks = []; while (true) { const { done, value } = await reader.read(); if (done) break; chunks.push(value); receivedLength += value.length; progress[index] = Math.floor((receivedLength / totalSize) * 100); } let blob = new Blob(chunks); clearInterval(ins[index]); progress[index] = 100; element.find('.text').text("完成~"); // 下载文件 const downloadUrl = URL.createObjectURL(blob); const a = document.createElement('a'); a.href = downloadUrl; a.download = `${data.name}.pdf`; a.click(); URL.revokeObjectURL(downloadUrl); setTimeout(function () { element.find('.text').text("下载"); $(this).prop('disabled', false); }.bind(this), 3000); } catch (error) { clearInterval(ins[index]); console.error("【国家中小学智慧教育平台助手】即时\n下载教材时出错", error) element.find('.text').text("失败,重试?"); $(this).prop('disabled', false).one('click', async function () { $(this).trigger('click'); }); } }) itemDom.find('#info').after(downloadDom) } //if (itemIndex === 0) itemDom.click() }) }) } catch (error) { console.error('【国家中小学智慧教育平台助手】\n获取数据时发生错误:', error); } } // 遇到 PDF 预览器的 iframe waitForKeyElements("iframe[src*='/web/viewer.html']", (element) => { let parent = element.parent(); let button = $(`

    国家智慧教育公共服务平台助手

    正在加载 PDF 文件…

    `) element.css({ "background": "#232d2a" }) let pdfWindow = element[0].contentWindow button.find('.index-module_pdf-refresh').on('click', function () { let button = $(this) button.find('.text').text('等待'); pdfWindow.location.reload(); element.on('load', function () { button.find('.text').text('刷新'); }); }) button.find('.index-module_pdf-fullscreen').on('click', function () { let element = $(this) if (!parent.hasClass('fullscreen-dom')) { // 进入全屏模式 $('body').addClass('no-scroll'); parent.addClass('fullscreen-dom'); element.html('退出全屏 (Esc)'); $('#header').fadeOut(); button.find('.index-module_pdf-rating').fadeOut(); button.find('.index-module_pdf-refresh').fadeOut(); button.find('.index-module_pdf-download').fadeOut(); } else { // 退出全屏模式 $('body').removeClass('no-scroll'); parent.removeClass('fullscreen-dom'); element.html('网页全屏 (Esc)'); $('#header').fadeIn(); button.find('.index-module_pdf-rating').fadeIn(); button.find('.index-module_pdf-refresh').fadeIn(); button.find('.index-module_pdf-download').fadeIn(); } }); $(document).off('keydown').on('keydown', function (event) { if (event.key === 'Escape' || event.which === 27) button.find('.index-module_pdf-fullscreen').click(); }); element.on('load', function () { let loading = $('body').find('#tch-loading'); loading.fadeOut(); try { const iframeDocument = this.contentDocument || this.contentWindow.document; $(iframeDocument).off('keydown').on('keydown', function (event) { if (event.key === 'Escape' || event.which === 27) { button.find('.index-module_pdf-fullscreen').click(); } }); } catch (e) { } }); button.find('.index-module_pdf-download').on('click', async function () { let src = new URL(pdfWindow.location.href) let url = src.searchParams.get("file") let fileName = decodeURIComponent(url.substring(url.lastIndexOf('/') + 1)) let title = $(document).find('h3[title]').attr("title") || fileName || document.title let data = { title: title, url: url } let element = $(this) let ins = {}; let progress = {}; let index = element.data('index'); element.prop('disabled', true); ins[index] = setInterval(function () { let prog = +progress[index] || 0; element.find('.text').text(prog + "%"); }, 10); try { let response = await fetch(data.url); if (!response.ok) throw new Error('网络响应失败'); let totalSize = response.headers.get('Content-Length') || 0; let reader = response.body.getReader(); let receivedLength = 0; let chunks = []; while (true) { const { done, value } = await reader.read(); if (done) break; chunks.push(value); receivedLength += value.length; progress[index] = Math.floor((receivedLength / totalSize) * 100); } let blob = new Blob(chunks); clearInterval(ins[index]); progress[index] = 100; element.find('.text').text("完成~"); // 下载文件 const downloadUrl = URL.createObjectURL(blob); const a = document.createElement('a'); a.href = downloadUrl; a.download = `${data.title}.pdf`; a.click(); URL.revokeObjectURL(downloadUrl); setTimeout(function () { element.find('.text').text("下载"); $(this).prop('disabled', false); }.bind(this), 3000); } catch (error) { clearInterval(ins[index]); console.error("【国家中小学智慧教育平台助手】即时\n下载教材时出错", error) element.find('.text').text("失败,重试?"); $(this).prop('disabled', false).one('click', async function () { $(this).trigger('click'); }); } }); element.before(button) }) // 预览页面增强 if (/^\/pdfjs\/.*\/web\/viewer.html/.test(location.pathname) || /^\/web\/viewer.html/.test(location.pathname)) { let file = new URL(location.href).searchParams.get("file"); if (file) { let noPrivateFile = file.replace(/(r\d)-ndr-\w+/, '$1-ndr'); let url = new URL(`${location.origin}${location.pathname}?file=${noPrivateFile}&disableAutoFetch=true&page=1`).toString() if (url !== location.href) location.replace(url); } $('html').css({ "background-color": "#232d2a", "--body-bg-color": "#232d2a", "--toolbar-bg-color": "#2a3632", "--sidebar-toolbar-bg-color": "#2a3632", "--doorhanger-border-color": "#232d2a", "--doorhanger-bg-color": "#2a3632", "--dropdown-btn-bg-color": "#232d2a", "--button-hover-color": "#32403b", "--doorhanger-hover-bg-color": "#32403b", "--overlay-button-bg-color": "#232d2a", "--overlay-button-hover-bg-color": "#32403b", "--field-bg-color": "#232d2a", "--field-border-color": "#32403b", }) $('#loadingBar').css({ "top": "0", "opacity": "0.8", "height": "10px", "margin": "8px 6px", "border": "none", "border-radius": "10px", "background": "#232323" }).appendTo('body'); $('#loadingBar .progress').css({ "border-radius": "50px" }) $('#toolbarViewerMiddle').attr('id', 'toolbarViewerLeft'); $('#viewerContainer').css({ "background-color": "#232d2a" }) $('#sidebarContent').css({ "background-color": "#2a3632" }) // 取消隐藏预览PDF页面的隐藏功能(例如下载等功能,打印没法用除外) waitForKeyElements('button[hidden]:not(#print)', function (element) { if (element.attr('hidden') !== undefined) { element.removeAttr('hidden'); } }); // 根据链接判断是教材才显示切换版本按钮 if (file.includes("edu_product/esp/assets/") || file.includes("edu_product/esp/assets_document/")) { let contentIdMatch = file.match(/([a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12})/i); let contentId = contentIdMatch ? contentIdMatch[1] : null; let canFound = false if (!contentId) return; await getFileSize(`https://s-file-1.ykt.cbern.com.cn/zxx/ndrv2/resources/tch_material/details/${contentId}.json`) .then(() => { canFound = true }) .catch(() => { canFound = false }); if (!canFound) return; // 教材信息 let data = await Promise.any(["1", "2", "3"].map(async base => { try { let data = await request({ url: `https://s-file-${base}.ykt.cbern.com.cn/zxx/ndrv2/resources/tch_material/details/${contentId}.json`, headers: { 'Cache-Control': 'no-cache' } }); if (data.toString().startsWith(' item.ti_format === "pdf")?.ti_storages?.[base - 1].replace(/(r\d)-ndr-\w+/, '$1-ndr'), }; return item } catch (error) { console.error('【国家中小学智慧教育平台助手】\n获取数据时发生错误:', error); } })); console.log("【国家中小学智慧教育平台助手】即时\nPDF预览:书籍数据加载成功\n", data) let version let fileName = decodeURIComponent(file.substring(file.lastIndexOf('/') + 1)) let newFileName = decodeURIComponent(data.url.substring(data.url.lastIndexOf('/') + 1)) if (file.includes('edu_product/esp/assets_document/')) { // 先版 -> 中版 file = file.replace('edu_product/esp/assets_document/', 'edu_product/esp/assets/'); version = '中版'; } else if (file.includes('edu_product/esp/assets/')) { if (fileName === 'pdf.pdf' && fileName !== newFileName) { // 中版 -> 新版 file = file.replace('pdf.pdf', encodeURIComponent(newFileName)); version = '新版'; } else if (fileName !== 'pdf.pdf' && fileName === newFileName) { // 验证是否有先版教材 await getFileSize(file.replace('edu_product/esp/assets/', 'edu_product/esp/assets_document/').replace(newFileName, 'pdf.pdf')).then(() => { // 新版 -> 先版 file = file.replace('edu_product/esp/assets/', 'edu_product/esp/assets_document/'); file = file.replace(newFileName, 'pdf.pdf'); version = '先版'; }).catch(() => { // 新版 -> 中版 file = file.replace(encodeURIComponent(newFileName), 'pdf.pdf').replace(newFileName, 'pdf.pdf'); version = '中版'; }); } } let url = new URL(`${location.origin}${location.pathname}?file=${file}&disableAutoFetch=true&page=1`).toString() getFileSize(file).then(size => { let button = $(` `) button.on('click', function () { if (url !== location.href) location.replace(url); }) $("#sidebarToggle").after(button) }); } } let lastTitle setInterval(() => { const linkText = $('.fish-breadcrumb-link').last().text(); const currentTitle = $('title').text(); if (linkText && currentTitle && lastTitle !== linkText && currentTitle === '国家中小学智慧教育平台') { $('title').html(`${linkText} - 国家中小学智慧教育平台`); lastTitle = linkText } else if (lastTitle !== linkText && currentTitle === '国家中小学智慧教育平台') { $('title').html(`国家中小学智慧教育平台`); lastTitle = linkText } }, 1000); function Time() { function repair(i) { if (i >= 0 && i <= 9) { return "0" + i; } else { return i; } } var date = new Date(); var year = date.getFullYear(); var month = repair(date.getMonth() + 1); var day = repair(date.getDate()); var hours = repair(date.getHours()); var minute = repair(date.getMinutes()); var second = repair(date.getSeconds()); var curTime = year + "年 - " + month + "月 - " + day + "日 " + hours + "时 : " + minute + "分 : " + second + "秒"; return curTime; } function Greeting() { var date = new Date(); var hour = date.getHours(); var greeting = ''; if (hour >= 0 && hour <= 10) { greeting = '早上好'; } else if (hour > 10 && hour <= 14) { greeting = '中午好'; } else if (hour > 14 && hour <= 18) { greeting = '下午好'; } else if (hour > 18 && hour <= 24) { greeting = '晚上好'; } return greeting; } async function Poem(element) { try { let data = await request({ data: "", url: "https://v1.hitokoto.cn/?encode=json&c=i", headers: { "origin": "https://hitokoto.cn", "referer": "https://hitokoto.cn", 'Cache-Control': 'no-cache' } }) // 使用数据更新页面元素 element.text(`「${data.hitokoto}」${data.from_who || ""}`); element.css({ "cursor": "pointer" }); } catch (error) { // 处理错误情况 console.error('获取诗词时发生错误:', error); element.text('诗词加载失败'); element.css({ "cursor": "pointer" }); } } function request(option) { return new Promise((resolve, reject) => { let httpRequest = typeof GM_xmlhttpRequest !== "undefined" ? GM_xmlhttpRequest : GM.xmlHttpRequest; httpRequest({ method: 'get', ...option, onload: (response) => { if (response.responseText === undefined) return; let res try { res = JSON.parse(response.responseText); } catch (e) { res = response.response; } resolve(res); }, onerror: (error) => { resolve(null); }, }); }); } function addStyle(id, tag, css) { tag = tag || 'style'; let doc = document, styleDom = doc.getElementById(id); if (styleDom) styleDom.remove(); let style = doc.createElement(tag); style.rel = 'stylesheet'; style.id = id; tag === 'style' ? style.innerHTML = css : style.href = css; $('body').before(style) } function addDownStyle() { const url = new URL(location); let deColor = url.searchParams.get("x-edu-theme") === 'dark' ? "#0e9aff" : "#1e62ec"; let hvColor = url.searchParams.get("x-edu-theme") === 'dark' ? "#1e62ec" : "#4079ef"; let bgColor = url.searchParams.get("x-edu-theme") === 'dark' ? "rgba(42, 54, 50, 0.5)" : "rgba(255, 255, 255, 0.5)"; let spColor = url.searchParams.get("x-edu-theme") === 'dark' ? "#323e3a" : "#eee"; let txColor = url.searchParams.get("x-edu-theme") === 'dark' ? "#fff" : "#666"; let css = ` a, button, ul li, ui, label, input, select { transition: all 0.25s !important; -webkit-transition: all 0.25s !important; } [class^="index-module_header-wrap_"], [class*="index-module_header"] { height: auto !important; } [class^="index-module_header"] [class^="index-module_nav-normal"], [class^="index-module_header-wrap_"] [class^="index-module_header"] .theme-menu-sticky { height: auto !important; -webkit-backdrop-filter: blur(15px); backdrop-filter: blur(15px); background-color: ${bgColor}; } .course-document .viewerContainer { background-color: ${spColor} !important; } #smartEduTime{ margin: 0 5px; color: ${deColor}; } .index-module_tip { cursor: default; text-align: center; display: inline-flex; justify-content: center; align-items: center; margin: 10px 0; width: 100%; transition: all 0.3s !important; -webkit-transition: all 0.3s !important; } .index-module_tip img, .index-module_tip svg { margin: 0 5px; width: 20px; height: 20px; } .index-module_tip svg { color: ${deColor}; cursor: pointer; } .index-module_tip svg:hover { color: ${hvColor}; transition: all 0.3s !important; -webkit-transition: all 0.3s !important; } .index-module_pdf-control { display: flex; justify-content: center; position: absolute; top: 5%; pointer-events: none; width: auto !important; height: auto !important; } .index-module_pdf-control.left { left: 3%; } .index-module_pdf-control.right { right: 3%; } .index-module_pdf-button { background-color: ${deColor}cc !important; text-align: center !important; color: #fff !important; padding: 7px 20px !important; border-radius: 144.889px !important; border: 0 !important; margin: 0 5px !important; display: inline-flex; vertical-align: middle !important; pointer-events: all !important; transition: all 0.3s !important; -webkit-transition: all 0.3s !important; } .index-module_pdf-button:hover,.index-module_pdf-button:disabled { background-color: ${hvColor}cc !important; } .index-module_pdf-button svg { height: 18px !important; width: 17px !important; margin-right: 5px !important; vertical-align: middle !important; fill: #fff !important; } .index-module_item { cursor: pointer; font-size: 15px; -webkit-user-select: none; -ms-user-select: none; user-select: none; vertical-align: middle; display: flex; align-items: center; padding: 5px 0; justify-content: center; border-bottom: 1px solid ${spColor}; flex: 1; transition: all 0.3s !important; -webkit-transition: all 0.3s !important; } .index-module_item svg { height: 20px; width: 20px; margin-right: 4px; vertical-align: middle; } .index-module_item:hover { color: #fff; background-color: ${deColor}; } .index-module_download_tool_bar { display: flex !important; color: #fff !important; border: none !important; border-radius: 50px !important; white-space: nowrap !important; cursor: pointer !important; padding: 2px 9px 4px 9px !important; background-color: ${deColor} !important; transition: all 0.3s !important; -webkit-transition: all 0.3s !important; } .index-module_download_tool_bar:hover { background-color: ${hvColor} !important; } .index-module_download_tool_bar:disabled { background-color: ${hvColor} !important; } .index-module_download_tool_bar svg { margin-right: 3px !important; } .index-module_audio-player-control { width: 40px !important; height: 40px !important; display: inline-flex; justify-content: center !important; align-items: center !important; color: #fff !important; border-radius: 50px !important; cursor: pointer !important; background-color: ${deColor} !important; transition: all 0.3s !important; -webkit-transition: all 0.3s !important; } .index-module_audio-player-control:hover { background-color: ${hvColor} !important; } #listTitle { color: ${txColor}!important; } [class*="audioList-module_audio-list"] [class*="audioList-module_audio-item-active"] [class*="audioList-module_center"], [class*="audioList-module_audio-list"] [class*="audioList-module_audio-item-active"] [class*="audioList-module_right"] { color: ${deColor}; } div .fullscreen-dom { position: fixed !important; top: 0 !important; left: 0 !important; width: 100% !important; height: 100% !important; z-index: 99999 !important; } ::-webkit-scrollbar { width: 10px; height: 10px; background-color: transparent; z-index: 999999999999999; } ::-webkit-scrollbar-thumb { background: ${deColor}; border-radius: 50px; z-index: 999999999999999; } ::-webkit-scrollbar-thumb:hover { background: ${hvColor}; z-index: 999999999999999; } .no-scroll { overflow: hidden !important; } .no-select { user-select: none; -webkit-user-select: none; -moz-user-select: none; } @keyframes spin { 0% { transform: rotateZ(0) } 100% { transform: rotateZ(360deg) } } .tch-loading { position: absolute; top: 45%; left: 50%; transform: translateX(-50%); text-align: center; background: radial-gradient(29% 54% at 84% 95%,#f1f1f5 28%,#eff2f9 77%); padding: 45px 50px 40px; border-radius: 20px; box-shadow: 0 0 20px 5px rgba(0,0,0,0.5); margin-top: -47px!important; z-index: 989; -webkit-backdrop-filter: blur(15px); backdrop-filter: blur(15px); } .tch-loading img { height: 48px; width: 48px; border-radius: 50%; animation: spin 0.5s linear infinite } .tch-loading p { color: #232323; font-size: 14px; line-height: 22px; margin: 0 } .tch-loading p:nth-child(2) { margin: 24px 0 0 } /* 修复 PDFJS 预览界面搜索框的位置 */ html[dir=ltr] .findbar { right: 166px; } html[dir=rtl] .findbar { left: 166px; } @media (max-width: 770px) { html[dir=ltr] .findbar { right: 50px } html[dir=rtl] .findbar { left: 50px } } `; addStyle('BasicSmartEdu-User', 'style', css); return deColor } function play(id, url, urls, cover, actionFunction) { if (id && url) { var art = new Artplayer({ id: id, container: '.index-module_art-player', url: url, volume: 1, autoPlayback: true, customType: { m3u8: function (video, url, art) { if (Hls.isSupported()) { if (art.hls) art.hls.destroy(); const hls = new Hls(); hls.loadSource(url); hls.attachMedia(video); art.hls = hls; art.on('destroy', () => hls.destroy()); } else if (video.canPlayType('application/vnd.apple.mpegurl')) { video.src = url; } else { art.notice.show = 'Unsupported playback format: m3u8'; } }, }, controls: [ { name: 'goodRing', index: 21, position: 'left', html: '', tooltip: '好评', click: function () { GM_openInTab('https://greasyfork.org/scripts/459404/feedback', { active: true }); }, } ], icons: { state: '', indicator: '', play: '', pause: '', setting: '', fullscreenOn: '', fullscreenOff: '', playbackRate: '', aspectRatio: '', check: '', flip: '', arrowUp: '', arrowDown: '', arrowLeft: '', arrowRight: '', download: '', info: '', close: '', }, setting: true, theme: addDownStyle(), hotkey: true, autoMini: true, flip: true, playbackRate: true, aspectRatio: true, miniProgressBar: true, fullscreen: true, fullscreenWeb: true, }) urls ? (art.quality = [...urls]) : "" var contextmenuStyle = { "display": "flex", "justify-content": "center", "align-items": "center", "border-bottom": "none" } art.contextmenu.add({ name: 'appTitle', index: 1, html: `国家智慧教育公共服务平台助手`, style: contextmenuStyle, click: function () { GM_openInTab('https://greasyfork.org/scripts/459404/feedback', { active: true }); }, }) art.contextmenu.update({ name: 'version', index: 2, html: `Artplayer Ultra ${Artplayer.version}`, click: function () { GM_openInTab('https://artplayer.org/', { active: true }); }, style: contextmenuStyle }) art.contextmenu.update({ name: 'info', index: 40, html: `${$(art.icons.info).html()}${art.i18n.language["Video Info"]}`, style: contextmenuStyle }) art.contextmenu.update({ name: 'close', index: 50, html: `${$(art.icons.close).html()}${art.i18n.language["Close"]}`, style: contextmenuStyle }) art.on('fullscreen', () => { $('#header').fadeToggle(); $(art.controls.fullscreenWeb).toggle(); $(art.controls.hideList).toggle(); $(art.controls.goodRing).toggle(); if (!$('body').hasClass('no-scroll')) { $('body').addClass('no-scroll'); } else { $('body').removeClass('no-scroll'); } }); art.on('fullscreenWeb', () => { $('#header').fadeToggle(); $(art.controls.fullscreen).toggle(); $(art.controls.hideList).toggle(); $(art.controls.goodRing).toggle(); if (!$('body').hasClass('no-scroll')) { $('body').addClass('no-scroll'); } else { $('body').removeClass('no-scroll'); } }); $(art.template.$container).find(".icon").removeClass("icon") addStyle('Artplayer-User', 'style', ` .art-contextmenu svg, .art-contextmenu img { vertical-align: top; margin-right: 5px } `); cover ? (art.poster = cover) : "" actionFunction ? actionFunction(art) : "" } else { document.querySelector('.index-module_art-player').innerHTML = `

    抱歉

    此媒体未满足所需参数

    可能是脚本出了问题

    请登录后关闭脚本查看

    ` } } // 格式化文件大小函数 function formatFileSize(bytes) { if (bytes === 0) return '0 Bytes'; const k = 1024; const sizes = ['B', 'KB', 'MB', 'GB', 'TB']; const i = Math.floor(Math.log(bytes) / Math.log(k)); return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i]; } // 获取文件大小函数 function getFileSize(url) { return new Promise((resolve, reject) => { fetch(url, { method: 'HEAD' }) .then(response => { if (!response.ok) throw new Error('找不到该文件'); let contentLength = response.headers.get('Content-Length'); resolve(parseInt(contentLength)); }) .catch(error => { reject(error); }) }); } /*--- waitForKeyElements(): 一个实用函数,用于 Greasemonkey 脚本, 它可以检测和处理AJAX加载的内容,且支持在使用 `shadowRoot` 的页面上运行。 此外,该函数允许多次调用时对相同的元素执行独立的操作,避免前后调用之间的干扰。 使用示例: base.waitForKeyElements ( "div.comments", commentCallbackFunction ); // 页面特定的函数,用于在找到节点时执行我们想要的操作。 function commentCallbackFunction (jNode) { jNode.text("waitForKeyElements() 更改了这段注释。"); } 重要提示:这个函数需要你的脚本加载了jQuery。 更新说明: - 每次调用时使用独立的标识符来控制对元素的处理,支持多次调用时重复执行操作, 即使是同一个元素也能在不同的调用中执行操作,而不会受之前调用的影响。 */ var uniqueIdCounter = 0; function waitForKeyElements(selectorTxt, actionFunction, bWaitOnce, iframeSelector) { function findInShadowRoots(root, selector) { let elements = $(root).find(selector).toArray(); $(root).find('*').each(function () { const shadowRoot = this.shadowRoot; if (shadowRoot) { elements = elements.concat(findInShadowRoots(shadowRoot, selector)); } }); return elements; } var targetElements; if (iframeSelector) { targetElements = $(iframeSelector).contents(); } else { targetElements = $(document); } let allElements = findInShadowRoots(targetElements, selectorTxt); if (allElements.length > 0) { allElements.forEach(function (element) { var jThis = $(element); //var uniqueIdentifier = 'alreadyFound_' + uniqueIdCounter; var uniqueIdentifier = 'alreadyFound'; var alreadyFound = jThis.data(uniqueIdentifier) || false; if (!alreadyFound) { var cancelFound = actionFunction(jThis); if (cancelFound) { return false; } else { jThis.data(uniqueIdentifier, true); } } }); } var controlObj = waitForKeyElements.controlObj || {}; var controlKey = selectorTxt.replace(/[^\w]/g, "_"); var timeControl = controlObj[controlKey]; if (allElements.length > 0 && bWaitOnce && timeControl) { clearInterval(timeControl); delete controlObj[controlKey]; } else { if (!timeControl) { timeControl = setInterval(function () { waitForKeyElements(selectorTxt, actionFunction, bWaitOnce, iframeSelector); }, 500); controlObj[controlKey] = timeControl; uniqueIdCounter++; } } waitForKeyElements.controlObj = controlObj; } function getIcon(character) { let icon_document = ``; let icon_video = ``; let icon_audio = ``; let icon_courseVideo = ``; let icon_courseWare = ``; let icon_coursePlan = ``; let icon_courseTask = ``; let icon_coursePractice = ``; const iconMap = { '文档': icon_document, '视频': icon_video, '音频': icon_audio, '视频课程': icon_courseVideo, '课件': icon_courseWare, '教学设计': icon_coursePlan, '学习任务单': icon_courseTask, '课后练习': icon_coursePractice }; return iconMap[character] || icon_document; } $("head").before($(` `)) })(); // “我是中国人。我们都是中国人。中华民族是一家。”