// ==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  // @downloadURL none // ==/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); // }; }