您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
在中原iLearning 2.0 課程頁依種類分類課程段落,提供新分頁開啟PDF教材的功能,提供下載影片的功能
// ==UserScript== // @name 中原iLearning 2.0 頁面體驗增強 // @namespace http://tampermonkey.net/ // @version 2.1 // @description 在中原iLearning 2.0 課程頁依種類分類課程段落,提供新分頁開啟PDF教材的功能,提供下載影片的功能 // @icon  // @match *://ilearning.cycu.edu.tw/* // @grant none // @run-at document-end // @license MIT // ==/UserScript== (function () { 'use strict'; const config = { "討論區": { "title": "討論區", "logo": "https://ilearning.cycu.edu.tw/theme/image.php/boost_union/forum/1744246650/monologo?filtericon=1" }, "作業": { "title": "作業", "logo": "https://ilearning.cycu.edu.tw/theme/image.php/boost_union/assign/1744246650/monologo?filtericon=1" }, "檔案": { "title": "檔案", "logo": "https://ilearning.cycu.edu.tw/theme/image.php/boost_union/resource/1757814017/monologo" }, "PDF Annotation": { "title": "PDF檔", "logo": "https://ilearning.cycu.edu.tw/theme/image.php/boost_union/pdfannotator/1744246650/monologo?filtericon=1" }, "超級影片": { "title": "影片檔", "logo": "https://ilearning.cycu.edu.tw/theme/image.php/boost_union/supervideo/1744246650/monologo?filtericon=1" }, "網址": { "title": "網址", "logo": "https://ilearning.cycu.edu.tw/theme/image.php/boost_union/url/1757814017/monologo?filtericon=1" }, "回饋單": { "title": "問卷", "logo": "https://ilearning.cycu.edu.tw/theme/image.php/boost_union/feedback/1744246650/monologo?filtericon=1" }, "頁面": { "title": "文章", "logo": "https://ilearning.cycu.edu.tw/theme/image.php/boost_union/page/1757814017/monologo?filtericon=1" } }; const order = ["頁面", "討論區", "作業", "PDF Annotation", "檔案", "超級影片", "網址", "回饋單"]; function extractFullUrl() { const scripts = document.scripts; for (let script of scripts) { const match = script.textContent.match(/"fullurl":\s*"([^"]+)"/); if (match) { return match[1].replace(/\\/g, ''); // 去除反斜線 } } return null; } function createDownloadButton(fullUrl) { const button = document.createElement('button'); button.id = 'pdf-download-btn'; button.title = '下載 PDF'; button.innerHTML = '➜'; // 下載符號 button.onclick = function () { window.open(fullUrl, '_blank'); }; Object.assign(button.style, { position: 'fixed', bottom: '8rem', right: '2rem', width: '36px', height: '36px', backgroundColor: 'orange', color: 'white', border: 'none', borderRadius: '50%', fontSize: '26px', padding: '0 0 2px 0', display: 'flex', alignItems: 'center', justifyContent: 'center', cursor: 'pointer', boxShadow: '0px 4px 6px rgba(0, 0, 0, 0.1)', transform: 'rotate(90deg)' }); document.body.appendChild(button); } function getCookie(name) { const value = `; ${document.cookie}`; const parts = value.split(`; ${name}=`); if (parts.length === 2) return parts.pop().split(';').shift(); return null; } function setCookie(name, value, days = 30) { const date = new Date(); date.setTime(date.getTime() + (days * 24 * 60 * 60 * 1000)); document.cookie = `${name}=${value};expires=${date.toUTCString()};path=/`; } function getMoremenu() { const ul = document.querySelector('ul.nav.more-nav.nav-tabs'); if (ul) { const li = document.createElement('li'); li.className = 'nav-item'; const a = document.createElement('a'); a.className = 'nav-link'; a.setAttribute('role', 'menuitem'); a.setAttribute('style', 'color:#FF359A !important;'); a.textContent = '★精簡化'; a.onclick = showMenu; li.appendChild(a); ul.appendChild(li); } } function getItems() { const result = {}; // 取得課程 ID(從網址中抓取 ?id= 後的數字) const courseIdMatch = window.location.href.match(/course\/view\.php\?id=(\d+)/); const courseId = courseIdMatch ? courseIdMatch[1] : null; if (!courseId) return result; let sections = []; let main_section = {}; for (let i = 0; i < sessionStorage.length; i++) { const key = sessionStorage.key(i); if (key.includes(`${courseId}/staticState`)) { try { const json = JSON.parse(sessionStorage.getItem(key)); if (json) { if (Array.isArray(json.section)) { sections = json.section; for (const sec of json.section) { main_section[sec.id] = sec.parentsectionid || sec.id; } } console.log(main_section); if (Array.isArray(json.cm)) { for (const item of json.cm) { // 跳過子單元 if (item.modname === "子單元") continue; // 處理子單元歸位 item.sectionid = main_section[item.sectionid]; item.sectionnumber = item.sectionnumber = sections.findIndex(section => section.id === item.sectionid); const mod = item.modname; if (!result[mod]) result[mod] = []; result[mod].push(item); } } console.log(result); } } catch (e) { console.error('JSON 解析錯誤:', e); } } } // 按照指定順序排序 const sortedResult = {}; for (const mod of order) { if (result[mod]) { sortedResult[mod] = result[mod]; } } // 添加未在order中定義的類型 for (const mod in result) { if (!sortedResult[mod]) { sortedResult[mod] = result[mod]; } } // 按照周次排序 for (const mod in sortedResult) { if (mod === "討論區") { sortedResult[mod].sort((a, b) => { if (a.sectionnumber === b.sectionnumber) { return b.id - a.id; } if (a.sectionnumber === Math.min(...sortedResult[mod].map(i => i.sectionnumber))) return -1; if (b.sectionnumber === Math.min(...sortedResult[mod].map(i => i.sectionnumber))) return 1; return b.sectionnumber - a.sectionnumber; }); } else { sortedResult[mod].sort((a, b) => b.sectionnumber - a.sectionnumber); } } return { items: sortedResult, sections }; } function showMenu() { const original = document.querySelector('ul.weeks'); const side = document.querySelector('#menuside'); if (original) { original.style.display = 'none'; const data = getItems(); const items = data.items; const sections = data.sections; let container = null; if (side) { container = side; container.innerHTML = ""; } else { container = document.createElement('ul'); container.className = 'weeks'; container.id = 'menuside'; container.setAttribute('data-for', 'course_sectionlist'); } // 添加排序選擇器 const sortContainer = document.createElement('div'); sortContainer.className = 'mb-3'; sortContainer.innerHTML = ` <select class="form-select" id="week-sort-order"> <option value="desc">降序</option> <option value="asc">升序</option> </select> `; // 設置默認值 const savedOrder = getCookie('weekSortOrder') || 'desc'; sortContainer.querySelector('#week-sort-order').value = savedOrder; // 監聽變化 sortContainer.querySelector('#week-sort-order').addEventListener('change', function () { setCookie('weekSortOrder', this.value); sortContainer.innerHTML = ""; showMenu(); // 重新渲染 }); let sectionNum = 1; // 獲取當前周次 const currentWeekSection = sections.find(s => s.current); for (const modname in items) { const section = document.createElement('li'); section.className = 'section course-section main clearfix'; section.id = `side-section-${sectionNum}`; // 按照選擇的排序方式處理周次 const weekItems = {}; for (const item of items[modname]) { if (!weekItems[item.sectionnumber]) { weekItems[item.sectionnumber] = []; } weekItems[item.sectionnumber].push(item); } console.log(items) console.log(weekItems) // 獲取所有周次並排序 let weekNumbers = Object.keys(weekItems).map(Number); // 第0周始終置頂 const week0 = weekNumbers.includes(0) ? [0] : []; const otherWeeks = weekNumbers.filter(w => w !== 0); // 按照選擇的排序方式排序 const sortOrder = getCookie('weekSortOrder') || 'asc'; if (sortOrder === 'asc') { otherWeeks.sort((a, b) => a - b); } else { otherWeeks.sort((a, b) => b - a); } // 將當前周次提前 let sortedWeeks = week0; if (currentWeekSection && otherWeeks.includes(currentWeekSection.sectionnumber)) { sortedWeeks.push(currentWeekSection.sectionnumber); sortedWeeks = sortedWeeks.concat(otherWeeks.filter(w => w !== currentWeekSection.sectionnumber)); } else { sortedWeeks = sortedWeeks.concat(otherWeeks); } let sectionHTML = ` <div class="section-item"> <div class="course-section-header d-flex"> <div class="d-flex align-items-center position-relative"> <a role="button" class="btn btn-icon me-3 icons-collapse-expand justify-content-center collapsed" aria-expanded="false" href="#side-coursecontentcollapse${sectionNum}" data-for="sectiontoggler" data-toggle="collapse"> <span class="expanded-icon icon-no-margin p-2" title="展延"> <i class="icon fa fa-chevron-down fa-fw" aria-hidden="true"></i> <span class="sr-only">展延</span> </span> <span class="collapsed-icon icon-no-margin p-2" title="展延"> <span class="dir-rtl-hide"><i class="icon fa fa-chevron-right fa-fw" aria-hidden="true"></i></span> <span class="dir-ltr-hide"><i class="icon fa fa-chevron-left fa-fw" aria-hidden="true"></i></span> <span class="sr-only">展延</span> </span> </a> <h3 class="h4 sectionname course-content-item d-flex align-self-stretch align-items-center mb-0" id="side-sectionid-${sectionNum}-title"> ${config[modname]?.title || modname} </h3> </div> </div> <div id="side-coursecontentcollapse${sectionNum}" class="content course-content-item-content collapse"> <div class="my-3" data-for="sectioninfo"> <div class="section_availability"></div> </div> <ul class="section m-0 p-0 img-text d-block" data-for="cmlist">`; for (const week of sortedWeeks) { const weekItemsList = weekItems[week]; if (!weekItemsList) continue; let weekTitle = `第${week}週`; if (week === 0) { weekTitle = "公告"; } // 添加當前周標記 let currentBadge = ""; let currentCSS = ""; if (currentWeekSection.id == weekItemsList[0].sectionid) { currentBadge = '<span class="badge rounded-pill bg-primary text-white order-1 ms-2">本週</span>'; currentCSS = "course-content current"; } sectionHTML += ` <li class="activity activity-wrapper pdfannotator modtype_pdfannotator hasinfo" data-indexed="true"> <div class="week-title fw-bold fs-5 mb-2">${weekTitle}${currentBadge}</div><div class="${currentCSS}">`; for (const item of weekItemsList) { // 確定圖標 let logoUrl = config[modname]?.logo; if (modname === "PDF Annotation") { logoUrl = config["PDF Annotation"].logo; } sectionHTML += `<div class="activity-item focus-control mb-2"> <div class="activity-grid"> <div class="activity-icon activityiconcontainer smaller communication courseicon align-self-start me-2"> ${logoUrl ? `<img src="${logoUrl}" class="activityicon">` : ''} </div> <div class="activity-name-area activity-instance d-flex flex-column me-2"> <div class="activitytitle modtype_pdfannotator position-relative align-self-start"> <div class="activityname"> <a href="${item.url}" class="aalink stretched-link"> <span class="instancename">${item.name}</span> </a> </div> </div> </div> </div> </div>`; } sectionHTML += `</div></li>`; } sectionHTML += `</ul> </div> </div>`; section.innerHTML = sectionHTML; container.appendChild(section); sectionNum++; } original.parentNode.insertBefore(container, original.nextSibling); original.parentNode.insertBefore(sortContainer, original.nextSibling); } } function handleVideoPage() { // 查找MP4連結 const mp4Regex = /https:\/\/[^"]+\.mp4/g; let mp4Url = null; let doc = document.documentElement.innerHTML; const match = doc.match(mp4Regex); if (match && match.length > 0) { mp4Url = match[0]; } if (mp4Url) { // 創建下載按鈕 const button = document.createElement('button'); button.id = 'video-download-btn'; button.title = '下載影片'; button.innerHTML = '➜'; button.onclick = function () { const link = document.createElement('a'); link.href = mp4Url; link.download = ''; document.body.appendChild(link); link.click(); document.body.removeChild(link); if (confirm('停留此頁面將降低下載速率,是否離開此頁面?')) { window.location.href = 'https://ilearning.cycu.edu.tw/my/courses.php'; } else { } }; Object.assign(button.style, { position: 'fixed', bottom: '8rem', right: '2rem', width: '36px', height: '36px', backgroundColor: 'blue', color: 'white', border: 'none', borderRadius: '50%', fontSize: '26px', padding: '0 0 2px 0', display: 'flex', alignItems: 'center', justifyContent: 'center', cursor: 'pointer', boxShadow: '0px 4px 6px rgba(0, 0, 0, 0.1)', transform: 'rotate(90deg)' }); document.body.appendChild(button); } } function init() { const url = window.location.href; if (/^https:\/\/ilearning\.cycu\.edu\.tw\/mod\/pdfannotator\//.test(url)) { const fullUrl = extractFullUrl(); if (fullUrl) { createDownloadButton(fullUrl); } } else if (/^https:\/\/ilearning\.cycu\.edu\.tw\/(mod\/supervideo\/|mod\/resource\/)/.test(url)) { console.log("找尋影片網址中..."); handleVideoPage(); } else if (/^https:\/\/ilearning\.cycu\.edu\.tw\/course\//.test(url)) { console.log("找尋影片網址中..."); getMoremenu(); } } window.addEventListener('load', init); })();