Teachable 課程章節收藏

在 Teachable 課程側邊欄為每個章節新增收藏按鈕,並提供篩選功能

// ==UserScript==
// @name         Teachable 課程章節收藏
// @namespace    https://gist.github.com/Aya-X
// @version      1.0.0
// @description  在 Teachable 課程側邊欄為每個章節新增收藏按鈕,並提供篩選功能
// @author       Aya
// @match        *://*.teachable.com/courses/*
// @match        *://*.hexschool.com/courses/*
// @grant        GM_addStyle
// @icon         
// @license      MIT
// ==/UserScript==

/* eslint-env browser, greasemonkey */

(function () {
    'use strict';

    // === 事件發送器 ===
    /**
     * @class EventEmitter
     * @description 事件發送器實作
     */
    class EventEmitter {
        constructor() {
            this.events = {};
        }

        on(event, callback) {
            if (!this.events[event]) this.events[event] = [];
            this.events[event].push(callback);
        }

        emit(event, ...args) {
            if (!this.events[event]) return;
            this.events[event].forEach(callback => callback(...args));
        }

        off(event, callback) {
            if (!this.events[event]) return;
            this.events[event] = this.events[event].filter(cb => cb !== callback);
        }
    }

    // === 設定用類別 ===
    /**
     * @class Config
     * @description 儲存腳本所需的設定
     */
    class Config {
        static LESSON_ITEM_SELECTOR = 'li.section-item';
        static LESSON_ID_ATTRIBUTE = 'data-lecture-id';
        static STORAGE_KEY = 'teachable_favorite_lessons';
        static BUTTON_CONTAINER_SELECTOR = '.lecture-left';
        static COURSE_SECTION_SELECTOR = '.course-section';
        static COURSE_SIDEBAR_SELECTOR = '.course-sidebar';
        static FILTER_CLASS_NAME = 'favorite-filter-active';

        static ICONS = {
            OUTLINED: `<svg xmlns="http://www.w3.org/2000/svg" height="20px" viewBox="0 0 24 24" width="20px" fill="#555555"><path d="M0 0h24v24H0V0z" fill="none"/><path d="M17 3H7c-1.1 0-2 .9-2 2v16l7-3 7 3V5c0-1.1-.9-2-2-2zm0 15l-5-2.18L7 18V5h10v13z"/></svg>`,
            FILLED: `<svg xmlns="http://www.w3.org/2000/svg" height="20px" viewBox="0 0 24 24" width="20px" fill="#FB8C00"><path d="M0 0h24v24H0V0z" fill="none"/><path d="M17 3H7c-1.1 0-2 .9-2 2v16l7-3 7 3V5c0-1.1-.9-2-2-2z"/></svg>`,
            FILTER: `<svg xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 0 24 24" width="24"><path d="M0 0h24v24H0V0z" fill="none"/><path d="M10 18h4v-2h-4v2zM3 6v2h18V6H3zm3 7h12v-2H6v2z"/></svg>`
        };

        static STYLES = `
            .teachable-course ${Config.LESSON_ITEM_SELECTOR} a.item {
                position: relative;
                display: flex;
                align-items: center;
                width: 100%;
            }
            .favorite-btn-container {
                position: absolute;
                right: 8px;
                top: 50%;
                transform: translateY(-50%);
                z-index: 10;
            }
            .favorite-btn {
                background: none;
                border: none;
                cursor: pointer;
                padding: 4px;
                border-radius: 50%;
                display: inline-flex;
                justify-content: center;
                align-items: center;
                transition: background-color 0.2s;
            }
            .favorite-btn:hover {
                background-color: rgba(0, 0, 0, 0.08);
            }
            .teachable-course ${Config.LESSON_ITEM_SELECTOR} .title-container {
                padding-right: 35px;
            }
            .favorite-filter-icon {
                display: inline-block;
                vertical-align: middle;
                line-height: 1;
                cursor: pointer;
                padding: 0 4px;
            }
            .favorite-filter-icon svg {
                display: block;
                fill: #ffffff;
                transition: fill 0.2s;
            }
            .favorite-filter-icon:hover svg {
                fill: #555555;
            }
            .favorite-filter-icon.active svg {
                fill: #FB8C00;
            }
            .${Config.FILTER_CLASS_NAME} ${Config.LESSON_ITEM_SELECTOR}:not(.is-favorite),
            .${Config.FILTER_CLASS_NAME} ${Config.COURSE_SECTION_SELECTOR}:not(.has-favorite) {
                display: none;
            }
        `;
    }

    // === 工具 ===
    /**
     * @class Utils
     * @description 提供通用的工具函式
     */
    class Utils {
        /**
         * 建立一個 Debounce 函式,該函式會延遲執行 func 函式,直到使用者停止觸發事件後的一段時間
         * @param {Function} func - 要進行 Debounce 的函式
         * @param {number} wait - 延遲的毫秒數
         * @returns {Function} 回傳新的 Debounce 化的函式
         */
        static debounce(func, wait) {
            let timeout;
            return function executedFunction(...args) {
                const later = () => {
                    clearTimeout(timeout);
                    func(...args);
                };
                clearTimeout(timeout);
                timeout = setTimeout(later, wait);
            };
        }
    }

    // === 資料管理類別 ===
    /**
     * @class FavoriteStorage
     * @description 處理收藏章節資料的載入、儲存和管理,使用 localStorage
     */
    class FavoriteStorage extends EventEmitter {
        /**
         * @param {string} [storageKey=Config.STORAGE_KEY] - 用於 localStorage 的鍵名
         */
        constructor(storageKey = Config.STORAGE_KEY) {
            super();
            this.storageKey = storageKey;
            this.favorites = new Set();
            this.load();
        }

        /**
         * 從 localStorage 載入收藏的章節 ID
         * @private
         */
        load() {
            try {
                const stored = localStorage.getItem(this.storageKey);
                if (stored) {
                    const parsedData = JSON.parse(stored);
                    this.favorites = new Set(Array.isArray(parsedData) ? parsedData : []);
                }
            } catch (error) {
                console.warn('載入收藏資料失敗:', error);
                this.favorites = new Set();
            }
        }

        /**
         * 將目前的收藏章節 ID 儲存到 localStorage
         * @private
         */
        save() {
            try {
                localStorage.setItem(this.storageKey, JSON.stringify(Array.from(this.favorites)));
            } catch (error) {
                console.error('儲存收藏資料失敗:', error);
            }
        }

        /**
         * 新增一個收藏章節
         * @param {string} lessonId - 章節 ID
         */
        add(lessonId) {
            this.favorites.add(lessonId);
            this.save();
            this.emit('change', { lessonId, action: 'add' });
        }

        /**
         * 移除一個收藏章節。
         * @param {string} lessonId - 章節 ID
         */
        remove(lessonId) {
            this.favorites.delete(lessonId);
            this.save();
            this.emit('change', { lessonId, action: 'remove' });
        }

        /**
         * 檢查某個章節是否已被收藏
         * @param {string} lessonId - 章節 ID
         * @returns {boolean} 如果已收藏則回傳 true,否則 false
         */
        has(lessonId) {
            return this.favorites.has(lessonId);
        }

        /**
         * 切換一個章節的收藏狀態
         * @param {string} lessonId - 章節 ID
         * @returns {boolean} 如果切換後為收藏狀態,回傳 true,否則 false
         */
        toggle(lessonId) {
            if (this.has(lessonId)) {
                this.remove(lessonId);
                return false;
            } else {
                this.add(lessonId);
                return true;
            }
        }

        /**
         * 取得所有收藏章節的 ID 列表
         * @returns {string[]} 收藏章節 ID 的陣列
         */
        getAll() {
            return Array.from(this.favorites);
        }

        /**
         * 取得收藏章節的總數
         * @returns {number} 收藏章節的數量
         */
        getCount() {
            return this.favorites.size;
        }

        /**
         * 清除所有收藏
         */
        clear() {
            this.favorites.clear();
            this.save();
            this.emit('clear');
        }
    }

    // === UI 類別 ===
    /**
     * @class FavoriteButton
     * @description 代表一個章節旁的收藏按鈕 UI 元件
     */
    class FavoriteButton extends EventEmitter {
        /**
         * @param {string} lessonId - 此按鈕對應的章節 ID
         * @param {FavoriteStorage} storage - 用於儲存狀態的 storage
         */
        constructor(lessonId, storage) {
            super();
            this.lessonId = lessonId;
            this.storage = storage;
            this.element = this.createElement();
            this.button = this.element.querySelector('.favorite-btn');
            this.updateState();
        }

        /**
         * 建立按鈕的 DOM 結構
         * @returns {HTMLDivElement} 包含按鈕的容器元素
         * @private
         */
        createElement() {
            const container = document.createElement('div');
            container.className = 'favorite-btn-container';
            const button = document.createElement('button');
            button.className = 'favorite-btn';
            button.addEventListener('click', this.handleClick.bind(this));
            container.appendChild(button);
            return container;
        }

        /**
         * 處理按鈕點擊事件
         * @param {MouseEvent} event - 點擊事件物件
         * @private
         */
        handleClick(event) {
            event.preventDefault();
            event.stopPropagation();
            const newState = this.storage.toggle(this.lessonId);
            this.updateState();
            this.emit('stateChanged', {
                lessonId: this.lessonId,
                isFavorited: newState
            });
        }

        /**
         * 根據收藏狀態更新按鈕的圖示、標題和 aria 屬性
         */
        updateState() {
            const isFavorited = this.storage.has(this.lessonId);
            this.button.innerHTML = isFavorited ? Config.ICONS.FILLED : Config.ICONS.OUTLINED;
            this.button.setAttribute('title', isFavorited ? '取消收藏' : '收藏此章節');
            this.button.setAttribute('aria-pressed', isFavorited.toString());
        }

        /**
         * 將按鈕附加到指定的元素
         * @param {HTMLElement} parentElement - 要附加到的元素
         */
        appendTo(parentElement) {
            parentElement.appendChild(this.element);
        }

        /**
         * 從 DOM 中移除此按鈕
         */
        remove() {
            if (this.element && this.element.parentNode) {
                this.element.parentNode.removeChild(this.element);
            }
        }

        /**
         * 檢查此按鈕對應的章節是否已收藏
         * @returns {boolean} 是否已收藏
         */
        isFavorited() {
            return this.storage.has(this.lessonId);
        }

        /**
         * 檢查按鈕元素是否仍在 DOM 中
         * @returns {boolean} 是否已連接到 DOM
         */
        isConnected() {
            return this.element && this.element.isConnected;
        }
    }

    // === 主要管理類別 ===
    /**
     * @class TeachableFavoriteManager
     * @description 腳本的主要控制器,管理收藏按鈕、篩選器和 DOM 監聽
     */
    class TeachableFavoriteManager extends EventEmitter {
        constructor() {
            super();
            this.storage = new FavoriteStorage();
            this.observer = null;
            this.lessonButtons = new Map();
            this.isFilterActive = false;
            this.filterButton = null;

            // 快取相關
            this._lessonItemsCache = null;
            this._cacheTimeout = null;
            this._observerTarget = null;

            this.debouncedProcess = Utils.debounce(() => {
                this.processExistingLessons();
                this.createFilterButton();
                if (this.isFilterActive) {
                    this.applyFavoriteFilter();
                }
            }, 300);

            // 監聽 storage 事件
            this.storage.on('change', () => {
                if (this.isFilterActive) {
                    this.applyFavoriteFilter();
                }
            });

            this.init();
        }

        /**
         * 初始化腳本,注入樣式、設定監聽器並處理現有章節
         * @private
         */
        init() {
            document.body.classList.add('teachable-course');
            this.injectStyles();
            this.createFilterButton();
            this.setupObserver();
            this.processExistingLessons();
        }

        /**
         * 注入腳本所需的 CSS 樣式
         * @private
         */
        injectStyles() {
            GM_addStyle(Config.STYLES);
        }

        /**
         * 取得章節項目,支援快取
         * @param {boolean} forceRefresh - 是否強制重新查詢
         * @returns {NodeList}
         */
        getLessonItems(forceRefresh = false) {
            if (!forceRefresh && this._lessonItemsCache) {
                return this._lessonItemsCache;
            }

            this._lessonItemsCache = document.querySelectorAll(Config.LESSON_ITEM_SELECTOR);

            // 5秒後清除快取
            clearTimeout(this._cacheTimeout);
            this._cacheTimeout = setTimeout(() => {
                this._lessonItemsCache = null;
            }, 5000);

            return this._lessonItemsCache;
        }

        /**
         * 設定 MutationObserver 來監聽課程列表的 DOM 變動,以動態新增或移除按鈕
         * @private
         */
        setupObserver() {
            // 優先找到更具體的容器
            this._observerTarget = document.querySelector(Config.COURSE_SIDEBAR_SELECTOR) ||
                document.querySelector('.course-content') ||
                document.body;

            const config = {
                childList: true,
                subtree: true,
                attributeFilter: ['data-lecture-id'] // 只監聽特定屬性的變化
            };

            this.observer = new MutationObserver((mutations) => {
                // 過濾相關的變動
                const relevantMutation = mutations.some(mutation => {
                    // 檢查是否有相關的節點變動
                    if (mutation.type === 'childList') {
                        // 檢查新增或移除的節點是否包含課程項目
                        return [...mutation.addedNodes, ...mutation.removedNodes].some(node => {
                            if (node.nodeType !== Node.ELEMENT_NODE) return false;
                            return node.matches && (
                                node.matches(Config.LESSON_ITEM_SELECTOR) ||
                                (node.querySelector && node.querySelector(Config.LESSON_ITEM_SELECTOR))
                            );
                        });
                    }
                    return false;
                });

                if (!relevantMutation) return;

                // 清除快取,因為 DOM 已改變
                this._lessonItemsCache = null;

                if (!document.querySelector(Config.BUTTON_CONTAINER_SELECTOR)) return;
                this.debouncedProcess();
            });

            this.observer.observe(this._observerTarget, config);
        }

        /**
         * 處理頁面上所有已存在的章節,為它們新增收藏按鈕
         */
        processExistingLessons() {
            const lessonItems = this.getLessonItems();
            this.cleanupDisconnectedButtons();

            lessonItems.forEach(item => {
                const lessonId = item.getAttribute(Config.LESSON_ID_ATTRIBUTE);
                if (!lessonId) return;

                // 如果按鈕已存在且仍在 DOM 中,則跳過
                if (this.lessonButtons.has(lessonId) && this.lessonButtons.get(lessonId).button.isConnected()) {
                    return;
                }

                const linkElement = item.querySelector('a.item');
                if (linkElement && linkElement.isConnected) {
                    this.addFavoriteButton(item, linkElement, lessonId);
                }
            });

            // 更新收藏狀態的 CSS 類別
            this.updateFavoriteClasses();
        }

        /**
         * 更新所有項目的收藏狀態 CSS 類別
         * @private
         */
        updateFavoriteClasses() {
            this.lessonButtons.forEach(({ item }, lessonId) => {
                item.classList.toggle('is-favorite', this.storage.has(lessonId));
            });

            // 更新區塊的收藏狀態
            const sections = document.querySelectorAll(Config.COURSE_SECTION_SELECTOR);
            sections.forEach(section => {
                const hasFavorite = section.querySelector('.is-favorite');
                section.classList.toggle('has-favorite', !!hasFavorite);
            });
        }

        /**
         * 為單一章節新增收藏按鈕。
         * @param {HTMLLIElement} item - 章節的 li 元素
         * @param {HTMLAnchorElement} linkElement - 章節的 a 連結元素
         * @param {string} lessonId - 章節 ID
         * @private
         */
        addFavoriteButton(item, linkElement, lessonId) {
            const favoriteButton = new FavoriteButton(lessonId, this.storage);

            // 監聽按鈕狀態改變事件
            favoriteButton.on('stateChanged', ({ lessonId, isFavorited }) => {
                item.classList.toggle('is-favorite', isFavorited);

                // 更新所屬區塊的狀態
                const section = item.closest(Config.COURSE_SECTION_SELECTOR);
                if (section) {
                    const hasFavorite = section.querySelector('.is-favorite');
                    section.classList.toggle('has-favorite', !!hasFavorite);
                }

                // 發送事件
                this.emit('favoriteChanged', { lessonId, isFavorited });
            });

            favoriteButton.appendTo(linkElement);
            const titleContainer = item.querySelector('.title-container');
            if (titleContainer) {
                titleContainer.style.paddingRight = '35px';
            }
            this.lessonButtons.set(lessonId, { item, button: favoriteButton });
        }

        /**
         * 清理已從 DOM 中移除的章節所對應的按鈕
         * @private
         */
        cleanupDisconnectedButtons() {
            for (const [lessonId, { item, button }] of this.lessonButtons.entries()) {
                if (!item.isConnected || !button.isConnected()) {
                    button.remove();
                    this.lessonButtons.delete(lessonId);
                }
            }
        }

        /**
         * 在課程側邊欄頂部建立「篩選收藏」按鈕
         */
        createFilterButton() {
            const container = document.querySelector(Config.BUTTON_CONTAINER_SELECTOR);
            if (container && !document.getElementById('favorite-filter-btn')) {
                const a = document.createElement('a');
                a.id = 'favorite-filter-btn';
                a.className = 'nav-icon-settings favorite-filter-icon';
                a.href = '#';
                a.setAttribute('role', 'button');
                a.title = '顯示已收藏章節';
                a.setAttribute('aria-label', '顯示已收藏章節');
                a.innerHTML = Config.ICONS.FILTER;

                a.addEventListener('click', (event) => {
                    event.preventDefault();
                    this.toggleFavoriteFilter();
                });

                const settingsDropdown = container.querySelector('.settings-dropdown');
                if (settingsDropdown) {
                    container.insertBefore(a, settingsDropdown);
                } else {
                    container.appendChild(a);
                }
                this.filterButton = a;
            }
        }

        /**
         * 切換收藏篩選器的啟用狀態
         */
        toggleFavoriteFilter() {
            this.isFilterActive = !this.isFilterActive;
            if (this.isFilterActive) {
                this.applyFavoriteFilter();
                this.filterButton.title = '顯示所有章節';
                this.filterButton.setAttribute('aria-label', '顯示所有章節');
                this.filterButton.classList.add('active');
            } else {
                this.clearFavoriteFilter();
                this.filterButton.title = '顯示已收藏章節';
                this.filterButton.setAttribute('aria-label', '顯示已收藏章節');
                this.filterButton.classList.remove('active');
            }
            this.emit('filterToggled', { isActive: this.isFilterActive });
        }

        /**
         * 套用收藏篩選,使用 CSS 類別控制顯示
         */
        applyFavoriteFilter() {
            // 先確保所有收藏狀態的類別都是最新的
            this.updateFavoriteClasses();
            // 只需要新增一個類別到 body,CSS 會處理其餘的顯示邏輯
            document.body.classList.add(Config.FILTER_CLASS_NAME);
        }

        /**
         * 清除收藏篩選
         */
        clearFavoriteFilter() {
            document.body.classList.remove(Config.FILTER_CLASS_NAME);
        }

        /**
         * 更新指定章節 ID 的按鈕狀態
         * @param {string} lessonId - 要更新按鈕的章節 ID
         */
        updateButtonState(lessonId) {
            const buttonInfo = this.lessonButtons.get(lessonId);
            if (buttonInfo && buttonInfo.button.isConnected()) {
                buttonInfo.button.updateState();
            }
        }

        /**
         * 更新所有收藏按鈕的狀態
         */
        updateAllButtons() {
            this.lessonButtons.forEach(({ button }) => {
                if (button.isConnected()) button.updateState();
            });
            this.updateFavoriteClasses();
        }

        /**
         * 移除所有監聽器、按鈕和樣式
         */
        destroy() {
            if (this.observer) {
                this.observer.disconnect();
                this.observer = null;
            }
            this.lessonButtons.forEach(({ button }) => button.remove());
            this.lessonButtons.clear();
            if (this.filterButton) {
                this.filterButton.remove();
                this.filterButton = null;
            }
            clearTimeout(this._cacheTimeout);
            this._lessonItemsCache = null;
        }

        /**
         * 重新執行一次初始化
         */
        refresh() {
            this.destroy();
            this.init();
        }

        // --- Public API methods ---
        getFavorites() {
            return this.storage.getAll();
        }
        addFavorite(lessonId) {
            this.storage.add(lessonId);
            this.updateButtonState(lessonId);
            this.updateFavoriteClasses();
        }
        removeFavorite(lessonId) {
            this.storage.remove(lessonId);
            this.updateButtonState(lessonId);
            this.updateFavoriteClasses();
        }
        isFavorite(lessonId) {
            return this.storage.has(lessonId);
        }
        getFavoriteCount() {
            return this.storage.getCount();
        }
        clearAllFavorites() {
            this.storage.clear();
            this.updateAllButtons();
        }
    }

    // === 腳本啟動 ===
    let favoriteManager = null;

    /**
     * 初始化應用程式,建立 TeachableFavoriteManager
     */
    function initializeApp() {
        try {
            if (!window.TeachableFavorites || !window.TeachableFavorites.getManager()) {
                favoriteManager = new TeachableFavoriteManager();
                console.log('Teachable 收藏功能已啟動 (優化版)');
            }
        } catch (error) {
            console.error('Teachable 收藏功能啟動失敗:', error);
        }
    }

    if (document.readyState === 'loading') {
        document.addEventListener('DOMContentLoaded', initializeApp);
    } else {
        initializeApp();
    }

    /**
     * @namespace TeachableFavorites
     * @description 將收藏功能 API 掛到 window 物件,以便從瀏覽器主控台進行操作
     */
    window.TeachableFavorites = {
        getFavorites: () => favoriteManager?.getFavorites() || [],
        addFavorite: (lessonId) => favoriteManager?.addFavorite(lessonId),
        removeFavorite: (lessonId) => favoriteManager?.removeFavorite(lessonId),
        isFavorite: (lessonId) => favoriteManager?.isFavorite(lessonId) || false,
        getCount: () => favoriteManager?.getFavoriteCount() || 0,
        clearAll: () => favoriteManager?.clearAllFavorites(),
        refresh: () => favoriteManager?.refresh(),
        getManager: () => favoriteManager,
        on: (event, callback) => favoriteManager?.on(event, callback),
        off: (event, callback) => favoriteManager?.off(event, callback)
    };
})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址