// ==UserScript== // @name Bilibili 旧播放页 // @namespace Motoori Kashin // @version 2.5.7 // @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 *://*.bilibili.com/* // @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'; 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
', "playlist" : '哔哩哔哩 (゜-゜)つロ 干杯~-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;}, // 早期B站播放器,无弹幕列表框 "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;}, // 无需cid的播放器 "playlist" : (aid,cid,pl) => {if(pl){return "https://www.bilibili.com/blackboard/playlist-player.html?pl=" + pl;}else{return "https://www.bilibili.com/blackboard/playlist-player.html?aid=" + aid + "&cid=" + cid;}} // 输入pl请将aid,cid置null,反之请将pl置空 } const xhr = { // xhr接口(cookies) "false" : (url)=>{ // 同步请求 const xhr = new XMLHttpRequest(); xhr.open('GET', url, false); xhr.withCredentials = true; xhr.send(null); if (xhr.status === 200) { return xhr.responseText;} }, "true" : (url,callback)=>{ // 异步请求 const xhr = new XMLHttpRequest(); xhr.open('GET', url, true); xhr.withCredentials = true; xhr.onload = () => { callback(xhr.responseText); } xhr.send(); } } 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 reHead = window.setInterval(()=>{ let head = document.getElementsByClassName("bili-header-m"); if (head[1]){ head[1].remove(); document.getElementById("bofqi").removeAttribute("style"); clearInterval(reHead);} },100); }, "deleteNewEntry" : ()=>{ // 新版入口 let deleteEntry = window.setInterval(()=>{ let new_entry = document.getElementsByClassName("new-entry")[0]; if (new_entry){new_entry.setAttribute("style","visibility: hidden;");clearInterval(deleteEntry);} },100); }, "setMiniHead" : (ele)=>{ // 迷你版头 let div = document.createElement("div"); div.setAttribute("class","z-top-container"); ele.replaceWith(div); 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 div = document.createElement("div"); div.setAttribute("class","z-top-container has-menu"); ele.replaceWith(div); 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 div = document.createElement("div"); div.setAttribute("class","footer bili-footer report-wrap-module"); div.setAttribute("id","home_footer"); ele.replaceWith(div); 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();}} }, "setJoinTime" : (data)=>{ // 注册时间 data = JSON.parse(data); let joinTime = functionInterface.timeFormat(data.data.jointime * 1000); let birthdate = data.data.birthdate; document.addEventListener("DOMNodeInserted",() => { let birthday = document.getElementsByClassName("birthday"); if(birthday[0]){ let birth = birthday[0].getElementsByClassName("text"); if(birth[0]){birth[0].text = birthdate;} if(document.getElementsByClassName("jointime")[0]){return;} else{ let div = document.createElement("div"); let icon = document.createElement("span"); let text = document.createElement("span"); let style = document.createElement("style"); div.setAttribute("class","item jointime"); birthday[0].parentNode.appendChild(div); icon.setAttribute("class","icon"); div.appendChild(icon); text.setAttribute("class","text"); text.innerText = joinTime; div.appendChild(text); style.setAttribute("type","text/css"); document.head.appendChild(style); style.appendChild(document.createTextNode(".user .info .meta .row {height: 88px;white-space: normal;}.user .info .jointime .icon {background-position: -209px -84px;}.user .info .jointime .text {color: #00a1d6;}}")); }}}); }, "timeFormat" : (time)=>{ // 格式化时间,格式:xxxx-xx-xx xx:xx:xx let date = new Date(time); let Y = date.getFullYear() + '-'; let M = (date.getMonth()+1 < 10 ? '0'+(date.getMonth()+1) : date.getMonth()+1) + '-'; let D = (date.getDate() < 10 ? '0' + (date.getDate()) : date.getDate()) + ' '; let h = (date.getHours() < 10 ? '0' + date.getHours() : date.getHours()) + ':'; let m = (date.getMinutes() <10 ? '0' + date.getMinutes() : date.getMinutes()) + ':'; let s = (date.getSeconds() <10 ? '0' + date.getSeconds() : date.getSeconds()); return Y+M+D+h+m+s; }, "setFavlist" : (num,type)=>{ // 失效视频 let small_item = document.getElementsByClassName("small-item"); let item_change = "small-item";if(type == "cid"){item_change = "small-item fakeDanmu-item"} if(small_item[0]){ for(let i=0;i= 1000){ window.tid = Date.parse( new Date()); functionInterface.refav(num,type);}}}}} }, "refav" : (num,type)=>{ // 失效视频数据 let url = "https://api.bilibili.com/medialist/gateway/base/spaceDetail?media_id="+ num +"&pn=1&ps=20&keyword=&order=mtime&type=0&tid=0&jsonp=jsonp"; if(type == "cid"){url = "https://api.bilibili.com/x/space/channel/video?mid=" + window.mid + "&cid=" + num + "&pn=1&ps=30&order=0&jsonp=jsonp";} xhr.true(url,functionInterface.callbackRefav); }, "callbackRefav" : (data)=>{ // 失效视频回调 data = JSON.parse(data).data; let type = "fid";if(data.list){type = "cid"} let disabled = document.getElementsByClassName("small-item"); for(let i=0;i{ // 免接口的失效视频 let small_item = document.getElementsByClassName("small-item"); if(small_item[0]){ for(let i=0;i{ // 在线显示 let style = document.createElement("style"); style.setAttribute("type","text/css"); document.head.appendChild(style); style.appendChild(document.createTextNode(".online a {color: rgb(109, 117, 122);}.popularize-module .online em {display: inline-block;height: 10px;line-height: 10px;vertical-align: top;border-left: 1px solid rgb(184, 192, 204);margin: 12px 15px 0px;}")); let ck_online = window.setInterval(()=>{ let online = document.getElementsByClassName("online"); if(online[0]){ clearInterval(ck_online); let url = "https://api.bilibili.com/x/web-interface/online"; xhr.true(url,functionInterface.callbackOnline); } },1000); }, "callbackOnline" : (data)=>{ // 在线显示回调 data = JSON.parse(data).data; let all_count = data.all_count; let web_online = data.web_online; let play_online = data.play_online; let online = document.getElementsByClassName("online")[0]; if(online.tagName == "DIV"){online = online.getElementsByTagName("a")[0];} else{ let parent = online.parentNode; online.remove(); let div = document.createElement("div"); let a = document.createElement("a"); div.setAttribute("class","online"); parent.insertBefore(div,parent.firstChild); a.setAttribute("href","//www.bilibili.com/video/online.html"); a.setAttribute("target","_blank"); div.appendChild(a); online = a;} online.setAttribute("title","在线观看:" + play_online); online.text = "在线人数:" + web_online; if(!online.parentNode.getElementsByTagName("em")[0]){ let em = document.createElement("em"); let count = document.createElement("a"); online.parentNode.insertBefore(em,online.nextSibling); count.setAttribute("href","//www.bilibili.com/newlist.html"); count.setAttribute("target","_blank"); online.parentNode.insertBefore(count,em.nextSibling); count.text = "最新投稿:" + all_count;} else{ let count = online.parentNode.getElementsByTagName("a")[1]; count.text = "最新投稿:" + all_count;} window.setTimeout(functionInterface.online,60000); // 在线数据轮循 } } 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(); }); }, "space" : ()=>{ // 个人空间 if(window.DedeUserID && window.DedeUserID==window.mid){ //注册时间 let url = "https://member.bilibili.com/x2/creative/h5/calendar/card?ts=0"; xhr.true(url,functionInterface.setJoinTime); } document.addEventListener("DOMNodeInserted",()=>{ // 失效视频 let fav_item = document.getElementsByClassName("fav-item"); let breadcrumb = document.getElementsByClassName("breadcrumb"); let channel_item = document.getElementsByClassName("channel-item"); if(fav_item[0]){ for(let i=0;i{ // video INITIAL_DOCUMENT = xhr.false(location.href); if (INITIAL_DOCUMENT.match(/__INITIAL_STATE__/)){ 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();} }, "watchlater" : ()=>{ // 稍后再看 let html = page.watchlater; functionInterface.rewritePage(html); functionInterface.selectDanmu(); }, "bangumi" : ()=>{ // 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" : ()=>{ // 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 url = location.href; let aid = 1 * url.match(/aid=[0-9]*/)[0].replace(/aid=/,""); let cid = url.match(/cid=[0-9]*/);if(cid && cid[0]){cid = 1 * cid[0].replace(/cid=/,"");} if(!cid){ let url = '//api.bilibili.com/x/player/pagelist?aid=' + aid + '&jsonp=jsonp' let cid = JSON.parse(xhr.false(url)).data[0].cid; location.replace(iframeplayer.blackboard(aid,cid)); } else{ location.replace(iframeplayer.blackboard(aid,cid)); } }, "home" : ()=>{ // Bilibili主页(当前尚未启用,且实现上仍有问题:丢失了主推和推广) //let html = page.home; //functionInterface.rewritePage(html); functionInterface.setOnline(); }, "playlist" : ()=>{ // 播单页(被废弃的旧版播放页遗存) //let html = page.playlist; //functionInterface.rewritePage(html); /* 暂不需要重写,统一下播放器布局并修复版头失效问题 */ window.onload = ()=>{ document.getElementsByClassName("bili-header-m")[0].remove(); let div = document.createElement("div"); div.setAttribute("class","z-top-container has-menu"); document.body.insertBefore(div,document.body.firstChild); 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); let style = document.createElement("style"); style.setAttribute("type","text/css"); document.head.appendChild(style); style.appendChild(document.createTextNode("#bofqi .player {width:980px;height:620px;display:block;}@media screen and (min-width:1400px){#bofqi .player{width:1160px;height:720px}}")); } } } /* 脚本主入口 */ if(INITIAL_PATH[3]){ // 单独处理入口 if(INITIAL_PATH[3] == 'video' && INITIAL_PATH[4].startsWith('av')){rewritePage.av();} if(INITIAL_PATH[3] == 'watchlater'){rewritePage.watchlater();} if(INITIAL_PATH[3] == 'bangumi' && INITIAL_PATH[4] == 'play'){rewritePage.bangumi();} if(INITIAL_PATH[3] == 'blackboard' && INITIAL_PATH[4] && INITIAL_PATH[4].startsWith('newplayer')){rewritePage.blackboard();} if(INITIAL_PATH[3] == 'playlist' && INITIAL_PATH[4] == 'video' && INITIAL_PATH[5].startsWith('pl')){rewritePage.playlist();} if(INITIAL_PATH[2] != 'live.bilibili.com'){global.rewriteSction();} if(INITIAL_PATH[2] == 'space.bilibili.com'){global.space();} } else{ if(INITIAL_PATH[2] == 'www.bilibili.com'){rewritePage.home();} } if(INITIAL_PATH[2].match(/live/) == null){ // 全局处理入口 document.addEventListener("DOMContentLoaded",global.resetSction()); } })();