Greasy Fork

Greasy Fork is available in English.

自动设置B站的自动连播(自动切集)

B站的多数页面中,自动切集功能在播放器里,修改起来很麻烦,所以按照我的习惯做了在不同页面自动切换的功能(脚本运行后可在存储中修改配置),具体包括:1.【自己的收藏等列表页面】自动连播;2.【普通视频/稍后再看/番剧】不自动连播;3.其余情况默认也不自动连播。【吐槽】视频或番剧看完可能会想看评论区,或者点赞等,所以不想自动连播,尤其是番剧最新一集播完可能连播到奇怪的视频。

当前为 2024-01-06 提交的版本,查看 最新版本

// ==UserScript==
// @name         自动设置B站的自动连播(自动切集)
// @namespace    https://github.com/SineObama/bilibili-player-auto-set-playtype
// @homepage     https://github.com/SineObama/bilibili-player-auto-set-playtype
// @version      0.1.2
// @description  B站的多数页面中,自动切集功能在播放器里,修改起来很麻烦,所以按照我的习惯做了在不同页面自动切换的功能(脚本运行后可在存储中修改配置),具体包括:1.【自己的收藏等列表页面】自动连播;2.【普通视频/稍后再看/番剧】不自动连播;3.其余情况默认也不自动连播。【吐槽】视频或番剧看完可能会想看评论区,或者点赞等,所以不想自动连播,尤其是番剧最新一集播完可能连播到奇怪的视频。
// @author       SineObama
// @match        https://www.bilibili.com/*
// @grant        GM_setValue
// @grant        GM_getValue
// @license      MIT
// @icon         
// ==/UserScript==

'use strict';

// 方便自己随时修改代码,固定使用代码中的最新配置内容
var reset = loadStorage('__ALWAYS_RESET_CONFIG__', false);
// 实验性功能:阻止番剧最后一集播完后自动跳转
var experimentalFeature = loadStorage('__EXPERIMENTAL_FEATURE__', false);
// 配置需要开启功能的页面
var urlsToOpen = loadStorage('urlsToOpen', [
    'bilibili.com/list',
], reset);
// 配置需要关闭功能的页面
var urlsToClose = loadStorage('urlsToClose', [
    'bilibili.com/video',
    'bilibili.com/bangumi',
    'bilibili.com/list/watchlater',
], reset);
// 配置默认行为(没有匹配以上规则时):true/false 设置为开启/关闭; null 不修改
var unmatchBehavior = loadStorage('unmatchBehavior', false, reset);

// ==== 配置结束 ====

var myBlockJump = experimentalFeature;

doCheck(unmatchBehavior);

function doCheck(isOpen) {

    isOpen = travelList(urlsToOpen, true, isOpen);
    isOpen = travelList(urlsToClose, false, isOpen);

    if (isOpen === null) {
        return;
    }

    if (myBlockJump && _matchLocation('bilibili.com/bangumi')) {
        doBlockJump();
    }

    var radioIdx = isOpen ? 0 : 1;
    // 需要多次检查结果,避免被B站设置覆盖
    var keepCount = 0;
    var id = setInterval(function () {
        var el = document.getElementsByClassName('bpx-player-ctrl-setting-handoff')[0];
        if (el) {
            console.debug('video_status.playtype radioIdx', radioIdx, 'keep', keepCount);
            if (el.getElementsByClassName('bui-radio-input')[radioIdx].checked) {
                keepCount++;
                if (keepCount > 2) {
                    clearInterval(id);
                }
            } else {
                el.getElementsByClassName('bui-radio-item')[radioIdx].click();
                keepCount = 0;
            }
        }
    }, 1000);
}

function loadStorage(key, defaultValue, isReset) {
    if (!isReset) {
        var value = GM_getValue(key);
        if (value !== undefined) {
            return value;
        }
    }
    GM_setValue(key, defaultValue);
    return defaultValue;
}

