Greasy Fork is available in English.

腾讯课堂签到提醒

腾讯课堂签到时弹出通知,可配置自动签到

// ==UserScript==
// @name         腾讯课堂签到提醒
// @namespace    http://tampermonkey.net/
// @version      2.1
// @description  腾讯课堂签到时弹出通知,可配置自动签到
// @author       askar882
// @match        https://ke.qq.com/webcourse/*
// @icon         
// @grant        none
// @run-at       document-idle
// ==/UserScript==

/* jshint esversion: 6, browser: true */

(function() {
    'use strict';

    // 启用自动签到
    const AUTO_SIGN_ENABLED = false;

    // 日志
    // const log = (...msg) => console.info(`%c${new Date().toISOString()}  腾讯课堂签到  `, 'color: blue;', ...msg);

    // 使用Notification API弹出通知
    const notify = (title, body, { tag = "", requireInteraction = true, callback = null } = {}) => {
        const createNotification = () => {
            const notification = new Notification(title, { body, tag, requireInteraction });
            notification.addEventListener('click', () => {
                window.focus();
                notification.close();
            });
            if (callback) {
                notification.addEventListener('click', callback);
            }
        };
        if(Notification.permission === 'granted' && title) {
            createNotification();
        } else {
            Notification.requestPermission().then(permission => {
                if(permission === 'granted' && title) {
                    createNotification();
                }
            });
        }
    };
    // Unix时间戳
    const getTime = () => Math.floor(new Date().getTime() / 1000);
    // 签到弹窗父元素
    const wrapper = document.getElementById('react-body');
    // 签到按钮
    const signButton = () => document.querySelector('#react-body > div.sign-dialog > div.im-dialog-wrap > div > div.btn-group > span');
    // 自动签到
    const autoSign = () => {
        if (signButton()) {
            signButton().click();
        }
    };
    // 最近一次签到时间,用于对通知进行节流
    let lastSignTime = getTime();
    // MutationObserver侦听器
    const wrapperMutation = () => {
        // 判断签到按钮是否存在
        if(signButton() && (getTime() - lastSignTime > 30)) {
            // log('签到');
            if (AUTO_SIGN_ENABLED) {
                autoSign();
                notify('腾讯课堂', '已完成自动签到', { callback: autoSign });
            } else {
                notify('腾讯课堂', '签到');
            }
            lastSignTime = getTime();
        }
    };
    // 按需申请通知权限
    if (Notification.permission !== 'granted') {
        notify(null, null);
    }
    const wrapperObserver = new MutationObserver(wrapperMutation);
    // 观察DOM树的父元素所有子孙节点的新增和删除
    wrapperObserver.observe(wrapper, {attributes: false, childList: true, subtree: true});
})();