// ==UserScript== // @name Bilibili 旧播放页 // @namespace Motoori Kashin // @version 2.5.4 // @description 切换Bilibili旧版HTML5播放器,恢复2019年12月09日之前的界面。已实现video/bangumi/watchlater/mylist及嵌入式播放器。 // @author Motoori Kashin // @homepageURL https://github.com/MotooriKashin/Bilibili-Old/ // @supportURL https://github.com/MotooriKashin/Bilibili-Old/issues // @match *://*/* // @license MIT License // @run-at document-start // @icon https://static.hdslb.com/images/favicon.ico // @compatible chrome // @grant none // @downloadURL none // ==/UserScript== (function() { 'use strict'; /* 设置为全网脚本是处理bilibili嵌入播放器的跨域权限 */ /* 若有强迫症,可自行将上面@match后面替换为【*://*.bilibili.com/*】 */ /* 那将导致非bilibili域名下的嵌入播放器替换失败 */ let INITIAL_DOCUMENT = ""; // 网页源代码,为避免多余请求,需要时再通过url获取(尚不知如何直接从本地获取) let INITIAL_TITLE = document.getElementsByTagName("title");if (INITIAL_TITLE[0]){INITIAL_TITLE = INITIAL_TITLE[0].innerText;} // 网页原标题 let INITIAL_PATH = document.location.href.split('/'); // 当前网址(包括嵌入的子页面) const page = { // 网页框架 "video" : '' + INITIAL_TITLE + '
', "watchlater" : '哔哩哔哩 (゜-゜)つロ 干杯~-bilibili
', "bangumi" : '' + INITIAL_TITLE + '
', "special" : '' + INITIAL_TITLE + '
', "home" : '哔哩哔哩 (゜-゜)つロ 干杯~-bilibili
' } const iframeplayer = { // 嵌入式播放器,当前取第一个(blackboard) "blackboard" : (aid,cid) => {return "https://www.bilibili.com/blackboard/html5player.html?aid=" + aid + "&cid=" + cid + "&as_wide=1&player_type=2&urlparam=module%253DbangumicrossDomain=true";}, "ancient" : (aid,cid) => {return "https://www.bilibili.com/blackboard/activity-ancient-player.html?aid=" + aid + "&cid=" + cid;}, "normal" : (aid,cid) => {return "https://player.bilibili.com/player.html?aid=" + aid + "&cid=" + cid + "&page=1";}, "html" : (aid,cid) => {return "https://www.bilibili.com/html/player.html?wmode=transparent&aid=" + aid;} } const InitialState = { // bangumi播放信息(INITIAL_STATE) "bangumi" : (xhr,epId)=>{ let is = JSON.parse(xhr).result; // url返回的INITIAL_STATE let ep = 0; // 布尔值,判断当前集数用,需登录且有播放记录才不为0 let ic = JSON.parse(INITIAL_DOCUMENT.match(/INITIAL_STATE__=.+?\;\(function/)[0].replace(/INITIAL_STATE__=/,"").replace(/\;\(function/,"")); // 网页源INITIAL_STATE let pug = JSON.parse(INITIAL_DOCUMENT.match(/PGC_USERSTATE__=.+?<\/script>/)[0].replace(/PGC_USERSTATE__=/,"").replace(/<\/script>/,"")); // 网页源用户信息(PGC_USERSTATE) let dat = {"ver":{},"loginInfo":{},"canReview":false,"userShortReview":{},"userLongReview":{},"userScore":0,"userCoined":false,"isPlayerTrigger":false,"special":false,"area":0,"app":false,"recomList":[],"playerRecomList":[],"paster":{},"payPack":{},"payMent":{},"activity":{},"spending":0,"sponsorTotal":{"code":0,"result":{"ep_bp":0,"users":0,"mine":{},"list":[]}},"sponsorWeek":{"code":0,"result":{"ep_bp":0,"users":0,"mine":{},"list":[]}},"sponsorTotalCount":0,"miniOn":true,"seasonFollowed":false,"epStat":{},"ssStat":{}}; if (epId){dat.epId = 1 * epId;ep = 1;}else{dat.epId = "";if(pug.hasOwnProperty("progress")){dat.epId = pug.progress.last_ep_id;ep = 1;}} // 当前选集,ep=0时取第一集 dat.ssId = is.season_id; // 番剧id dat.mdId = 1 * is.link.match(/[0-9][0-9]*/)[0]; // 番剧详情id dat.mediaInfo = {};/* 番剧基本信息 */dat.mediaInfo.actors = is.actors;dat.mediaInfo.alias = is.alias;dat.mediaInfo.areas = is.areas;dat.mediaInfo.cover = is.cover;dat.mediaInfo.evaluate = is.evaluate;dat.mediaInfo.is_paster_ads = is.is_paster_ads;dat.mediaInfo.jp_title = is.jp_title;dat.mediaInfo.link = is.link;dat.mediaInfo.media_id = is.media_id;dat.mediaInfo.mode = is.mode;dat.mediaInfo.season_id = is.season_id;dat.mediaInfo.season_status = is.season_status;dat.mediaInfo.season_title = is.season_title;dat.mediaInfo.season_type = is.season_type;dat.mediaInfo.series_title = is.series_title;dat.mediaInfo.square_cover = is.square_cover;dat.mediaInfo.staff = is.staff;dat.mediaInfo.style = is.style;dat.mediaInfo.title = is.title;dat.mediaInfo.total_ep = is.total_ep; dat.mediaRating = is.rating; // 番剧评分相关 dat.epList = is.episodes;/*番剧分集信息*/if (ep==0){dat.epId=dat.epList[0].ep_id;}for (let i = 0;i{ let ini = JSON.parse(xhr).result;let ep = 0; let pug = JSON.parse(INITIAL_DOCUMENT.match(/PGC_USERSTATE__=.+?<\/script>/)[0].replace(/PGC_USERSTATE__=/,"").replace(/<\/script>/,"")); let is = JSON.parse(INITIAL_DOCUMENT.match(/INITIAL_STATE__=.+?\;\(function/)[0].replace(/INITIAL_STATE__=/,"").replace(/\;\(function/,"")); let dat = {"ver":{"mobile":false,"ios":false,"android":false,"windowsPhone":false,"iPhone":false,"ios9":false,"iPad":false,"webApp":false,"microMessenger":false,"weibo":false,"uc":false,"qq":false,"baidu":false,"mqq":false,"mBaidu":false,"iqiyi":false,"qqLive":false,"safari":true,"youku":false,"ie":false,"edge":false,"bili":false,"biliVer":0},"loginInfo":{},"canReview":false,"userShortReview":{},"userLongReview":{},"userScore":0,"userCoined":false,"isPlayerTrigger":false,"special":true,"area":0,"app":false,"mediaRating":{},"recomList":[],"playerRecomList":[],"paster":{},"payPack":{},"payMent":{},"activity":{},"spending":0,"sponsorTotal":{"code":0,"result":{"ep_bp":0,"users":0,"mine":{},"list":[]}},"sponsorWeek":{"code":0,"result":{"ep_bp":0,"users":0,"mine":{},"list":[]}},"sponsorTotalCount":0,"miniOn":true,"seasonFollowed":false}; if (epId){dat.epId = 1 * epId;ep = 1;}else{dat.epId = "";if(pug.hasOwnProperty("progress")){dat.epId = pug.progress.last_ep_id;ep = 1;}} dat.ssId = ini.season_id; dat.mdId = 1 * ini.link.match(/[0-9][0-9]*/)[0]; dat.mediaInfo = {};dat.mediaInfo.actors = ini.actors;dat.mediaInfo.alias = ini.alias;dat.mediaInfo.areas = ini.areas;dat.mediaInfo.bkg_cover = ini.bkg_cover;dat.mediaInfo.cover = ini.cover;dat.mediaInfo.evaluate = ini.evaluate;dat.mediaInfo.is_paster_ads = ini.is_paster_ads;dat.jp_title = ini.jp_title;dat.mediaInfo.link = ini.link;dat.mediaInfo.media_id = ini.media_id;dat.mediaInfo.mode = ini.mode;dat.mediaInfo.season_id = ini.season_id;dat.mediaInfo.season_status = ini.season_status;dat.mediaInfo.season_title = ini.season_title;dat.mediaInfo.season_type = ini.season_type;dat.mediaInfo.square_cover = ini.square_cover;dat.mediaInfo.staff = ini.staff;dat.mediaInfo.stat = ini.state;dat.mediaInfo.style = ini.style;dat.mediaInfo.title = ini.title; dat.mediaRating = ini.rating; dat.epList = ini.episodes;if (ep==0){dat.epId=dat.epList[0].ep_id;}for (let i = 0;i{ // 重写页面 document.open(); document.write(html); document.close();}, "selectDanmu" : ()=>{ // 弹幕列表 let danmuku = window.setInterval(()=>{ let setDanmu = document.getElementsByClassName("bilibili-player-filter-btn")[1]; if (setDanmu){setDanmu.click();clearInterval(danmuku);} },100); }, "deleteHead" : ()=>{ // 失效版头 let reh = window.setInterval(()=>{ let reHead = document.getElementsByClassName("bili-header-m"); if (reHead[1]){ reHead[1].remove(); document.getElementById("bofqi").removeAttribute("style"); clearInterval(reh);} },100); }, "deleteNewEntry" : ()=>{ // 新版入口 let reh = window.setInterval(()=>{ let etN = document.getElementsByClassName("new-entry")[0]; if (etN){etN.setAttribute("style","visibility: hidden;");clearInterval(reh);} },100); }, "setMiniHead" : (ele)=>{ // 迷你版头 let reh = document.createElement("div"); reh.setAttribute("class","z-top-container"); ele.replaceWith(reh); let script = document.createElement("script"); script.setAttribute("type","text/javascript"); script.setAttribute("src","//s1.hdslb.com/bfs/seed/jinkela/header/header.js"); document.body.appendChild(script); }, "setTotalHead" : (ele)=>{ // 完整版头 let reh = document.createElement("div"); reh.setAttribute("class","z-top-container has-menu"); ele.replaceWith(reh); let script = document.createElement("script"); script.setAttribute("type","text/javascript"); script.setAttribute("src","//s1.hdslb.com/bfs/seed/jinkela/header/header.js"); document.body.appendChild(script); }, "setFoot" : (ele)=>{ // 版底 let ret = document.createElement("div"); ret.setAttribute("class","footer bili-footer report-wrap-module"); ret.setAttribute("id","home_footer"); ele.replaceWith(ret); let script = document.createElement("script"); script.setAttribute("type","text/javascript"); script.setAttribute("src","//static.hdslb.com/common/js/footer.js"); document.body.appendChild(script); }, "removeBlur" : ()=>{ // 版头蒙板 let blur = document.getElementsByClassName("blur-bg"); if (blur[0]){ blur[0].removeAttribute("style");} }, "removePreview" : ()=>{ // 6分钟预览 let hint = document.getElementsByClassName("video-float-hint-btn"); if (hint[0]){ let i = 10; // 倒计时长度,可自行修改,单位/s if(document.getElementsByClassName("second-cut")[0]){return;} else{ let sec = document.createElement("span"); sec.setAttribute("class","video-float-hint-btn second-cut"); hint[0].parentNode.appendChild(sec); function cut(){ sec.innerText = i - 1 + "s"; if(i==0){hint[0].parentNode.remove();return;} i = i - 1; window.setTimeout(cut,1000); } new cut(); } } } } const xhr = { // xhr接口 "false" : (url)=>{ const xhr = new XMLHttpRequest(); xhr.open('GET', url, false); xhr.send(null); if (xhr.status === 200) { return xhr.responseText;}}, "true" : (url)=>{ const xhr = new XMLHttpRequest(); xhr.open('GET', url, true); xhr.onload = () => { return xhr.responseText; xhr.send();}} } const global = { "rewriteSction" : ()=>{ // 版头和版底 document.addEventListener("DOMNodeInserted",() => { let inh = document.getElementById("internationalHeader"); let inf = document.getElementsByClassName("international-footer"); if (inh){ // 判断版头类型 let ppt = document.getElementById("primaryPageTab"); if (ppt){functionInterface.setTotalHead(inh);}else{functionInterface.setMiniHead(inh);} } if (inf[0]){functionInterface.setFoot(inf[0]);} // 判断版底 });}, "resetSction" : ()=>{ // 其他全局入口 document.addEventListener("DOMNodeInserted",() => { functionInterface.removeBlur(); functionInterface.removePreview(); }); } } const rewritePage = { "av" : ()=>{ INITIAL_DOCUMENT = xhr.false(location.href); if (INITIAL_DOCUMENT.match(/biliconfig/) == null && INITIAL_DOCUMENT.match(/"trueCode":-404/) == null){ window.__INITIAL_STATE__ = JSON.parse(INITIAL_DOCUMENT.match(/INITIAL_STATE__=.+?\;\(function/)[0].replace(/INITIAL_STATE__=/,"").replace(/\;\(function/,"")); let html = page.video; functionInterface.rewritePage(html); functionInterface.selectDanmu(); functionInterface.deleteHead(); } else{ console.log("Error!Old Bilibili: 跳转页 ",INITIAL_DOCUMENT.match(/biliconfig/)); console.log("Error!Old Bilibili: 报错页 ",INITIAL_DOCUMENT.match(/"trueCode":-404/)); } }, "watchlater" : ()=>{ let html = page.watchlater; functionInterface.rewritePage(html); functionInterface.selectDanmu(); }, "bangumi" : ()=>{ INITIAL_DOCUMENT = xhr.false(location.href); if (INITIAL_DOCUMENT.match(/__INITIAL_STATE__/)){ if (INITIAL_DOCUMENT.match(/"specialCover":""/)){ if(window.__INITIAL_STATE__){ Reflect.deleteProperty(window, "__INITIAL_STATE__");} let id = location.href.match(/[0-9]+/); if(INITIAL_PATH[5].startsWith('ss')){ let url = "https://bangumi.bilibili.com/view/web_api/season?season_id=" + id[0]; let ini = xhr.false(url); InitialState.bangumi(ini,null);} if(INITIAL_PATH[5].startsWith('ep')){ let url = "https://bangumi.bilibili.com/view/web_api/season?ep_id=" + id[0]; let ini = xhr.false(url); InitialState.bangumi(ini,id[0]);} let html = page.bangumi; functionInterface.rewritePage(html); functionInterface.selectDanmu(); functionInterface.deleteNewEntry(); } else{rewritePage.special();}} }, "special" : ()=>{ if(window.__INITIAL_STATE__){ Reflect.deleteProperty(window, "__INITIAL_STATE__");} let id = location.href.match(/[0-9]+/); if(INITIAL_PATH[5].startsWith('ss')){ let url = "https://bangumi.bilibili.com/view/web_api/season?season_id=" + id[0]; let ini = xhr.false(url); InitialState.special(ini,null);} if(INITIAL_PATH[5].startsWith('ep')){ let url = "https://bangumi.bilibili.com/view/web_api/season?ep_id=" + id[0]; let ini = xhr.false(url); InitialState.special(ini,id[0]);} let html = page.special; functionInterface.rewritePage(html); }, "blackboard" : ()=>{ let player = document.getElementsByTagName("iframe"); if (player[0]){ for (let i = 0;i < player.length;i++){ let src = player[i].src; console.log(player[i]); if (src && src.match(/newplayer/)){ let aid = 1 * src.match(/aid=[0-9]*/)[0].replace(/aid=/,""); let cid = src.match(/cid=[0-9]*/);if(cid && cid[0]){cid = 1 * cid[0].replace(/cid=/,"");} let element = player[i]; let width = player[i].offsetWidth;let height = player[i].offsetHeight; if(!cid){ // 尝试用URL获取cid(可能会跨域注意!) let xhr = new XMLHttpRequest(); let url = '//api.bilibili.com/x/player/pagelist?aid=' + aid + '&jsonp=jsonp'; xhr.open('GET',url,true); xhr.onload = () => { let cid = JSON.parse(xhr.responseText).data[0].cid; let iframe = document.createElement("iframe"); iframe.setAttribute("src",iframeplayer.blackboard(aid,cid)); iframe.setAttribute("style",'width: '+ width +'px;height: ' + height + 'px;'); element.replaceWith(iframe); } xhr.send(); } else{ let iframe = document.createElement("iframe"); iframe.setAttribute("src",iframeplayer.blackboard(aid,cid)); iframe.setAttribute("style",'width: '+ width +'px;height: ' + height + 'px;'); element.replaceWith(iframe); } } } } }, "home" : ()=>{ let html = page.home; functionInterface.rewritePage(html); }, } if(INITIAL_PATH[2] && INITIAL_PATH[2].match(/bilibili\./)){ // 匹配*.bilibili.com if(INITIAL_PATH[3]){ if(INITIAL_PATH[3] == 'video' && INITIAL_PATH[4].startsWith('av')){rewritePage.av();} // 普通av页 if(INITIAL_PATH[3] == 'watchlater'){rewritePage.watchlater();} // 稍后再看 if(INITIAL_PATH[3] == 'bangumi' && INITIAL_PATH[4] == 'play'){rewritePage.bangumi();} // bangumi(包括特殊页面) if(INITIAL_PATH[3] == 'blackboard' && INITIAL_PATH[4] && INITIAL_PATH[4].startsWith('newplayer')){window.parent.postMessage("newplayer", "*")} // 嵌入式页面 if(INITIAL_PATH[2] != 'live.bilibili.com'){global.rewriteSction();} // 不处理直播页版面 } if(INITIAL_PATH[2].match(/live/) == null){ document.addEventListener("DOMContentLoaded",global.resetSction()); // 不对直播页进行其他全局处理 } } /* 监听子页面传送的嵌入式播放器信息 */ window.addEventListener('message',(event)=>{if(event.data == "newplayer"){console.log("发现嵌入播放器!开始替换……",event.data + " -> oldplayer");rewritePage.blackboard();}},false); })();