function travelList(list, target, defaultResult) {
    var result = defaultResult;
    for (var i = 0; i < list.length; i++) {
        if (_matchLocation(list[i])) {
            result = target;
            break;
        }
    }
    return result;
}

function _matchLocation(matcher) {
    if (typeof matcher === 'string') {
        if (location.href.indexOf(matcher) > -1) {
            return true;
        }
    } else {
        if (matcher.test(location.href)) {
            return true;
        }
    }
    return false;
}

// 失败方法:

// var settings = JSON.parse(localStorage.getItem('bilibili_player_settings'));
// if (/bilibili.com\/list/.test(location.href) && location.href.indexOf('bilibili.com/list/watchlater') < 0) {
//     if (settings.video_status.playtype === 2) {
//         settings.video_status.playtype = 1;
//         localStorage.setItem('bilibili_player_settings', JSON.stringify(settings));
//     }
// } else {
//     if (settings.video_status.playtype === 1) {
//         settings.video_status.playtype = 2;
//         localStorage.setItem('bilibili_player_settings', JSON.stringify(settings));
//     }
// }

function doBlockJump() {

    // 阻止 History.pushState 方法的调用
    var pushStateAssign = window.History.prototype.pushState;

    Object.defineProperty(window.History.prototype, 'pushState', {
        get: function () {
            // 启用阻止时,通过异常阻止页面跳转!偶然发现的方法,单纯throw不行,不清楚原理
            return myBlockJump ? pushStateAssign.get.call(this) : pushStateAssign;
        }
    });

    // 如果是正常的人为点击操作,则允许页面跳转
    window.addEventListener('click', releaseJump, true);

    function releaseJump(e) {
        if (!myBlockJump) {
            return;
        }

        // 如果点击的元素在A标签中则视为人为点击跳转链接,允许跳转
        var release = false;
        var el = e.target;
        while (el) {
            if (el.tagName === 'A') {
                release = true;
                break;
            }
            el = el.parentNode;
        }

        if (release) {
            // 通过暂时关闭阻止功能来允许页面跳转
            myBlockJump = false;
            setTimeout(function () {
                myBlockJump = true;
            }, 500);
        }
    }

    // 以下研究其他跳转方式的阻止方法,但B站没用或者对B站没用
    // // 阻止页面的 location.href 和 location.replace 方法的调用
    // var originalHref = Object.getOwnPropertyDescriptor(window.Location.prototype, 'href');
    // var originalReplace = Object.getOwnPropertyDescriptor(window.Location.prototype, 'replace');
    //
    // Object.defineProperty(window.Location.prototype, 'href', {
    //     set: function (value) {
    //         // 阻止设置新的 URL
    //         console.log('Blocked href:', value);
    //     },
    //     get: function () {
    //         return originalHref.get.call(this);
    //     }
    // });
    //
    // Object.defineProperty(window.Location.prototype, 'replace', {
    //     set: function (value) {
    //         // 阻止替换当前 URL
    //         console.log('Blocked replace:', value);
    //     },
    //     get: function () {
    //         return originalReplace.get.call(this);
    //     }
    // });
    //
    //
    // // 阻止 window.open 方法的调用
    // var originalOpen = window.open;
    //
    // window.open = function (url, target, features) {
    //     // 阻止打开新窗口或标签页
    //     console.log('Blocked window.open:', url, target, features);
    // };
    //
    //
    // // 阻止 location.assign 方法的调用
    // var originalAssign = window.Location.prototype.assign;
    //
    // Object.defineProperty(window.Location.prototype, 'assign', {
    //     set: function (value) {
    //         // 阻止设置新的 URL
    //         console.log('Blocked assign:', value);
    //     },
    //     get: function () {
    //         return originalAssign.get.call(this);
    //     }
    // });
    //
    //
    // // 阻止 History.pushState 方法的调用
    // var originalPushState = history.pushState;
    //
    // history.pushState = function (state, title, url) {
    //     // 阻止改变浏览器的 URL 和历史记录
    //     console.log('Blocked pushState:', state, title, url);
    // };
}