中原iLearning 2.0 頁面體驗增強

在中原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);
})();