// ==UserScript==
// @name Bilibili 旧播放页
// @namespace MotooriKashin
// @version 3.6.2
// @description 恢复原生的旧版页面,包括主页和播放页。
// @author MotooriKashin, wly5556
// @supportURL https://github.com/MotooriKashin/Bilibili-Old/issues
// @match *://*.bilibili.com/*
// @connect bilibili.com
// @connect biliplus.com
// @connect jijidown.com
// @connect mcbbs.net
// @require https://cdn.jsdelivr.net/npm/[email protected]/dist/protobuf.js
// @icon https://static.hdslb.com/images/favicon.ico
// @grant GM_xmlhttpRequest
// @grant GM_getValue
// @grant GM_setValue
// @run-at document-start
// @license MIT License
// ==/UserScript==
(function() {
'use strict';
// 全局变量
let ml, aid, big, cid, mid, oid, pgc, src, tid, uid, url, xml, bvid, limit, defig;
let arr = [], ids = [], obj = {}, mdf = {}, hash = [], bloburl = {};
let DOCUMENT, __playinfo__, __INITIAL_STATE__;
let LOCATION = document.location.href.split('/');
// protobuf初始化
const root = window.protobuf.Root.fromJSON(JSON.parse('{"nested":{"bilibili":{"nested":{"DmWebViewReply":{"fields":{"state":{"type":"int32","id":1},"text":{"type":"string","id":2},"textSide":{"type":"string","id":3},"dmSge":{"type":"DmSegConfig","id":4},"flag":{"type":"DanmakuFlagConfig","id":5},"specialDms":{"rule":"repeated","type":"string","id":6},"checkBox":{"type":"bool","id":7},"count":{"type":"int64","id":8},"commandDms":{"rule":"repeated","type":"CommandDm","id":9},"dmSetting":{"type":"DanmuWebPlayerConfig","id":10}}},"CommandDm":{"fields":{"id":{"type":"int64","id":1},"oid":{"type":"int64","id":2},"mid":{"type":"int64","id":3},"command":{"type":"string","id":4},"content":{"type":"string","id":5},"progress":{"type":"int32","id":6},"ctime":{"type":"string","id":7},"mtime":{"type":"string","id":8},"extra":{"type":"string","id":9},"idStr":{"type":"string","id":10}}},"DmSegConfig":{"fields":{"pageSize":{"type":"int64","id":1},"total":{"type":"int64","id":2}}},"DanmakuFlagConfig":{"fields":{"recFlag":{"type":"int32","id":1},"recText":{"type":"string","id":2},"recSwitch":{"type":"int32","id":3}}},"DmSegMobileReply":{"fields":{"elems":{"rule":"repeated","type":"DanmakuElem","id":1}}},"DanmakuElem":{"fields":{"id":{"type":"int64","id":1},"progress":{"type":"int32","id":2},"mode":{"type":"int32","id":3},"fontsize":{"type":"int32","id":4},"color":{"type":"uint32","id":5},"midHash":{"type":"string","id":6},"content":{"type":"string","id":7},"ctime":{"type":"int64","id":8},"weight":{"type":"int32","id":9},"action":{"type":"string","id":10},"pool":{"type":"int32","id":11},"idStr":{"type":"string","id":12}}},"DanmuWebPlayerConfig":{"fields":{"dmSwitch":{"type":"bool","id":1},"aiSwitch":{"type":"bool","id":2},"aiLevel":{"type":"int32","id":3},"blocktop":{"type":"bool","id":4},"blockscroll":{"type":"bool","id":5},"blockbottom":{"type":"bool","id":6},"blockcolor":{"type":"bool","id":7},"blockspecial":{"type":"bool","id":8},"preventshade":{"type":"bool","id":9},"dmask":{"type":"bool","id":10},"opacity":{"type":"float","id":11},"dmarea":{"type":"int32","id":12},"speedplus":{"type":"float","id":13},"fontsize":{"type":"float","id":14},"screensync":{"type":"bool","id":15},"speedsync":{"type":"bool","id":16},"fontfamily":{"type":"string","id":17},"bold":{"type":"bool","id":18},"fontborder":{"type":"int32","id":19},"drawType":{"type":"string","id":20}}}}}}}'));
const protoSeg = root.lookupType('bilibili.DmSegMobileReply');
const protoView = root.lookupType('bilibili.DmWebViewReply');
// 脚本默认设置:0 - 关闭,1 - 开启
// 一般直接修改这里无效,脚本读取的是管理器中的数据
const config = {
rewrite : {
av : 1,
bangumi : 1,
watchlater : 1,
frame : 1,
home : 1,
playlist : 1,
medialist : 1,
rank : 1
},
reset : {
xhrhook : 1,
danmuku : 1,
livechat : 1,
limit : 0,
accesskey : 0,
grobalboard : 1,
replyfloor : 1,
headblur : 0,
preview : 1,
jointime : 1,
lostvideo : 1,
bvid2av : 1,
selectdanmu : 0,
episodedata : 1,
like : 1,
static : 1,
download : 1,
heartbeat : 0,
carousel : 0,
adloc : 0,
roomplay : 0,
history : 0,
electric : 0,
panel : 0,
midcrc : 0,
viewbofqi : 0,
widescreen : 0,
danmakuoff : 0
}
}
// 统一api接口
const API = {
// 网页框架
pageframe : {
watchlater : '<!DOCTYPE html><html><head><meta charset="utf-8"><title>哔哩哔哩 (゜-゜)つロ 干杯~-bilibili</title><meta name="description" content="bilibili是国内知名的视频弹幕网站,这里有最及时的动漫新番,最棒的ACG氛围,最有创意的Up主。大家可以在这里找到许多欢乐。"><meta name="keywords" content="B站,弹幕,字幕,AMV,MAD,MTV,ANIME,动漫,动漫音乐,游戏,游戏解说,ACG,galgame,动画,番组,新番,初音,洛天依,vocaloid"><meta name="renderer" content="webkit"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="spm_prefix" content="333.342"/><link rel="shortcut icon" href="//static.hdslb.com/images/favicon.ico"><link rel="search" type="application/opensearchdescription+xml" href="//static.hdslb.com/opensearch.xml" title="哔哩哔哩"><link rel="stylesheet" href="//static.hdslb.com/phoenix/dist/css/comment.min.css" type="text/css"><link rel="stylesheet" href="//static.hdslb.com/elec_2/dist/css/later_elec.css" type="text/css"><link rel="stylesheet" href="//static.hdslb.com/tag/css/tag-index2.0.css" type="text/css"><link href="//s1.hdslb.com/bfs/static/phoenix/viewlater/static/css/main.d9641d2f4dc42228ea8c2650e1b98b0b.css" rel="stylesheet"><style type="text/css">#bofqi .player {width:980px;height:620px;display:block;}@media screen and (min-width:1400px){#bofqi .player{width:1160px;height:720px}}</style></head><body><div class="z-top-container has-menu"></div><div id="viewlater-app"><app></app></div><div class="footer bili-footer"></div><script type="text/javascript" src="//static.hdslb.com/js/jquery.min.js"></script><script type="text/javascript" src="//static.hdslb.com/js/jquery.qrcode.min.js"></script><script type="text/javascript" src="//s1.hdslb.com/bfs/seed/jinkela/header/header.js"></script><script type="text/javascript" src="//static.hdslb.com/common/js/footer.js"></script><script type="text/javascript" src="//static.hdslb.com/js/swfobject.js"></script><script type="text/javascript" src="//static.hdslb.com/js/video.min.js"></script><script type="text/javascript" src="//static.hdslb.com/account/bili_quick_login.js"></script><script type="text/javascript" src="//static.hdslb.com/phoenix/dist/js/comment.min.js"></script><script type="text/javascript" src="//static.hdslb.com/mstation/js/upload/moxie.js"></script><script type="text/javascript" src="//static.hdslb.com/mstation/js/upload/plupload.js"></script><script type="text/javascript" src="//static.hdslb.com/elec_2/dist/js/later_elec.js"></script><script type="text/javascript" src="//s1.hdslb.com/bfs/static/phoenix/viewlater/static/js/main.2111469a1bbc20e2e885.js"></script></body></html>',
playlist : '<!DOCTYPE html><html><head><title>哔哩哔哩 (゜-゜)つロ 干杯~-bilibili</title><meta charset=utf-8><meta http-equiv=X-UA-Compatible content="IE=edge"><meta name=renderer content=webkit><meta name=description content=bilibili是国内知名的视频弹幕网站,这里有最及时的动漫新番,最棒的ACG氛围,最有创意的Up主。大家可以在这里找到许多欢乐。><meta name=keywords content=B站,弹幕,字幕,AMV,MAD,MTV,ANIME,动漫,动漫音乐,游戏,游戏解说,ACG,galgame,动画,番组,新番,初音,洛天依,vocaloid><meta name=spm_prefix content=333.44><link rel=stylesheet href=//static.hdslb.com/phoenix/dist/css/comment.min.css type=text/css><meta charset=utf-8><meta http-equiv=X-UA-Compatible content="IE=edge"><meta name=renderer content=webkit><meta name=description content=bilibili是国内知名的视频弹幕网站,这里有最及时的动漫新番,最棒的ACG氛围,最有创意的Up主。大家可以在这里找到许多欢乐。><meta name=keywords content=B站,弹幕,字幕,AMV,MAD,MTV,ANIME,动漫,动漫音乐,游戏,游戏解说,ACG,galgame,动画,番组,新番,初音,洛天依,vocaloid><meta name=spm_prefix content=0><link href=//s1.hdslb.com/bfs/static/jinkela/playlist-video/css/playlist_video.0.87292febba67b03f65d05c15d03e325d9db4f56a.css rel=stylesheet><style type="text/css">#bofqi .player {width:980px;height:620px;display:block;}@media screen and (min-width:1400px){#bofqi .player{width:1160px;height:720px}}</style></head><body><div id=playlist-video-app></div><div class="footer bili-footer report-wrap-module"></div><script type=text/javascript src=//s1.hdslb.com/bfs/static/jinkela/long/js/jquery/jquery1.7.2.min.js></script><script type=text/javascript src=//static.hdslb.com/js/jquery.qrcode.min.js></script><script type=text/javascript charset=utf-8 src=//static.hdslb.com/common/js/footer.js></script><script type=text/javascript src=//static.hdslb.com/js/swfobject.js></script><script type=text/javascript src=//static.hdslb.com/js/video.min.js></script><script type=text/javascript src=//static.hdslb.com/mstation/js/upload/moxie.js></script><script type=text/javascript src=//static.hdslb.com/mstation/js/upload/plupload.js></script><script type=text/javascript src=//static.hdslb.com/phoenix/dist/js/comment.min.js></script><script type=text/javascript src=//s1.hdslb.com/bfs/static/jinkela/playlist-video/1.playlist_video.87292febba67b03f65d05c15d03e325d9db4f56a.js></script><script type=text/javascript src=//s1.hdslb.com/bfs/static/jinkela/playlist-video/playlist_video.87292febba67b03f65d05c15d03e325d9db4f56a.js></script></body></html>',
bangumi : '<!DOCTYPE html><html><head><meta charset="utf-8"><title>哔哩哔哩 (゜-゜)つロ 干杯~-bilibili</title><meta name="description" content="bilibili是国内知名的视频弹幕网站,这里有最及时的动漫新番,最棒的ACG氛围,最有创意的Up主。大家可以在这里找到许多欢乐。"><meta name="keywords" content="Bilibili,哔哩哔哩,哔哩哔哩动画,哔哩哔哩弹幕网,弹幕视频,B站,弹幕,字幕,AMV,MAD,MTV,ANIME,动漫,动漫音乐,游戏,游戏解说,二次元,游戏视频,ACG,galgame,动画,番组,新番,初音,洛天依,vocaloid,日本动漫,国产动漫,手机游戏,网络游戏,电子竞技,ACG燃曲,ACG神曲,追新番,新番动漫,新番吐槽,巡音,镜音双子,千本樱,初音MIKU,舞蹈MMD,MIKUMIKUDANCE,洛天依原创曲,洛天依翻唱曲,洛天依投食歌,洛天依MMD,vocaloid家族,OST,BGM,动漫歌曲,日本动漫音乐,宫崎骏动漫音乐,动漫音乐推荐,燃系mad,治愈系mad,MAD MOVIE,MAD高燃"><meta name="renderer" content="webkit"><meta http-equiv="X-UA-Compatible" content="IE=edge"><link rel="search" type="application/opensearchdescription+xml" href="//static.hdslb.com/opensearch.xml" title="哔哩哔哩"><link rel="stylesheet" href="//static.hdslb.com/phoenix/dist/css/comment.min.css" type="text/css" /><script type="text/javascript" src="//static.hdslb.com/js/jquery.min.js"></script><script type="text/javascript" src="//static.hdslb.com/js/video.min.js"></script><script type="text/javascript" src="//static.hdslb.com/vip/dist/js/vipPlugin.v2.js"></script><script type="text/javascript" src="//static.hdslb.com/js/promise.auto.min.js"></script><script type="text/javascript" src="//s1.hdslb.com/bfs/seed/jinkela/header/header.js"></script><link rel="stylesheet" href="//s1.hdslb.com/bfs/static/bangumi/play/css/bangumi-play.0.809bd6f6d1fba866255d2e6c5dc06dabba9ce8b4.css" /></head><body><div class="z-top-container has-menu"></div><div id="app" data-server-rendered="true" class="main-container"></div><script src="//s1.hdslb.com/bfs/static/bangumi/play/1.bangumi-play.809bd6f6d1fba866255d2e6c5dc06dabba9ce8b4.js" crossorigin="" defer="defer"></script><script src="//s1.hdslb.com/bfs/static/bangumi/play/bangumi-play.809bd6f6d1fba866255d2e6c5dc06dabba9ce8b4.js" crossorigin="" defer="defer"></script><script type="text/javascript">0</script><div class="footer bili-footer report-wrap-module" id="home_footer"></div><script type="text/javascript" src="//static.hdslb.com/common/js/footer.js"></script><script src="//s1.hdslb.com/bfs/static/plugin/vip/BilAccountThaw.js"></script></body></html>',
detail : '<!DOCTYPE html><html xmlns="http://www.w3.org/1999/xhtml"><head><title>哔哩哔哩 (゜-゜)つロ 干杯~-bilibili</title><meta charset="utf-8" /><meta http-equiv="X-UA-Compatible" content="IE=edge" /><meta name="renderer" content="webkit" /><meta name="spm_prefix" content="333.43" /><meta name="description" content="bilibili是国内知名的视频弹幕网站,这里有最及时的动漫新番,最棒的ACG氛围,最有创意的Up主。大家可以在这里找到许多欢乐。" /><link rel="stylesheet" href="//static.hdslb.com/phoenix/dist/css/comment.min.css" type="text/css" /><meta name="keywords" content="B站,弹幕,字幕,AMV,MAD,MTV,ANIME,动漫,动漫音乐,游戏,游戏解说,ACG,galgame,动画,番组,新番,初音,洛天依,vocaloid" /><script type="text/javascript" src="//static.hdslb.com/js/jquery.min.js"></script><link rel="preload" href="//s1.hdslb.com/bfs/static/jinkela/playlist-detail/manifest.dc2f20722afb93b15bbf7a30436f70ff31fb0a05.js" as="script" /><link rel="preload" href="//s1.hdslb.com/bfs/static/jinkela/playlist-detail/vendor.dc2f20722afb93b15bbf7a30436f70ff31fb0a05.js" as="script" /><link rel="preload" href="//s1.hdslb.com/bfs/static/jinkela/playlist-detail/playlist_detail.dc2f20722afb93b15bbf7a30436f70ff31fb0a05.js" as="script" /><link rel="preload" href="//s1.hdslb.com/bfs/static/jinkela/playlist-detail/css/playlist_detail.1.dc2f20722afb93b15bbf7a30436f70ff31fb0a05.css" as="style" /><link rel="stylesheet" href="//s1.hdslb.com/bfs/static/jinkela/playlist-detail/css/playlist_detail.1.dc2f20722afb93b15bbf7a30436f70ff31fb0a05.css" /></head><body><div id="playlist-detail-app"></div><div id="app" data-server-rendered="true" class="pl-app"></div><script src="//s1.hdslb.com/bfs/static/jinkela/playlist-detail/manifest.dc2f20722afb93b15bbf7a30436f70ff31fb0a05.js" defer="defer"></script><script src="//s1.hdslb.com/bfs/static/jinkela/playlist-detail/vendor.dc2f20722afb93b15bbf7a30436f70ff31fb0a05.js" defer="defer"></script><script src="//s1.hdslb.com/bfs/static/jinkela/playlist-detail/playlist_detail.dc2f20722afb93b15bbf7a30436f70ff31fb0a05.js" defer="defer"></script><div class="footer bili-footer report-wrap-module"></div><script type="text/javascript" charset="utf-8" src="//static.hdslb.com/common/js/footer.js"></script><script type="text/javascript" src="//static.hdslb.com/phoenix/dist/js/comment.min.js"></script></body></html>',
cinema : '<!DOCTYPE html><html><head><meta charset="utf-8"><title>哔哩哔哩 (゜-゜)つロ 干杯~-bilibili</title><meta name="description" content="bilibili是国内知名的视频弹幕网站,这里有最及时的动漫新番,最棒的ACG氛围,最有创意的Up主。大家可以在这里找到许多欢乐。"><meta name="keywords" content="Bilibili,哔哩哔哩,哔哩哔哩动画,哔哩哔哩弹幕网,弹幕视频,B站,弹幕,字幕,AMV,MAD,MTV,ANIME,动漫,动漫音乐,游戏,游戏解说,二次元,游戏视频,ACG,galgame,动画,番组,新番,初音,洛天依,vocaloid,日本动漫,国产动漫,手机游戏,网络游戏,电子竞技,ACG燃曲,ACG神曲,追新番,新番动漫,新番吐槽,巡音,镜音双子,千本樱,初音MIKU,舞蹈MMD,MIKUMIKUDANCE,洛天依原创曲,洛天依翻唱曲,洛天依投食歌,洛天依MMD,vocaloid家族,OST,BGM,动漫歌曲,日本动漫音乐,宫崎骏动漫音乐,动漫音乐推荐,燃系mad,治愈系mad,MAD MOVIE,MAD高燃"><meta name="renderer" content="webkit"><meta http-equiv="X-UA-Compatible" content="IE=edge"><link rel="search" type="application/opensearchdescription+xml" href="//static.hdslb.com/opensearch.xml" title="哔哩哔哩"><link rel="stylesheet" href="//static.hdslb.com/phoenix/dist/css/comment.min.css" type="text/css" /><script type="text/javascript" src="//static.hdslb.com/js/jquery.min.js"></script><script type="text/javascript" src="//static.hdslb.com/js/video.min.js"></script><script type="text/javascript" src="//static.hdslb.com/vip/dist/js/vipPlugin.v2.js"></script><script type="text/javascript" src="//static.hdslb.com/js/promise.auto.min.js"></script><script type="text/javascript" src="//s1.hdslb.com/bfs/seed/jinkela/header/header.js"></script><link rel="stylesheet" href="//s1.hdslb.com/bfs/static/bangumi/play/css/bangumi-play.0.809bd6f6d1fba866255d2e6c5dc06dabba9ce8b4.css" /></head><body><div class="z-top-container " style="height:42px"></div><div id="app" data-server-rendered="true" class="main-container special"></div><script src="//s1.hdslb.com/bfs/static/bangumi/play/1.bangumi-play.809bd6f6d1fba866255d2e6c5dc06dabba9ce8b4.js" crossorigin="" defer="defer"></script><script src="//s1.hdslb.com/bfs/static/bangumi/play/bangumi-play.809bd6f6d1fba866255d2e6c5dc06dabba9ce8b4.js" crossorigin="" defer="defer"></script><script type="text/javascript">0</script><div class="footer bili-footer report-wrap-module" id="home_footer"></div><script type="text/javascript" src="//static.hdslb.com/common/js/footer.js"></script><script src="//s1.hdslb.com/bfs/static/plugin/vip/BilAccountThaw.js"></script></body></html>',
video : '<!DOCTYPE html><html><head><meta charset="utf-8"><title>哔哩哔哩 (゜-゜)つロ 干杯~-bilibili</title><meta name="description" content="bilibili是国内知名的视频弹幕网站,这里有最及时的动漫新番,最棒的ACG氛围,最有创意的Up主。大家可以在这里找到许多欢乐。"><meta name="keywords" content="Bilibili,哔哩哔哩,哔哩哔哩动画,哔哩哔哩弹幕网,弹幕视频,B站,弹幕,字幕,AMV,MAD,MTV,ANIME,动漫,动漫音乐,游戏,游戏解说,二次元,游戏视频,ACG,galgame,动画,番组,新番,初音,洛天依,vocaloid,日本动漫,国产动漫,手机游戏,网络游戏,电子竞技,ACG燃曲,ACG神曲,追新番,新番动漫,新番吐槽,巡音,镜音双子,千本樱,初音MIKU,舞蹈MMD,MIKUMIKUDANCE,洛天依原创曲,洛天依翻唱曲,洛天依投食歌,洛天依MMD,vocaloid家族,OST,BGM,动漫歌曲,日本动漫音乐,宫崎骏动漫音乐,动漫音乐推荐,燃系mad,治愈系mad,MAD MOVIE,MAD高燃"><meta name="renderer" content="webkit"><meta http-equiv="X-UA-Compatible" content="IE=edge"><link rel="search" type="application/opensearchdescription+xml" href="//static.hdslb.com/opensearch.xml" title="哔哩哔哩"><link rel="stylesheet" href="//s1.hdslb.com/bfs/static/jinkela/videoplay/css/video.0.406cee7878545872b8dfbe73071d665dfb287c67.css" /><style type="text/css">#bofqi .player {width:980px;height:620px;display:block;}@media screen and (min-width:1400px){#bofqi .player{width:1160px;height:720px}} .video-info-m .number .like b, .video-info-m .number .like i {background : url(//static.hdslb.com/images/base/icons.png);}</style></head><body><script type="text/javascript" src="//static.hdslb.com/js/jquery.min.js"></script><div class="z-top-container has-menu"></div><div id="video-page-app"></div><div id="app" data-server-rendered="true"></div><div class="bili-wrapper" id="bofqi"></div><div class="footer bili-footer report-wrap-module"></div><script type="text/javascript" src="//s1.hdslb.com/bfs/seed/jinkela/header/header.js"></script><script type="text/javascript" src="//static.hdslb.com/js/video.min.js"></script><script type="text/javascript">function getQueryString(e){var r=new RegExp("(^|&)"+e+"=([^&]*)(&|$)"),i=window.location.search.substr(1).match(r);return null!=i?unescape(i[2]):null}window.getInternetExplorerVersion=function(){var e=-1;if("Microsoft Internet Explorer"==navigator.appName){var r=navigator.userAgent;null!=new RegExp("MSIE ([0-9]{1,}[.0-9]{0,})").exec(r)&&(e=parseFloat(RegExp.$1))}return e};var vd=window.__INITIAL_STATE__&&window.__INITIAL_STATE__.videoData;if(vd&&vd.aid&&9!==getInternetExplorerVersion()){if($("#__bofqi").innerHTML=\'<div class="bili-wrapper" id="bofqi"><div id="player_placeholder"></div></div>\',vd.embedPlayer){var p=getQueryString("p")?getQueryString("p")-1:0,player={aid:vd.aid,cid:vd.pages[p]&&vd.pages[p].cid||vd.pages[0].cid};EmbedPlayer("player","//static.hdslb.com/play.swf","cid="+player.cid+"&aid="+player.aid+"&pre_ad=")}vd.embed&&$("#bofqi").html(vd.embed)}else $("#bofqi").remove()</script><script src="//s1.hdslb.com/bfs/static/jinkela/videoplay/manifest.b1b7706abd590dd295794f540f7669a5d8d978b3.js" crossorigin="" defer="defer"></script><script src="//s1.hdslb.com/bfs/static/jinkela/videoplay/vendor.b1b7706abd590dd295794f540f7669a5d8d978b3.js" crossorigin="" defer="defer"></script><script src="//s1.hdslb.com/bfs/static/jinkela/videoplay/video.b1b7706abd590dd295794f540f7669a5d8d978b3.js" crossorigin="" defer="defer"></script><script type="text/javascript" src="//static.hdslb.com/phoenix/dist/js/comment.min.js"></script><link rel="stylesheet" href="//static.hdslb.com/phoenix/dist/css/comment.min.css" type="text/css" /><script type="text/javascript" src="//static.hdslb.com/js/jquery.qrcode.min.js"></script><script type="text/javascript" charset="utf-8" src="//static.hdslb.com/common/js/footer.js"></script></body></html>',
home : '<!DOCTYPE html><html lang="zh-Hans"><head><meta charset="utf-8"><title>哔哩哔哩 (゜-゜)つロ 干杯~-bilibili</title><meta name="description" content="bilibili是国内知名的视频弹幕网站,这里有最及时的动漫新番,最棒的ACG氛围,最有创意的Up主。大家可以在这里找到许多欢乐。"><meta name="keywords" content="Bilibili,哔哩哔哩,哔哩哔哩动画,哔哩哔哩弹幕网,弹幕视频,B站,弹幕,字幕,AMV,MAD,MTV,ANIME,动漫,动漫音乐,游戏,游戏解说,二次元,游戏视频,ACG,galgame,动画,番组,新番,初音,洛天依,vocaloid,日本动漫,国产动漫,手机游戏,网络游戏,电子竞技,ACG燃曲,ACG神曲,追新番,新番动漫,新番吐槽,巡音,镜音双子,千本樱,初音MIKU,舞蹈MMD,MIKUMIKUDANCE,洛天依原创曲,洛天依翻唱曲,洛天依投食歌,洛天依MMD,vocaloid家族,OST,BGM,动漫歌曲,日本动漫音乐,宫崎骏动漫音乐,动漫音乐推荐,燃系mad,治愈系mad,MAD MOVIE,MAD高燃"><meta name="renderer" content="webkit"><meta http-equiv="X-UA-Compatible" content="IE=edge"><link rel="search" type="application/opensearchdescription+xml" href="//static.hdslb.com/opensearch.xml" title="哔哩哔哩"><script type="text/javascript" src="//static.hdslb.com/js/jquery.min.js"></script><link rel="stylesheet" href="//s1.hdslb.com/bfs/static/jinkela/home/css/home.0.4eadf4209b1762230047120e0a9945a9f3b56fd1.css"></head><body><div id="home-app"></div><div id="app" data-server-rendered="true"></div><script src="//s1.hdslb.com/bfs/seed/jinkela/header/header.js"></script></script><script src="//s1.hdslb.com/bfs/static/jinkela/home/1.home.4eadf4209b1762230047120e0a9945a9f3b56fd1.js" defer></script><script src="//s1.hdslb.com/bfs/static/jinkela/home/home.4eadf4209b1762230047120e0a9945a9f3b56fd1.js" defer></script><div class="footer bili-footer report-wrap-module"></div><script type="text/javascript" src="//s1.hdslb.com/bfs/cm/st/bundle.js" crossorigin></script><script type="text/javascript" defer="defer" charset="utf-8" src="//static.hdslb.com/common/js/footer.js"></script><link rel="prefetch" as="script" href="//static.hdslb.com/js/video.min.js"></body></html>',
rank : '<!DOCTYPE html><html lang="zh-Hans" xmlns="http://www.w3.org/1999/xhtml" xml:lang="zh-Hans"><head><title>热门视频排行榜 - 哔哩哔哩 (゜-゜)つロ 干杯~-bilibili</title><meta charset="utf-8" /><meta name="spm_prefix" content="333.158" /><meta http-equiv="X-UA-Compatible" content="IE=edge" /><meta name="renderer" content="webkit" /><meta name="description" content="bilibili是国内知名的视频弹幕网站,这里有最及时的动漫新番,最棒的ACG氛围,最有创意的Up主。大家可以在这里找到许多欢乐。" /><meta name="keywords" content="B站,弹幕,字幕,AMV,MAD,MTV,ANIME,动漫,动漫音乐,游戏,游戏解说,ACG,galgame,动画,番组,新番,初音,洛天依,vocaloid" /><link rel="preload" href="//s1.hdslb.com/bfs/static/jinkela/rank/1.rank.ba58f8684a87651e0e1c576df8f918bfa10c1a90.js" as="script" /><link rel="preload" href="//s1.hdslb.com/bfs/static/jinkela/rank/css/rank.0.ba58f8684a87651e0e1c576df8f918bfa10c1a90.css" as="style" /><link rel="preload" href="//s1.hdslb.com/bfs/static/jinkela/rank/rank.ba58f8684a87651e0e1c576df8f918bfa10c1a90.js" as="script" /><link rel="stylesheet" href="//s1.hdslb.com/bfs/static/jinkela/rank/css/rank.0.ba58f8684a87651e0e1c576df8f918bfa10c1a90.css" /></head><body><div class="z-top-container has-menu"></div><div id="rank-app"></div><script type="text/javascript" src="//s1.hdslb.com/bfs/static/jinkela/long/js/jquery/jquery1.7.2.min.js"></script><script type="text/javascript" src="//s1.hdslb.com/bfs/seed/jinkela/header/header.js" defer="defer"></script><script type="text/javascript" src="//s1.hdslb.com/bfs/cm/st/bundle.js" crossorigin=""></script><div id="app" data-server-rendered="true"></div><script src="//s1.hdslb.com/bfs/static/jinkela/rank/1.rank.ba58f8684a87651e0e1c576df8f918bfa10c1a90.js" defer="defer"></script><script src="//s1.hdslb.com/bfs/static/jinkela/rank/rank.ba58f8684a87651e0e1c576df8f918bfa10c1a90.js" defer="defer"></script><div class="footer bili-footer report-wrap-module"></div><script type="text/javascript" src="//static.hdslb.com/common/js/footer.js" defer="defer"></script></body></html>'
},
// 样式表
style : {
playshadow : "#bilibiliPlayer, #bofqi.mini-player {box-shadow : 0px 2px 8px 0px rgba(0,160,216,0.3) !important;}",
download : "#bili-old-download-table {position : fixed;z-index : 3300;bottom : 0;background : #f6f6f6;width : 100%;text-align : center;}#bili-old-download-table .download-box {background-color : #fff;color : #000 !important;border : #ccc 1px solid;border-radius : 3px;display : inline-block;margin : 3px;}.download-mp4 {color : #fff !important;background-color : #c0f;background-image : linear-gradient(to right, #c0f, #90f);}.download-avc {color : #fff !important;background-color : #f00;background-image : linear-gradient(to right, #f00, #c00);}.download-hev {color : #fff !important;background-color : #ffe42b;background-image : linear-gradient(to right, #ffe42b, #dfb200);}.download-aac {color : #fff !important;background-color : #0d0;background-image : linear-gradient(to right, #0d0, #0a0);}.download-flv {color : #fff !important;background-color : #f90;background-image : linear-gradient(to right, #f90, #d70);}.download-type {color : #000 !important;display : table-cell;min-width : 1.5em;padding : 1px 3px;text-align : center;vertical-align : middle;}#bili-old-download-table a {display : table-cell;padding : 3px;text-decoration : none;}.quality-high {background-color : #c0f;}.quality-1080p {background-color : #f00;}.quality-720p {background-color : #f90;}.quality-480p {background-color : #00d;}.quality-360p {background-color : #0d0;}.download-quality {color : #fff !important;padding : 1px 3px;text-align : center;}.download-size {font-size : 90%;margin-top : 2px;padding : 1px 3px;text-align : center;}",
jointime : ".user .info .meta .row {height : 88px;white-space : normal;}.user .info .jointime .icon {background-position : -209px -84px;}.user .info .jointime .text {color : #00a1d6;}}",
online : ".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;}",
search : ".search-wrap .search-block .input-wrap input {font : 400 13.3333px Arial !important;}",
uiface : "#ui-face {box-sizing : content-box;color : #fff;background-color : rgb(255,255,255);border-radius:5px;position : fixed;padding : 4px;bottom : 65px;width : 56px;height : 40px;transition : right 0.7s;-moz-transition : right 0.7s;-webkit-transition : right 0.7s;-o-transition : right 0.7s;z-index : 1008;}#ui-face i {background-position : -471px -982px;display : block;width : 20px;height : 20px;margin : auto;transition : 0.2s;background-image : url(//static.hdslb.com/images/base/icons.png);}#ui-face span {font-size : 14px;display : block;width : 50%;margin : auto;transition : 0.2s;color : rgb(0,0,0)}#ui-table {box-sizing : content-box;color : #fff;background-color : rgb(255,255,255);border-radius:5px;font-size : 14px;position : fixed;padding : 4px;bottom : 30px;right : 58px;width : 200px;height : 360px;line-height : normal;box-shadow : rgba(0, 85, 255, 0.098) 0px 0px 20px 0px;border : 1px solid rgb(233, 234, 236);overflow-y : scroll;z-index : 10008;}.checke{float : right;position : relative;-webkit-appearance : none;width : 40px;height : 20px;line-height : 20px;background : #eee;border-radius : 10px;outline : none;border : 2px solid #999999;}.checke:before{position : absolute;left : 0;content : '';width : 12px;height : 12px;border-radius : 50%;background : #eee;box-shadow : 0px 0px 5px #ddd;transition : all 0.2s linear;border : 2px solid #999999;}.checke:checked{ background : #01a1d6;}.checke:checked:before{left : 20px;transition : all 0.2s linear;}#ui-state {border-radius : 5px;z-index : 1000;width : auto;position : fixed;right : 280px;color : #fff;background : #0008;padding : 1rem;font-size : 12pt;top : 50%;transform : translateY(-50%);transition : .2s ease-out .8s;max-width : 20%;line-height : 2;white-space : pre-wrap;pointer-events : none;opacity : 1;}.video_download {cursor : pointer;width : 46px;height : 48px;background-color : #f6f9fa;background-position : -1353px -1095px;background-repeat : no-repeat;border : 1px solid #e5e9ef;overflow : hidden;border-radius : 4px;display : inline-block;background-image : url(//static.hdslb.com/images/base/icons.png);}.video_download:hover {background-color : #00a1d6;border-color : #00a1d6;}.bili-header-m .head-banner{background-position: center 0 !important;background-size: cover !important;}",
bofqi : "#bofqi .player {width:980px;height:620px;display:block;}@media screen and (min-width:1400px){#bofqi .player{width:1160px;height:720px}}",
gray : "html {filter:grayscale(100%);-webkit-filter:grayscale(100%);-moz-filter:grayscale(100%);-ms-filter:grayscale(100%);-o-filter:grayscale(100%);filter:progid:DXImageTransform.Microsoft.BasicImage(grayscale=1);-webkit-filter:grayscale(1);}",
like : ".video-info-module .number .like b, .video-info-module .number .like i {background : url(//static.hdslb.com/images/base/icons.png);display : inline-block;margin-top : -3px;vertical-align : middle;}",
},
// 播放器框架
playerframe : {
html5player : "https://www.bilibili.com/blackboard/html5player.html", // aid, cid, season_type player_type + &as_wide=1
playlist : "https://www.bilibili.com/blackboard/playlist-player.html", // pl || aid, cid
ancient : "https://www.bilibili.com/blackboard/activity-ancient-player.html", // aid, cid
player : "https://player.bilibili.com/player.html", // aid,cid &| page
},
// URL
url : {
spacedetial : "https://api.bilibili.com/medialist/gateway/base/spaceDetail", // media_id, pn + &ps=20&keyword=&order=mtime&type=0&tid=0
channel : "https://api.bilibili.com/x/space/channel/video", // mid, cid, pn + &ps=30&order=0
biliplus : "https://www.biliplus.com/video/av",
jijidown : "https://www.jijidown.com/video/av",
online : "https://api.bilibili.com/x/web-interface/online",
stat : "https://api.bilibili.com/x/web-interface/archive/stat", // aid
replymain : "https://api.bilibili.com/x/v2/reply/main", // oid, type, mode &| next
reply : "https://api.bilibili.com/x/v2/reply", // type,sort,oid,pn
replycursor : "https://api.bilibili.com/x/v2/reply/reply/cursor", // oid, root, type &| sort
replydialog : "https://api.bilibili.com/x/v2/reply/dialog/cursor",
membercard : "https://account.bilibili.com/api/member/getCardByMid", // mid
season : "https://api.bilibili.com/pgc/view/web/season", // season_id || ep_id
pagelist : "https://api.bilibili.com/x/player/pagelist", // aid
view : "https://api.bilibili.com/x/web-interface/view", // aid || bvid
haslike : "https://api.bilibili.com/x/web-interface/archive/has/like", // aid
like : "https://api.bilibili.com/x/web-interface/archive/like",
ids4Player : "https://api.bilibili.com/x/v1/medialist/resource/ids4Player", // media_id
cards : "https://api.bilibili.com/x/article/cards", // ids
medialist : "https://api.bilibili.com/x/v1/medialist/detail", // media_id && pn=1&ps=1
x : "https://api.bilibili.com/x/player/playurl", // avid | bvid, cid, qn + fourk=1&type=&otype=json |+ &fnver=0&fnval=16
pgc : "https://api.bilibili.com/pgc/player/web/playurl", // avid | bvid, cid, qn + fourk=1&type=&otype=json |+ &fnver=0&fnval=16
sign : "https://interface.bilibili.com/v2/playurl", // appkey, cid=, otype=json, qn, quality, type
proj : "https://app.bilibili.com/v2/playurlproj", // appkey, cid=, otype=json, qn
pgcproj : "https://api.bilibili.com/pgc/player/api/playurlproj", // appkey, cid=, otype=json, platform=android_i, qn
BPplayurl : "https://www.biliplus.com/BPplayurl.php", // [origin] + &module=pgc&balh_ajax=1
ranklist : "https://api.bilibili.com/pgc/season/rank/web/list", // season_type, &day=3
detail : "https://api.bilibili.com/x/web-interface/view/detail", // aid
playlist : "https://api.bilibili.com/x/playlist/video/toview", // pid
card : "https://api.bilibili.com/x/web-interface/card", // mid
listso : "https://api.bilibili.com/x/v1/dm/list.so", //oid
ranking : "https://api.bilibili.com/x/web-interface/ranking" // rid=0&day=3&type=1&arc_type=0
},
// 未识别分区对照表
sort : {
1 : [1, "动画", "https://www.bilibili.com/v/douga/"],
3 : [3, "音乐", "https://www.bilibili.com/v/music/"],
29 : [3, "音乐现场", "https://www.bilibili.com/v/music/live"],
36 : [36, "科技", "https://www.bilibili.com/v/technology"],
86 : [1, "特摄", "https://www.bilibili.com/v/douga/"],
95 : [188, "手机平板", "https://www.bilibili.com/v/digital/mobile/"],
129 : [129, "舞蹈", "https://www.bilibili.com/v/dance"],
155 : [155, "时尚", "https://www.bilibili.com/v/fashion"],
160 : [160, "生活", "https://www.bilibili.com/v/life"],
168 : [168, "国创", "https://www.bilibili.com/guochuang"],
176 : [160, "汽车", "https://www.bilibili.com/v/life/automobile"],
188 : [188, "数码", "https://www.bilibili.com/v/digital"],
189 : [188, "电脑装机", "https://www.bilibili.com/v/digital/pc"],
190 : [188, "数码摄影", "https://www.bilibili.com/v/digital/photography"],
191 : [188, "影音智能", "https://www.bilibili.com/v/digital/intelligence_av"],
192 : [155, "风尚标", "https://www.bilibili.com/v/fashion/trends"],
193 : [3, "MV", "https://www.bilibili.com/v/music/mv"],
194 : [3, "电音", "https://www.bilibili.com/v/music/electronic"],
195 : [168, "动态漫·广播剧", "https://www.bilibili.com/v/guochuang/motioncomic"],
198 : [129, "街舞", "https://www.bilibili.com/v/dance/hiphop"],
199 : [129, "明星舞蹈", "https://www.bilibili.com/v/dance/star"],
200 : [129, "中国舞", "https://www.bilibili.com/v/dance/china"],
201 : [36, "科学科普", "https://www.bilibili.com/v/technology/science"],
202 : [202, "资讯", "https://www.bilibili.com/v/information/"],
203 : [202, "热点", "https://www.bilibili.com/v/information/hotspot/"],
204 : [202, "环球", "https://www.bilibili.com/v/information/global/"],
205 : [202, "社会", "https://www.bilibili.com/v/information/social/"],
206 : [202, "综合", "https://www.bilibili.com/v/information/multiple/"],
207 : [36, "财经", "https://www.bilibili.com/v/technology/finance"],
208 : [36, "校园学习", "https://www.bilibili.com/v/technology/campus"],
209 : [36, "职业职场", "https://www.bilibili.com/v/technology/career"],
210 : [1, "手办·模玩", "https://www.bilibili.com/v/douga/garage_kit"]
},
// 播放器通知
message : [
['https://www.bilibili.com/blackboard/activity-4KPC.html', '解锁超清4K画质'],
['https://www.bilibili.com/blackboard/activity-4K120FPS-PC.html', '4K120FPS投稿全量开放'],
['https://www.bilibili.com/blackboard/bilibili2009.html', '十年前的B站长啥样'],
['https://www.bilibili.com/blackboard/html5playerhelp.html', 'HTML5播放器试用'],
]
}
// 调试模块封装
const debug = {
log : (...msg) => console.log("[" + deliver.timeFormat(new Date()) + "]", "[Bilibili Old]", ...msg),
error : (...msg) => console.error("[" + deliver.timeFormat(new Date()) + "]", "[Bilibili Old]", ...msg),
warn : (...msg) => console.warn("[" + deliver.timeFormat(new Date()) + "]", "[Bilibili Old]", ...msg),
debug : (...msg) => console.debug("[" + deliver.timeFormat(new Date()) + "]", "[Bilibili Old]", ...msg),
msg : (...msg) => {deliver.debug(...msg)}
}
// XMLHttpReques封装,除同步方法外统一返回promise
const xhr = {
// 同步方法
'false' : (url) => {
const xhr = new XMLHttpRequest();
xhr.open('GET', url, false);
xhr.withCredentials = true;
xhr.send(null);
return xhr.responseText;
},
// 异步方法
'true' : (url) => {
return new Promise((resolve, reject) => {
let xhr = new XMLHttpRequest();
xhr.open('get', url, true);
xhr.withCredentials = true;
xhr.onload = () => resolve(xhr.response);
xhr.onerror = () => reject(xhr.statusText || url + " net::ERR_CONNECTION_TIMED_OUT");
xhr.send();
});
},
// 跨域方法
GM : (url) => {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method : "GET",
url : url,
onload : (xhr) => resolve(xhr.responseText),
onerror : (xhr) => reject(xhr.statusText || url + " net::ERR_CONNECTION_TIMED_OUT"),
});
})
},
// 表单方法
post : (url, header, data) => {
return new Promise((resolve, reject) => {
let xhr = new XMLHttpRequest();
header = header ? header : "application/x-www-form-urlencoded";
xhr.open('post', url, true);
xhr.setRequestHeader("Content-type", header);
xhr.withCredentials = true;
xhr.onload = () => resolve(xhr.response);
xhr.onerror = () => reject(xhr.statusText || url + " net::ERR_CONNECTION_TIMED_OUT");
xhr.send(data);
});
}
}
// 重构__INITIAL_STATE__
const INITIAL_STATE = {
// av/BV
av : (data) => {
try {
data = deliver.xhrJsonCheck(data).data;
aid = aid || data.View.aid;
cid = cid || data.View.cid;
let dat = {aid : -1, comment : {count : 0, list : []}, error : {}, isClient : false, p: "", player: "", playurl: {}, related : [], tags : [], upData : {}, videoData : {}}
dat.aid = data.View.aid;
dat.related = data.Related;
dat.tags = data.Tags;
dat.upData = data.Card.card;
dat.upData.archiveCount = data.Card.archive_count;
dat.videoData = data.View;
dat.videoData.embedPlayer = 'EmbedPlayer("player", "//static.hdslb.com/play.swf", "cid=' + cid +'&aid=' + aid +'&pre_ad=")'
return dat;
}
catch (e) {e = Array.isArray(e) ? e : [e]; debug.error("__INITIAL_STATE__·av", ...e)}
},
// bangumi
bangumi : (epId) => {
try {
let ep = 0, rp = {}, ini = {}, pug = {}, mode;
let dat = {"ver":{},"loginInfo":{},"canReview":false,"userShortReview":{},"userLongReview":{},"userScore":0,"userCoined":false,"isPlayerTrigger":false,"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,"epStat":{},"ssStat":{}};
if (DOCUMENT.startsWith("{")) {
// rp为api获取到的备用数据保存于DOCUMENT中,作为DOCUMENT被404的备用数据源,无法获取播放进度信息,以ss进入默认选择第一p
rp = deliver.xhrJsonCheck(DOCUMENT).result;
dat.special = rp.bkg_cover ? true : false;
if (epId) {dat.epId = 1 * epId; ep = 1;} else dat.epId = ""
dat.ssId = rp.season_id;
dat.mdId = rp.media_id;
dat.mediaInfo = {};
dat.mediaInfo.actors = rp.actors || "";
dat.mediaInfo.alias = rp.alias;
dat.mediaInfo.areas = rp.areas || [];
dat.mediaInfo.bkg_cover = rp.bkg_cover;
dat.mediaInfo.cover = rp.cover;
dat.mediaInfo.evaluate = rp.evaluate;
dat.mediaInfo.is_paster_ads = rp.is_paster_ads || 0;
dat.mediaInfo.jp_title = rp.jp_title;
dat.mediaInfo.link = rp.link;
dat.mediaInfo.media_id = rp.media_id;
dat.mediaInfo.mode = rp.mode;
dat.mediaInfo.paster_text = "";
dat.mediaInfo.season_id = rp.season_id;
dat.mediaInfo.season_status = rp.status;
dat.mediaInfo.season_title = rp.season_title;
dat.mediaInfo.season_type = rp.type;
dat.mediaInfo.square_cover = rp.square_cover;
dat.mediaInfo.staff = rp.staff || "";
dat.mediaInfo.stat = rp.state;
dat.mediaInfo.style = rp.style || [];
dat.mediaInfo.title = rp.title;
dat.mediaInfo.total_ep = rp.total;
dat.mediaRating = rp.rating;
dat.epList = rp.episodes;
if (ep == 0) dat.epId = (rp.episodes[0] && rp.episodes[0].id) || "";
for (let i = 0; i < dat.epList.length; i++) {
dat.epList[i].ep_id = dat.epList[i].id;
dat.epList[i].episode_status = dat.epList[i].status;
dat.epList[i].index = dat.epList[i].title;
dat.epList[i].index_title = dat.epList[i].long_title;
if(dat.epList[i].ep_id == dat.epId) dat.epInfo = dat.epList[i];
if (dat.epList[i].badge == "会员" || dat.epList[i].badge_type) ids.push(dat.epList[i].cid);
}
dat.newestEp = rp.new_ep;
dat.seasonList = rp.seasons;
dat.rightsInfo = rp.rights;
dat.pubInfo = rp.publish;
dat.upInfo = rp.up_info || {};
}
else {
// 正常DOCUMENT数据源,up组主数据可能无效,将指向uid=2(站长)
ini = JSON.parse(DOCUMENT.match(/INITIAL_STATE__=.+?\;\(function/)[0].replace(/INITIAL_STATE__=/,"").replace(/;\(function/,""));
pug = JSON.parse(DOCUMENT.match(/PGC_USERSTATE__=.+?<\/script>/)[0].replace(/PGC_USERSTATE__=/,"").replace(/<\/script>/,""));
dat.special = ini.mediaInfo.specialCover ? true : false;
mode = dat.special ? 1 : 2;
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.mediaInfo.ssId;
dat.mdId = ini.mediaInfo.id;
dat.mediaInfo = {};
dat.mediaInfo.actors = "";
dat.mediaInfo.alias = ini.mediaInfo.alias;
dat.mediaInfo.areas = [];
dat.mediaInfo.bkg_cover = ini.mediaInfo.specialCover;
dat.mediaInfo.cover = ini.mediaInfo.cover;
dat.mediaInfo.evaluate = ini.mediaInfo.evaluate;
dat.mediaInfo.is_paster_ads = 0;
dat.mediaInfo.jp_title = ini.mediaInfo.jpTitle;
dat.mediaInfo.link = "https://www.bilibili.com/bangumi/media/md" + dat.mdId;
dat.mediaInfo.media_id = dat.mdId;
dat.mediaInfo.mode = mode;
dat.mediaInfo.paster_text = "";
dat.mediaInfo.season_id = ini.mediaInfo.ssId;
dat.mediaInfo.season_status = ini.mediaInfo.status;
dat.mediaInfo.season_title = ini.mediaInfo.title;
dat.mediaInfo.season_type = ini.mediaInfo.ssType;
dat.mediaInfo.square_cover = ini.mediaInfo.squareCover;
dat.mediaInfo.staff = "";
dat.mediaInfo.stat = ini.mediaInfo.stat;
dat.mediaInfo.style = [];
dat.mediaInfo.title = ini.mediaInfo.title;
dat.mediaInfo.total_ep = ini.epList.length;
dat.mediaRating = ini.mediaInfo.rating;
dat.epList = [];
for (let i = 0; i < ini.sections.length; i++) for (let j = 0; j < ini.sections[i].epList.length; j++) ini.epList.push(ini.sections[i].epList[j]);
if (ep == 0) dat.epId = (ini.epList[0] && ini.epList[0].id) || "";
for (let i = 0; i < ini.epList.length; i++) {
dat.epList[i] = {};
dat.epList[i].aid = ini.epList[i].aid;
dat.epList[i].cid = ini.epList[i].cid;
dat.epList[i].badge = ini.epList[i].badge;
dat.epList[i].badge_type = ini.epList[i].badgeType;
dat.epList[i].cover = ini.epList[i].cover;
dat.epList[i].duration = -1;
dat.epList[i].ep_id = ini.epList[i].id;
dat.epList[i].episode_status = ini.epList[i].epStatus;
dat.epList[i].from = ini.epList[i].from;
dat.epList[i].index = ini.epList[i].title;
dat.epList[i].index_title = ini.epList[i].longTitle;
dat.epList[i].mid = ini.mediaInfo.upInfo.mid;
dat.epList[i].page = 1;
dat.epList[i].pub_real_time = ini.epList[i].releaseDate || ini.mediaInfo.pub.time;
dat.epList[i].section_id = -1;
dat.epList[i].section_type = 0;
dat.epList[i].vid = ini.epList[i].vid;
if (dat.epList[i].ep_id == dat.epId) dat.epInfo = dat.epList[i];
if (dat.epList[i].badge == "会员" || dat.epList[i].badge_type) ids.push(dat.epList[i].cid);
}
dat.newestEp = ini.mediaInfo.newestEp;
dat.seasonList = [];
for (let i = 0; i < ini.ssList.length; i++) {
dat.seasonList[i] = {};
dat.seasonList[i].badge = ini.ssList[i].badge;
dat.seasonList[i].badge_type = ini.ssList[i].badgeType;
dat.seasonList[i].cover = ini.ssList[i].cover;
dat.seasonList[i].media_id = -1;
dat.seasonList[i].new_ep = {
cover : ini.ssList[i].epCover,
id : -1,
index_show : ini.ssList[i].desc
};
dat.seasonList[i].season_id = ini.ssList[i].id;
dat.seasonList[i].season_title = ini.ssList[i].title;
dat.seasonList[i].season_type = ini.ssList[i].type;
dat.seasonList[i].stat = {
danmaku : 0,
follow : 0,
view : 0
};
dat.seasonList[i].title = ini.ssList[i].title;
}
dat.newestEp.isNew = dat.newestEp.isNew ? 1 : 0;
dat.rightsInfo = {};
dat.rightsInfo.allow_bp = ini.mediaInfo.rights.allowBp ? 1 : 0;
dat.rightsInfo.allow_download = 1;
dat.rightsInfo.allow_review = ini.mediaInfo.rights.allowReview ? 1 : 0;
dat.rightsInfo.copyright = "bilibili";
dat.rightsInfo.is_preview = ini.mediaInfo.rights.isPreview ? 1 : 0;
dat.rightsInfo.watch_platform = ini.mediaInfo.rights.appOnly ? 1 : 0;
dat.pubInfo = {};
dat.pubInfo.is_finish = ini.mediaInfo.pub.isFinish ? 1 : 0;
dat.pubInfo.is_started = ini.mediaInfo.pub.isStart ? 1 : 0;
dat.pubInfo.pub_time = ini.mediaInfo.pub.time;
dat.pubInfo.pub_time_show = ini.mediaInfo.pub.timeShow;
dat.pubInfo.weekday = -1;
dat.upInfo = {};
dat.upInfo.avatar = ini.mediaInfo.upInfo.avatar;
dat.upInfo.follower = "--";
dat.upInfo.is_vip = ini.mediaInfo.upInfo.isAnnualVip ? 1 : 0;
dat.upInfo.mid = ini.mediaInfo.upInfo.mid;
dat.upInfo.pendant = {
image : ini.mediaInfo.upInfo.pendantImage,
name : ini.mediaInfo.upInfo.pendantName,
pid : ini.mediaInfo.upInfo.pendantId
};
dat.upInfo.uname = ini.mediaInfo.upInfo.name;
dat.upInfo.verify_type = 6;
if (dat.upInfo.mid < 1) dat.upInfo = {avatar : "//i0.hdslb.com/bfs/face/ef0457addb24141e15dfac6fbf45293ccf1e32ab.jpg", follower : 897603, is_vip : 1, mid : 2, pendant : {image : "", name : "", pid : 0}, uname : "碧诗", verify_type : 2}
}
dat.seasonStat = {"views":0, "danmakus":0, "coins":0, "favorites":0};
dat.userStat = {"loaded":true, "error":false, "follow":0, "pay":0, "payPackPaid":0, "sponsor":0};
dat.userStat.watchProgress = pug.progress;
dat.userStat.vipInfo = pug.vip_info;
if (pug.dialog || pug.pay == 1) {
dat.payMent = {"price":"0.0", "promotion":"", "tip":"大会员专享观看特权哦~"};
if (pug.dialog) {
dat.payMent.vip_promotion = pug.dialog.title;
if (pug.dialog.btn_left) dat.payMent.price = pug.dialog.btn_left.title.match(/[0-9]+/)[0];
}
}
if (dat.epInfo.index >= 0) {dat.special = false; dat.mediaInfo.bkg_cover = "";}
return dat;
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("__INITIAL_STATE__·Bangumi", ...e)}
},
// 主页
home : (data) => {
try {
let dat = {};
let ini = JSON.parse(data);
dat.recommendData = [];
for (let i = 0; i < ini.recommendList.length; i++) {
dat.recommendData[i] = {};
dat.recommendData[i].aid = ini.recommendList[i].aid;
dat.recommendData[i].typename = ini.recommendList[i].tname;
dat.recommendData[i].title = ini.recommendList[i].title;
dat.recommendData[i].subtitle = "";
dat.recommendData[i].play = ini.recommendList[i].stat.view;
dat.recommendData[i].review = ini.recommendList[i].stat.reply;
dat.recommendData[i].video_review = "";
dat.recommendData[i].favorites = ini.recommendList[i].stat.favorite;
dat.recommendData[i].mid = ini.recommendList[i].owner.mid;
dat.recommendData[i].author = ini.recommendList[i].owner.name;
dat.recommendData[i].create = ini.recommendList[i].pubdate;
dat.recommendData[i].pic = ini.recommendList[i].pic;
dat.recommendData[i].coins = ini.recommendList[i].stat.coin;
dat.recommendData[i].duration = ini.recommendList[i].duration;
dat.recommendData[i].badgepay = false;
dat.recommendData[i].rights = ini.recommendList[i].rights;
}
dat.locsData = ini.locsData;
dat.locsData[23] = ini.locsData[3197];
if (config.reset.adloc) for (let key in dat.locsData) if (dat.locsData[key]) for (let i = dat.locsData[key].length - 1; i >= 0; i--) if (dat.locsData[key][i].is_ad) {debug.debug("移除广告", key, dat.locsData[key][i]);dat.locsData[key].splice(i, 1);}
if (dat.locsData[31][0] && dat.locsData[31][0].id == 0) dat.locsData[31] = [{"id":36585,"contract_id":"","pos_num":1,"name":"小黑屋弹幕举报","pic":"https://i0.hdslb.com/bfs/archive/0aa2f32c56cb65b6d453192a3015b65e62537b9a.jpg","litpic":"","url":"https://www.bilibili.com/blackboard/activity-dmjbfj.html","style":0,"agency":"","label":"","intro":"","creative_type":0,"request_id":"1546354354629q172a23a61a62q626","src_id":32,"area":0,"is_ad_loc":true,"ad_cb":"","title":"","server_type":0,"cm_mark":0,"stime":1520478000,"mid":"14629218"}];
return dat;
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("__INITIAL_STATE__·Home", ...e)}
}
}
// xhr hook,包括原生XMLHttpRequest、$.ajax中的jsonp、WebSocket
const intercept = {
init : () => {
const open = XMLHttpRequest.prototype.open;
const send = XMLHttpRequest.prototype.send;
const addEventListener = XMLHttpRequest.prototype.addEventListener;
// 为了防止pakku.js休眠时,send中主动请求分段的部分被重复调用而设置的标志
let segRequestOnlyOnce = true;
if (config.reset.livechat) {
let decoder = new TextDecoder();
let encoder = new TextEncoder();
let liveChatOld; // 对旧播放器建立的ws对象的引用
let liveChat;
// 为了获取ws对象的引用,hook WebSocket.send
let wsHookRunOnce = true;
const wssend = WebSocket.prototype.send;
WebSocket.prototype.send = function (...arg) {
if (wsHookRunOnce && this.url == 'wss://broadcast.chat.bilibili.com:4095/sub') {
liveChatOld = this;
// 切p和掉线之后需要重新启动hook,获得新的引用
let onclose = liveChatOld.onclose;
liveChatOld.onclose = function () {
wsHookRunOnce = true;
clearTimeout(liveChat.heatTimer);
liveChat.close();
onclose.call(this);
}
// 从bilibiliPlayer.js > b.prototype.xx复制过来
// 编码一个数据包
// body[Object] : 要发送的信息
// option[Number] : 数据包对应的行为
// =5 一条弹幕数据
// =7 首个数据包,建立与服务器的连接
// return[Buffer] : 包装好的数据
liveChatOld.convertToArrayBuffer = function (body, option) {
let header = [{"name":"Header Length","key":"headerLen","qg":2,"offset":4,"value":16},{"name":"Protocol Version","key":"ver","qg":2,"offset":6,"value":1},{"name":"Operation","key":"op","qg":4,"offset":8,"value" : option},{"name":"Sequence Id","key":"seq","qg":4,"offset":12,"value":1}];
let headerBuf = new ArrayBuffer(16);
let viewer = new DataView(headerBuf, 0);
let bodyBuf = encoder.encode(JSON.stringify(body));
viewer.setInt32(0, 16 + bodyBuf.byteLength);
header.forEach(function (b) {
4 === b.qg ? viewer.setInt32(b.offset, b.value) : 2 === b.qg && viewer.setInt16(b.offset, b.value)
})
return mergeArrayBuffer(headerBuf, bodyBuf);
}
wsHookRunOnce = false;
initLiveChat();
}
wssend.call(this, ...arg);
}
// 原函数位于bilibiliPlayer.js > c.a.eK 和 jsc-player > Dl.mergeArrayBuffer
// 连接两个buffer
function mergeArrayBuffer(headerBuf, bodyBuf) {
headerBuf = new Uint8Array(headerBuf);
bodyBuf = new Uint8Array(bodyBuf);
var d = new Uint8Array(headerBuf.byteLength + bodyBuf.byteLength);
d.set(headerBuf, 0);
d.set(bodyBuf, headerBuf.byteLength);
return d.buffer;
}
function initLiveChat() {
// 数据包对应的Operation常量表
let Pl = {"WS_OP_HEARTBEAT":2,"WS_OP_HEARTBEAT_REPLY":3,"WS_OP_DATA":1000,"WS_OP_BATCH_DATA":9,"WS_OP_DISCONNECT_REPLY":6,"WS_OP_USER_AUTHENTICATION":7,"WS_OP_CONNECT_SUCCESS":8,"WS_OP_CHANGEROOM":12,"WS_OP_CHANGEROOM_REPLY":13,"WS_OP_REGISTER":14,"WS_OP_REGISTER_REPLY":15,"WS_OP_UNREGISTER":16,"WS_OP_UNREGISTER_REPLY":17,"WS_OP_OGVCMD_REPLY":1015,"WS_PACKAGE_HEADER_TOTAL_LENGTH":18,"WS_PACKAGE_OFFSET":0,"WS_HEADER_OFFSET":4,"WS_VERSION_OFFSET":6,"WS_OPERATION_OFFSET":8,"WS_SEQUENCE_OFFSET":12,"WS_COMPRESS_OFFSET":16,"WS_CONTENTTYPE_OFFSET":17,"WS_BODY_PROTOCOL_VERSION":1,"WS_HEADER_DEFAULT_VERSION":1,"WS_HEADER_DEFAULT_OPERATION":1,"ws_header_default_sequence":1,"WS_HEADER_DEFAULT_COMPRESS":0,"WS_HEADER_DEFAULT_CONTENTTYPE":0};
// 请求头的参数表
let wsBinaryHeaderList = [{"name":"Header Length","key":"headerLen","bytes":2,"offset":4,"value":18},{"name":"Protocol Version","key":"ver","bytes":2,"offset":6,"value":1},{"name":"Operation","key":"op","bytes":4,"offset":8,"value":7},{"name":"Sequence Id","key":"seq","bytes":4,"offset":12,"value":2},{"name":"Compress","key":"compress","bytes":1,"offset":16,"value":0},{"name":"ContentType","key":"contentType","bytes":1,"offset":17,"value":0}]
liveChat = new WebSocket('wss://broadcast.chat.bilibili.com:7823/sub');
liveChat.binaryType = "arraybuffer";
liveChat.heatTimer = -1;
// 每30秒一个心跳包
liveChat.heartBeat = function () {
var i = this;
clearTimeout(this.heatTimer);
var e = this.convertToArrayBuffer({}, Pl.WS_OP_HEARTBEAT);
this.send(e);
this.heatTimer = window.setTimeout((function () {
i.heartBeat();
}), 1e3 * 30);
}
liveChat.onopen = function () {
let body = {
"room_id": "video://" + aid + "/" + cid,
"platform": "web",
"accepts": [1000, 1015]
};
return this.send(this.convertToArrayBuffer(body, 7));
}
liveChat.onmessage = function (i) {
try {
var t = this.convertToObject(i.data);
if (t) {
switch (t.op) {
case Pl.WS_OP_HEARTBEAT_REPLY:
// 对于心跳包,服务器响应当前在线人数的数据
// 旧播放器连接的4095端口,虽然不再下发实时弹幕,但依然照常响应在线人数
// 所以暂时不用替换成新版
// this.onHeartBeatReply(t.body);
break;
case Pl.WS_OP_CONNECT_SUCCESS:
this.heartBeat();
break;
// 旧播放器只能处理(连接成功,心跳响应,实时弹幕)三种响应信息
// 新播放器新增的指令和功能就不管了
case Pl.WS_OP_CHANGEROOM_REPLY:
//0 === Number(t.body.code) && this.options.onChangeRoomReply({ data : t && t.body });
break;
case Pl.WS_OP_REGISTER_REPLY:
//0 === Number(t.body.code) && this.options.onRegisterReply({ data : t && t.body });
break;
case Pl.WS_OP_UNREGISTER_REPLY:
//0 === Number(t.body.code) && this.options.onUnRegisterReply({ data : t && t.body });
break;
case Pl.WS_OP_DATA:
case Pl.WS_OP_BATCH_DATA:
t.body.forEach(function (v) {
// 记录实时弹幕哈希值
hash.push(v[0].split(",")[7]);
liveChatOld.onmessage({
data: liveChatOld.convertToArrayBuffer({
cmd: 'DM',
info: [v[0], v[1]]
}, 5)
});
});
break;
case Pl.WS_OP_OGVCMD_REPLY:
//this.onOgvCmdReply(t);
break;
default:
//this.msgReply(t)
}
}
} catch (i) {
console.error("WebSocket Error : ", i)
}
return this;
}
// jsc-player > i.prototype.convertToArrayBuffer,新版播放器的请求头信息更多,需要18字节
// 基本与liveChatOld.convertToArrayBuffer相同
liveChat.convertToArrayBuffer = function (body, option) {
let headerBuf = new ArrayBuffer(Pl.WS_PACKAGE_HEADER_TOTAL_LENGTH);
let viewer = new DataView(headerBuf, Pl.WS_PACKAGE_OFFSET);
let bodyBuf = encoder.encode(JSON.stringify(body));
viewer.setInt32(Pl.WS_PACKAGE_OFFSET, Pl.WS_PACKAGE_HEADER_TOTAL_LENGTH + bodyBuf.byteLength);
wsBinaryHeaderList[2].value = option;
wsBinaryHeaderList.forEach((function (i) {
4 === i.bytes ? (viewer.setInt32(i.offset, i.value),
"seq" === i.key && ++i.value) : 2 === i.bytes ? viewer.setInt16(i.offset, i.value) : 1 === i.bytes && viewer.setInt8(i.offset, i.value)
}));
return mergeArrayBuffer(headerBuf, bodyBuf);
}
// jsc-player > i.prototype.convertToObject
// convertToArrayBuffer对应的解码函数
liveChat.convertToObject = function (i) {
var e = new DataView(i), t = {};
t.packetLen = e.getInt32(Pl.WS_PACKAGE_OFFSET);
wsBinaryHeaderList.forEach((function (i) {
4 === i.bytes ? t[i.key] = e.getInt32(i.offset) : 2 === i.bytes ? t[i.key] = e.getInt16(i.offset) : 1 === i.bytes && (t[i.key] = e.getInt8(i.offset))
}));
if (t.op && t.op === Pl.WS_OP_BATCH_DATA) {
t.body = this.parseDanmaku(i, e, Pl.WS_PACKAGE_HEADER_TOTAL_LENGTH, t.packetLen);
}
else if (t.op && Pl.WS_OP_DATA === t.op) {
t.body = this.parseDanmaku(i, e, Pl.WS_PACKAGE_OFFSET, t.packetLen);
}
else if (t.op && t.op === Pl.WS_OP_OGVCMD_REPLY) {
t.body = ""; // this.parseOgvCmd(i, e, Pl.WS_PACKAGE_OFFSET, t.packetLen);
}
else if (t.op) {
t.body = [];
for (var a = Pl.WS_PACKAGE_OFFSET, r = t.packetLen, n = "", l = ""; a < i.byteLength; a += r) {
r = e.getInt32(a);
n = e.getInt16(a + Pl.WS_HEADER_OFFSET);
try {
l = JSON.parse(decoder.decode(i.slice(a + n, a + r)));
t.body = l;
} catch (e) {
l = decoder.decode(i.slice(a + n, a + r));
console.error("decode body error:", new Uint8Array(i), t);
}
}
}
return t;
}
// jsc-player > i.prototype.parseDanmaku
liveChat.parseDanmaku = function (i, e, t, a) {
for (var r, n = [], l = t; l < i.byteLength; l += a) {
a = e.getInt32(l);
r = e.getInt16(l + Pl.WS_HEADER_OFFSET);
try {
n.push(JSON.parse(decoder.decode(i.slice(l + r, l + a))));
} catch (e) {
n.push(decoder.decode(i.slice(l + r, l + a)));
console.error("decode body error:", new Uint8Array(i));
}
}
return n;
}
}
}
//Worker Hook
if (config.reset.danmuku && Worker) {
//hook postMessage来得到旧播放器用来 获取list.so 的worker对象
let workerPostMsg = Worker.prototype.postMessage;
let list_so;
let loadTime, parseTime; //旧播放器需要得到相关耗时数据,这里手动算
Worker.prototype.postMessage = function (aMessage, transferList) {
if (aMessage.url && aMessage.url.includes("list.so")) {
list_so = this;
loadTime = new Date();
let Segments = [];
deliver.getSegDanmaku(function (protoSegments) {
loadTime = new Date() - loadTime;
parseTime = new Date();
protoSegments.forEach(function (seg) {
Segments = Segments.concat(protoSeg.decode(new Uint8Array(seg)).elems);
});
Segments.sort(function (a, b) {
return a.progress - b.progress;
});
//转换到xml只是为了满足下载功能
if(config.reset.download) {
deliver.toXml(Segments, aid).then(function (result) {
// 备份弹幕
xml = result;
});
}
//将弹幕转换为旧格式
Segments = Segments.map(function (v) {
// 记录弹幕池哈希值
hash.push(v.midHash);
return {
class: 0,
color: v.color,
date: v.ctime,
dmid: v.id,
mode: v.mode,
size: v.fontsize,
stime: v.progress / 1000,
text: v.content,
uid: v.midHash
};
});
parseTime = new Date() - parseTime;
list_so.onmessage({
data: {
code: 0,
danmakuArray: Segments,
loadTime: loadTime,
parseTime: parseTime,
sendTip: "",
state: 0,
textSide: "",
total: Segments.length.toString()
}
});
});
} else {
workerPostMsg.call(this, aMessage, transferList);
}
}
}
// 原生xhr hook
XMLHttpRequest.prototype.open = function (method, url, ...rest) {
let _url = url, hook = [_url, ""];
let obj = deliver.search2obj(url);
// 拦截跟踪日志
if (url.includes('data.bilibili.com/log/web')) return open.call(this, "", null, ...rest);
// 替换视频心跳
if (url.includes('api.bilibili.com/x/report/web/heartbeat') && config.reset.heartbeat) {
url = url.replace('api.bilibili.com/x/report/web/heartbeat', 'api.bilibili.com/x/click-interface/web/heartbeat');
debug.debug("XHR重定向", "替换视频心跳", [_url, url]);
}
// 显示历史视频
if (url.includes('api.bilibili.com/x/web-interface/history/cursor') && url.includes("business") && config.reset.history) {
let max = obj.max || "", view_at = obj.view_at || "";
url = deliver.obj2search("//api.bilibili.com/x/web-interface/history/cursor", {max : max, view_at : view_at, type : "archive", ps : 20});
debug.debug("XHR重定向", "显示历史视频", [_url, url]);
}
// 修改正在直播
if (url.includes('api.live.bilibili.com/room/v1/RoomRecommend/biliIndexRecList')) {
this.addEventListener('readystatechange', () => {if ( this.readyState === 4 ) intercept.biliIndexRec(this, hook)});
url = hook[1] = url.replace('api.live.bilibili.com/room/v1/RoomRecommend/biliIndexRecList', 'api.live.bilibili.com/xlive/web-interface/v1/webMain/getList?platform=web');
}
// 修改直播动态
if (url.includes('api.live.bilibili.com/room/v1/RoomRecommend/biliIndexRecMore')) {
this.addEventListener('readystatechange', () => {if ( this.readyState === 4 ) intercept.biliIndexRec(this, hook)});
url = hook[1] = url.replace('api.live.bilibili.com/room/v1/RoomRecommend/biliIndexRecMore', 'api.live.bilibili.com/xlive/web-interface/v1/webMain/getMoreRecList?platform=web');
}
// 重定向番剧信息
if (url.includes('bangumi.bilibili.com/view/web_api/season?')) {
this.addEventListener('readystatechange', () => {if ( this.readyState === 4 ) intercept.season(this, hook)});
url = hook[1] = url.replace('bangumi.bilibili.com/view/web_api/season', 'api.bilibili.com/pgc/view/web/season');
}
// 重定向追番信息
if (url.includes('bangumi.bilibili.com/ext/web_api/season_count?')) {
this.addEventListener('readystatechange', () => {if ( this.readyState === 4 ) intercept.stat(this, hook)});
url = hook[1] = url.replace('bangumi.bilibili.com/ext/web_api/season_count', 'api.bilibili.com/pgc/web/season/stat');
}
// 修改番剧推荐
if (url.includes('api.bilibili.com/pgc/web/recommend/related/recommend')) {
this.addEventListener('readystatechange', () => {if ( this.readyState === 4 ) intercept.recommend(this)});
}
// 修改直播数据
if (url.includes('api.live.bilibili.com/xlive/web-room/v2/index/getRoomPlayInfo')) {
this.addEventListener('readystatechange', () => {if ( this.readyState === 4 ) intercept.getRoomPlayInfo(this)});
}
// 修改播放器通知
if (url.includes('api.bilibili.com/x/player/carousel')) {
this.addEventListener('readystatechange', () => {if ( this.readyState === 4 ) intercept.carousel(this)});
}
// 修改区域限制
if (url.includes('season/user/status?')) {
this.addEventListener('readystatechange', () => {if ( this.readyState === 4 ) intercept.status(this)});
}
// 监听视频链接
if (url.includes("/playurl?")) {
if (!url.includes("fourk") && !url.includes("sign")) {
url = url.replace("playurl?", "playurl?fourk=1&");
debug.debug("XHR重定向", "添加4K参数", [_url, url]);
}
cid = obj.cid || cid;
aid = obj.avid || aid;
bvid = obj.bvid || deliver.convertId(aid) || bvid;
pgc = url.includes("pgc") ? true : false;;
if (limit) this.url = url;
this.addEventListener('readystatechange', () => {if ( this.readyState === 4 ) intercept.playinfo(this, url)});
}
// 修改弹幕链接
if (url.includes("list.so") && config.reset.danmuku) {
// 这时pakku.js已经修改了xhr对象,需要另做处理
if (this.pakku_url) {
segRequestOnlyOnce = true;
let pid = aid;
// 更改pakku.js请求的url,使它过滤分段弹幕
this.pakku_url = url = "https://api.bilibili.com/x/v2/dm/web/seg.so?type=1&oid=" + cid + "&pid=" + pid + "&segment_index=1";
this.responseType = "arraybuffer";
let xhr = this;
let cb = [];
for (let i in this.pakku_load_callback) {
cb[i] = this.pakku_load_callback[i];
}
for (let i in this.pakku_load_callback) {
// 将pakku.js返回的数据转换回xml
this.pakku_load_callback[i] = function () {
deliver.toXml(protoSeg.decode(new Uint8Array(xhr.response)).elems, pid).then(function (xml) {
xhr.response = xhr.responseText = xml;
cb[i].call(xhr);
});
}
}
} else {
this.reqURL = url;
}
}
return open.call(this, method, url, ...rest);
}
// 部分功能依赖hook `XMLHttpRequest.prototype.send`
if (config.reset.xhrhook) {
XMLHttpRequest.prototype.send = async function (...arg) {
// pakku.js休眠中,钩子捕捉到首次对seg.so发起请求时触发
// (pakku.js正常运行时这个send()不会被调用)
if (config.reset.danmuku && (this.pakku_url && this.pakku_url.includes("seg.so") && segRequestOnlyOnce)) {
segRequestOnlyOnce = false;
// pakku.js会禁用Worker,这时需要模拟一个xhr响应
Object.defineProperty(this, "response", { writable : true });
Object.defineProperty(this, "responseText", { writable : true });
Object.defineProperty(this, "readyState", { writable : true });
Object.defineProperty(this, "status", { writable : true });
this.readyState = 4;
this.status = 200;
this.abort();
let callBack = this.callBack;
let xhr = this;
deliver.getSegDanmaku(function (protoSegments) {
let Segments = [];
protoSegments.forEach(function (seg) {
Segments = Segments.concat(protoSeg.decode(new Uint8Array(seg)).elems);
});
deliver.toXml(Segments, aid).then(function (toXml) {
callBack.forEach(function (f) {
xhr.response = xhr.responseText = toXml;
f.call(xhr);
});
// 备份弹幕
xml = xhr.response;
});
});
}
else if (this.url) {
try {
// 解除限制
Object.defineProperty(this, "response", { writable : true });
Object.defineProperty(this, "responseText", { writable : true });
Object.defineProperty(this, "readyState", { writable : true });
Object.defineProperty(this, "status", { writable : true });
this.readyState = 2;
this.status = 200;
this.abort();
let response, accesskey = "";
try {
if (limit) {
// 区域限制 + APP限制的DASH似乎缺少码率信息,现默认启用flv以规避,platform用于伪装成APP
if (uid && (ids.indexOf(1 * cid) >= 0) && config.reset.accesskey) accesskey = GM_getValue("access_key") || "";
let obj = Object.assign(deliver.search2obj(this.url), __INITIAL_STATE__.rightsInfo.watch_platform ? {access_key : accesskey, balh_ajax : 1, fnval : "", fnver : "", module : "pgc", platform : "android_i"} : {access_key : accesskey, balh_ajax : 1, module : "pgc"})
response = deliver.xhrJsonCheck(await xhr.true(deliver.obj2search(API.url.BPplayurl, obj)));
response = {"code" : 0, "message" : "success" , "result" : response};
}
}
catch (e) {debug.msg("解除限制失败 ಥ_ಥ", e); response = {"code" : -404, "message" : e , "data" : null};}
this.response = this.responseText = JSON.stringify(response);
this.readyState = 4;
this.onreadystatechange();
if (response.code !== 0) throw response.message;
__playinfo__ = response;
debug.log("解除限制", "aid=", aid, "cid=", cid);
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("解除限制", ...e)}
}
else {
send.call(this, ...arg);
}
}
XMLHttpRequest.prototype.addEventListener = function (name, callback) {
if (name == "load") {
this.callBack = this.callBack || [];
this.callBack.push(callback);
}
return addEventListener.call(this, name, callback);
}
}
// jsonp hook
function jsonp(){
const ajax = unsafeWindow.$.ajax;
unsafeWindow.$.ajax = function (obj, ...rest) {
if (obj) {
if (obj.dataType == "jsonp") {
let _obj = JSON.parse(JSON.stringify(obj));
if (obj.url.includes("region") && obj.data.rid == 165) {
// 替换广告区rid为资讯区rid
obj.data.rid = 202;
debug.debug("JSONP重定向", "替换广告区", [_obj, obj]);
}
if (obj.url.includes("region") && obj.data.original == 1) {
// 替换原创排行为全部排行
obj.data.original = 0;
debug.debug("JSONP重定向", "修复原创排行", [_obj, obj]);
}
if (obj.url.includes('api.bilibili.com/x/web-interface/ranking/index')) {
// 修改置顶推荐
obj.url = obj.url.replace('ranking/index', 'index/top');
debug.debug("JSONP重定向", "修复置顶推荐", [_obj, obj]);
}
}
}
return ajax.call(this, obj, ...rest);
}
}
// jsonp非原生调用,先判断jQuery是否载入,以免报错
if (unsafeWindow.$ && unsafeWindow.$.ajax) jsonp();
else {
let timer = setInterval(() => {
// 为不错过任何jsonp,轮循间隔设得有点小
if (unsafeWindow.$) {
clearInterval(timer);
jsonp();
}
},10);
setTimeout(() => clearInterval(timer), 5000);
}
},
// 修改首页直播推荐数据
biliIndexRec : (obj, hook = []) => {
try {
hook.push(deliver.xhrJsonCheck(obj.responseText));
let response = obj.responseText.replace(/preview_banner_list/, "preview").replace(/ranking_list/, "ranking").replace(/recommend_room_list/, "recommend");
response = JSON.parse(response);
response.data.text_link = {text : "233秒居然能做这些!", link : "//vc.bilibili.com"};
if (response.data.recommend) {
for (let i = 0; i < response.data.recommend.length; i++) {
response.data.recommend[i].pic = response.data.recommend[i].cover;
response.data.recommend[i].link = "//live.bilibili.com" + response.data.recommend[i].link;
}
}
if (response.data.preview) for (let i = 0; i < response.data.preview.length; i++) response.data.preview[i].url = response.data.preview[i].link;
hook.push(response);
debug.debug("XHR重定向", "修复正在直播", hook);
Object.defineProperty(obj, 'response', {writable : true});
Object.defineProperty(obj, 'responseText', {writable : true});
obj.response = obj.responseText = JSON.stringify(response);
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("首页推荐", ...e)}
},
// 修复番剧季度信息
season : (obj, hook = []) => {
try {
hook.push(deliver.xhrJsonCheck(obj.responseText));
let response = deliver.xhrJsonCheck(obj.responseText);
for (let i = 0; i < response.result.episodes.length; i++){
response.result.episodes[i].ep_id = response.result.episodes[i].id;
response.result.episodes[i].episode_status = response.result.episodes[i].status;
response.result.episodes[i].index = response.result.episodes[i].title;
response.result.episodes[i].index_title = response.result.episodes[i].long_title;
}
hook.push(response);
debug.debug("XHR重定向", "番剧季度信息", hook);
Object.defineProperty(obj, 'response', {writable : true});
Object.defineProperty(obj, 'responseText', {writable : true});
obj.response = obj.responseText = JSON.stringify(response);
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("番剧季度信息", ...e)}
},
// 修复番剧追番信息
stat : (obj, hook = []) => {
try {
hook.push(deliver.xhrJsonCheck(obj.responseText));
let response = deliver.xhrJsonCheck(obj.responseText);
response.result.favorites = response.result.follow;
hook.push(response);
debug.debug("XHR重定向", "番剧追番信息", hook);
Object.defineProperty(obj, 'response', {writable : true});
Object.defineProperty(obj, 'responseText', {writable : true});
obj.response = obj.responseText = JSON.stringify(response);
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("番剧季度信息", ...e)}
},
// 修改直播数据
getRoomPlayInfo : (obj, hook = []) => {
if (!config.reset.roomplay) return;
try {
hook.push(deliver.xhrJsonCheck(obj.responseText));
let response = deliver.xhrJsonCheck(obj.responseText);
if (response.data) {
response.data.live_status = 0;
response.data.live_time = -1;
response.data.playurl_info = null;
}
hook.push(response);
debug.debug("XHR重定向", "拦截直播媒体", hook);
Object.defineProperty(obj, 'response', {writable : true});
Object.defineProperty(obj, 'responseText', {writable : true});
obj.response = obj.responseText = JSON.stringify(response);
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("直播拦截", ...e)}
},
// 修改番剧推荐
recommend : (obj, hook = []) => {
try {
hook.push(deliver.xhrJsonCheck(obj.responseText));
let response = deliver.xhrJsonCheck(obj.responseText);
if (response.result && response.result.season) response.result = response.result.season;
hook.push(response);
debug.debug("XHR重定向", "修改番剧推荐", hook);
Object.defineProperty(obj, 'response', {writable : true});
Object.defineProperty(obj, 'responseText', {writable : true});
obj.response = obj.responseText = JSON.stringify(response);
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("番剧推荐", ...e)}
},
// 生成播放信息
carousel : (obj) => {
if (!config.reset.carousel) return;
try {
let msg = deliver.randomArray(API.message, 2);
let xmltext = '<msg><item tooltip="" bgcolor="#000000" catalog="system" resourceid="2319" srcid="2320" id="314825"><![CDATA[<a href="' + msg[0][0] + '" target="_blank"><font color="#FFFFFF">' + msg[0][1] + '</font></a>]]></item><item tooltip="" bgcolor="#000000" catalog="system" resourceid="2319" srcid="2321" id="314372"><![CDATA[<a href="' + msg[1][0] + '" target="_blank"><font color="#FFFFFF">' + msg[1][1] + '</font></a>]]></item></msg>';
let parser = new DOMParser(),
responseXML = parser.parseFromString(xmltext, "text/xml");
Object.defineProperty(obj, 'responseXML', {writable : true});
obj.responseXML = responseXML;
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("播放通知", ...e)}
},
// 强制载入播放器
status : (obj) => {
try {
let response = deliver.xhrJsonCheck(obj.responseText);
if (response.result) {
if (config.reset.limit && response.result.area_limit){
response.result.area_limit = 0;
response.ban_area_show = 1;
limit = true;
}
if (response.result.pay) big = 0;
if (!response.result.pay && config.big && response.result.dialog) {
response.result.pay = 1;
big = true;
}
if (limit || big) {
Object.defineProperty(obj, 'response', {writable : true});
Object.defineProperty(obj, 'responseText', {writable : true});
obj.response = obj.responseText = JSON.stringify(response);
}
}
}
catch (e) {e = Array.isArray(e) ? e : [e]; debug.error("强制启用播放器", ...e)}
},
// 监听视频地址
playinfo : (obj) => {
try {
if (!obj.response) throw obj;
__playinfo__ = typeof obj.response == "object" ? obj.response : deliver.xhrJsonCheck(obj.response, true);
// 刷新下载面板
if (document.getElementById("bili-old-download-table")) deliver.download.setTable();
}
catch (e) {e = Array.isArray(e) ? e : [e]; debug.error("视频监听", ...e)}
}
}
// MD5接口,修改自百度百科https://baike.baidu.com/item/MD5/212708
const md5 = (str) => {
let md5_RotateLeft = (lValue, iShiftBits) => (lValue << iShiftBits) | (lValue >>> (32 - iShiftBits));
let md5_AddUnsigned = (lX, lY) => {
let lX4, lY4, lX8, lY8, lResult;
lX8 = (lX & 0x80000000);
lY8 = (lY & 0x80000000);
lX4 = (lX & 0x40000000);
lY4 = (lY & 0x40000000);
lResult = (lX & 0x3FFFFFFF) + (lY & 0x3FFFFFFF);
if (lX4 & lY4) return (lResult ^ 0x80000000 ^ lX8 ^ lY8);
if (lX4 | lY4) {
if (lResult & 0x40000000) return (lResult ^ 0xC0000000 ^ lX8 ^ lY8);
else return (lResult ^ 0x40000000 ^ lX8 ^ lY8);
}
else return (lResult ^ lX8 ^ lY8);
}
let md5_F = (x, y, z) => (x & y) | ((~x) & z);
let md5_G = (x, y, z) => (x & z) | (y & (~z));
let md5_H = (x, y, z) => (x ^ y ^ z);
let md5_I = (x, y, z) => (y ^ (x | (~z)));
let md5_FF = (a, b, c, d, x, s, ac) => {
a = md5_AddUnsigned(a, md5_AddUnsigned(md5_AddUnsigned(md5_F(b, c, d), x), ac));
return md5_AddUnsigned(md5_RotateLeft(a, s), b);
};
let md5_GG = (a, b, c, d, x, s, ac) => {
a = md5_AddUnsigned(a, md5_AddUnsigned(md5_AddUnsigned(md5_G(b, c, d), x), ac));
return md5_AddUnsigned(md5_RotateLeft(a, s), b);
};
let md5_HH = (a, b, c, d, x, s, ac) => {
a = md5_AddUnsigned(a, md5_AddUnsigned(md5_AddUnsigned(md5_H(b, c, d), x), ac));
return md5_AddUnsigned(md5_RotateLeft(a, s), b);
};
let md5_II = (a, b, c, d, x, s, ac) => {
a = md5_AddUnsigned(a, md5_AddUnsigned(md5_AddUnsigned(md5_I(b, c, d), x), ac));
return md5_AddUnsigned(md5_RotateLeft(a, s), b);
};
let md5_ConvertToWordArray = (string) => {
let lWordCount;
let lMessageLength = string.length;
let lNumberOfWords_temp1 = lMessageLength + 8;
let lNumberOfWords_temp2 = (lNumberOfWords_temp1 - (lNumberOfWords_temp1 % 64)) / 64;
let lNumberOfWords = (lNumberOfWords_temp2 + 1) * 16;
let lWordArray = Array(lNumberOfWords - 1);
let lBytePosition = 0;
let lByteCount = 0;
while (lByteCount < lMessageLength) {
lWordCount = (lByteCount - (lByteCount % 4)) / 4;
lBytePosition = (lByteCount % 4) * 8;
lWordArray[lWordCount] = (lWordArray[lWordCount] | (string.charCodeAt(lByteCount) << lBytePosition));
lByteCount++;
}
lWordCount = (lByteCount - (lByteCount % 4)) / 4;
lBytePosition = (lByteCount % 4) * 8;
lWordArray[lWordCount] = lWordArray[lWordCount] | (0x80 << lBytePosition);
lWordArray[lNumberOfWords - 2] = lMessageLength << 3;
lWordArray[lNumberOfWords - 1] = lMessageLength >>> 29;
return lWordArray;
};
let md5_WordToHex = (lValue) => {
let WordToHexValue = "",
WordToHexValue_temp = "",
lByte, lCount;
for (lCount = 0; lCount <= 3; lCount++) {
lByte = (lValue >>> (lCount * 8)) & 255;
WordToHexValue_temp = "0" + lByte.toString(16);
WordToHexValue = WordToHexValue + WordToHexValue_temp.substr(WordToHexValue_temp.length - 2, 2);
}
return WordToHexValue;
};
let md5_Utf8Encode = (string) => {
string = string.replace(/\r\n/g, "\n");
let utftext = "";
for (let n = 0; n < string.length; n++) {
let c = string.charCodeAt(n);
if (c < 128) utftext += String.fromCharCode(c);
else if ((c > 127) && (c < 2048)) {
utftext += String.fromCharCode((c >> 6) | 192);
utftext += String.fromCharCode((c & 63) | 128);
}
else {
utftext += String.fromCharCode((c >> 12) | 224);
utftext += String.fromCharCode(((c >> 6) & 63) | 128);
utftext += String.fromCharCode((c & 63) | 128);
}
}
return utftext;
};
let x = Array();
let k, AA, BB, CC, DD, a, b, c, d;
let S11 = 7, S12 = 12, S13 = 17, S14 = 22;
let S21 = 5, S22 = 9, S23 = 14, S24 = 20;
let S31 = 4, S32 = 11, S33 = 16, S34 = 23;
let S41 = 6, S42 = 10, S43 = 15, S44 = 21;
str = md5_Utf8Encode(str);
x = md5_ConvertToWordArray(str);
a = 0x67452301;
b = 0xEFCDAB89;
c = 0x98BADCFE;
d = 0x10325476;
for (k = 0; k < x.length; k += 16) {
AA = a;
BB = b;
CC = c;
DD = d;
a = md5_FF(a, b, c, d, x[k + 0], S11, 0xD76AA478);
d = md5_FF(d, a, b, c, x[k + 1], S12, 0xE8C7B756);
c = md5_FF(c, d, a, b, x[k + 2], S13, 0x242070DB);
b = md5_FF(b, c, d, a, x[k + 3], S14, 0xC1BDCEEE);
a = md5_FF(a, b, c, d, x[k + 4], S11, 0xF57C0FAF);
d = md5_FF(d, a, b, c, x[k + 5], S12, 0x4787C62A);
c = md5_FF(c, d, a, b, x[k + 6], S13, 0xA8304613);
b = md5_FF(b, c, d, a, x[k + 7], S14, 0xFD469501);
a = md5_FF(a, b, c, d, x[k + 8], S11, 0x698098D8);
d = md5_FF(d, a, b, c, x[k + 9], S12, 0x8B44F7AF);
c = md5_FF(c, d, a, b, x[k + 10], S13, 0xFFFF5BB1);
b = md5_FF(b, c, d, a, x[k + 11], S14, 0x895CD7BE);
a = md5_FF(a, b, c, d, x[k + 12], S11, 0x6B901122);
d = md5_FF(d, a, b, c, x[k + 13], S12, 0xFD987193);
c = md5_FF(c, d, a, b, x[k + 14], S13, 0xA679438E);
b = md5_FF(b, c, d, a, x[k + 15], S14, 0x49B40821);
a = md5_GG(a, b, c, d, x[k + 1], S21, 0xF61E2562);
d = md5_GG(d, a, b, c, x[k + 6], S22, 0xC040B340);
c = md5_GG(c, d, a, b, x[k + 11], S23, 0x265E5A51);
b = md5_GG(b, c, d, a, x[k + 0], S24, 0xE9B6C7AA);
a = md5_GG(a, b, c, d, x[k + 5], S21, 0xD62F105D);
d = md5_GG(d, a, b, c, x[k + 10], S22, 0x2441453);
c = md5_GG(c, d, a, b, x[k + 15], S23, 0xD8A1E681);
b = md5_GG(b, c, d, a, x[k + 4], S24, 0xE7D3FBC8);
a = md5_GG(a, b, c, d, x[k + 9], S21, 0x21E1CDE6);
d = md5_GG(d, a, b, c, x[k + 14], S22, 0xC33707D6);
c = md5_GG(c, d, a, b, x[k + 3], S23, 0xF4D50D87);
b = md5_GG(b, c, d, a, x[k + 8], S24, 0x455A14ED);
a = md5_GG(a, b, c, d, x[k + 13], S21, 0xA9E3E905);
d = md5_GG(d, a, b, c, x[k + 2], S22, 0xFCEFA3F8);
c = md5_GG(c, d, a, b, x[k + 7], S23, 0x676F02D9);
b = md5_GG(b, c, d, a, x[k + 12], S24, 0x8D2A4C8A);
a = md5_HH(a, b, c, d, x[k + 5], S31, 0xFFFA3942);
d = md5_HH(d, a, b, c, x[k + 8], S32, 0x8771F681);
c = md5_HH(c, d, a, b, x[k + 11], S33, 0x6D9D6122);
b = md5_HH(b, c, d, a, x[k + 14], S34, 0xFDE5380C);
a = md5_HH(a, b, c, d, x[k + 1], S31, 0xA4BEEA44);
d = md5_HH(d, a, b, c, x[k + 4], S32, 0x4BDECFA9);
c = md5_HH(c, d, a, b, x[k + 7], S33, 0xF6BB4B60);
b = md5_HH(b, c, d, a, x[k + 10], S34, 0xBEBFBC70);
a = md5_HH(a, b, c, d, x[k + 13], S31, 0x289B7EC6);
d = md5_HH(d, a, b, c, x[k + 0], S32, 0xEAA127FA);
c = md5_HH(c, d, a, b, x[k + 3], S33, 0xD4EF3085);
b = md5_HH(b, c, d, a, x[k + 6], S34, 0x4881D05);
a = md5_HH(a, b, c, d, x[k + 9], S31, 0xD9D4D039);
d = md5_HH(d, a, b, c, x[k + 12], S32, 0xE6DB99E5);
c = md5_HH(c, d, a, b, x[k + 15], S33, 0x1FA27CF8);
b = md5_HH(b, c, d, a, x[k + 2], S34, 0xC4AC5665);
a = md5_II(a, b, c, d, x[k + 0], S41, 0xF4292244);
d = md5_II(d, a, b, c, x[k + 7], S42, 0x432AFF97);
c = md5_II(c, d, a, b, x[k + 14], S43, 0xAB9423A7);
b = md5_II(b, c, d, a, x[k + 5], S44, 0xFC93A039);
a = md5_II(a, b, c, d, x[k + 12], S41, 0x655B59C3);
d = md5_II(d, a, b, c, x[k + 3], S42, 0x8F0CCC92);
c = md5_II(c, d, a, b, x[k + 10], S43, 0xFFEFF47D);
b = md5_II(b, c, d, a, x[k + 1], S44, 0x85845DD1);
a = md5_II(a, b, c, d, x[k + 8], S41, 0x6FA87E4F);
d = md5_II(d, a, b, c, x[k + 15], S42, 0xFE2CE6E0);
c = md5_II(c, d, a, b, x[k + 6], S43, 0xA3014314);
b = md5_II(b, c, d, a, x[k + 13], S44, 0x4E0811A1);
a = md5_II(a, b, c, d, x[k + 4], S41, 0xF7537E82);
d = md5_II(d, a, b, c, x[k + 11], S42, 0xBD3AF235);
c = md5_II(c, d, a, b, x[k + 2], S43, 0x2AD7D2BB);
b = md5_II(b, c, d, a, x[k + 9], S44, 0xEB86D391);
a = md5_AddUnsigned(a, AA);
b = md5_AddUnsigned(b, BB);
c = md5_AddUnsigned(c, CC);
d = md5_AddUnsigned(d, DD);
}
return (md5_WordToHex(a) + md5_WordToHex(b) + md5_WordToHex(c) + md5_WordToHex(d)).toLowerCase();
}
// crc哈希转mid算法,来源https://github.com/esterTion/BiliBili_crc2mid
const BiliBili_midcrc = function () {
const CRCPOLYNOMIAL = 0xEDB88320;
let startTime = performance.now(),
crctable = new Array(256),
create_table = function () {
let crcreg,
i,
j;
for (i = 0; i < 256; ++i) {
crcreg = i;
for (j = 0; j < 8; ++j) {
if ((crcreg & 1) !== 0) {
crcreg = CRCPOLYNOMIAL ^ (crcreg >>> 1);
} else {
crcreg >>>= 1;
}
}
crctable[i] = crcreg;
}
},
crc32 = function (input) {
if (typeof(input) != 'string') input = input.toString();
let crcstart = 0xFFFFFFFF,
len = input.length,
index;
for (let i = 0; i < len; ++i) {
index = (crcstart ^ input.charCodeAt(i)) & 0xff;
crcstart = (crcstart >>> 8) ^ crctable[index];
}
return crcstart;
},
crc32lastindex = function (input) {
if (typeof(input) != 'string') input = input.toString();
let crcstart = 0xFFFFFFFF,
len = input.length,
index;
for (let i = 0; i < len; ++i) {
index = (crcstart ^ input.charCodeAt(i)) & 0xff;
crcstart = (crcstart >>> 8) ^ crctable[index];
}
return index;
},
getcrcindex = function (t) {
for (let i = 0; i < 256; i++) if (crctable[i] >>> 24 == t) return i;
return -1;
},
deepCheck = function (i, index) {
let tc = 0x00,
str = '',
hash = crc32(i);
tc = hash & 0xff ^ index[2];
if (!(tc <= 57 && tc >= 48)) return [0];
str += tc - 48;
hash = crctable[index[2]] ^ (hash >>> 8);
tc = hash & 0xff ^ index[1];
if (!(tc <= 57 && tc >= 48)) return [0];
str += tc - 48;
hash = crctable[index[1]] ^ (hash >>> 8);
tc = hash & 0xff ^ index[0];
if (!(tc <= 57 && tc >= 48)) return [0];
str += tc - 48;
hash = crctable[index[0]] ^ (hash >>> 8);
return [1, str];
};
create_table();
let index = new Array(4);
return function (input) {
let ht = parseInt('0x' + input) ^ 0xffffffff,
snum,
i,
lastindex,
deepCheckData;
for (i = 3; i >= 0; i--) {
index[3 - i] = getcrcindex(ht >>> (i * 8));
snum = crctable[index[3 - i]];
ht ^= snum >>> ((3 - i) * 8);
}
for (i = 0; i < 10000000; i++) {
lastindex = crc32lastindex(i);
if (lastindex == index[3]) {
deepCheckData = deepCheck(i, index);
if (deepCheckData[0]) break;
}
}
if (i == 10000000) return -1;
return i + '' + deepCheckData[1];
};
};
// 函数统一接口
const deliver = {
// 格式化时间戳,默认返回hh:mm:ss;指定type加上yy:mm:dd
timeFormat : (time,type) => {
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 type ? Y + M + D + h +m + s : h + m + s;
},
// 格式化字节,逢千换单位,最高到G字节
sizeFormat : (size) => {
let unit = ["B", "K", "M", "G"], i = unit.length - 1, dex = 1024 ** i, vor = 1000 ** i;
while (dex > 1) {
if (size >= vor) {
size = (size / dex).toFixed(2);
break;
}
dex = dex / 1024;
vor = vor / 1000;
i--;
}
return size + unit[i];
},
// 格式化计数法
unitFormat : (num) => {
let unit = ["", "万", "亿"], i = unit.length - 1, dex = 10000 ** i;
while (dex > 1) {
if (num >= dex) {
num = (num / dex).toFixed(1);
break;
}
dex = dex / 10000;
i--;
}
return num + unit[i];
},
// 数组冒泡排序,指定rev输出逆序
bubbleSort : (arr, rev) => {
let temp=[];
rev = rev ? true : false;
for (let i = 0; i < arr.length - 1; i++) {
let bool = true;
for (let j = 0; j < arr.length - 1 - i; j++) {
if(arr[j] > arr[j+1]){
temp = arr[j];
arr[j] = arr[j+1];
arr[j+1] = temp;
bool = false;
}
}
if (bool) break;
}
if (rev) return arr.reverse();
return arr;
},
// 数组随机提取,num指定随机提取几个
randomArray : (arr, num) => {
let out = [];
num = num || 1;
num = num < arr.length ? num : arr.length;
while(out.length < num){
var temp = (Math.random()*arr.length) >> 0;
out.push(arr.splice(temp,1)[0]);
}
return out;
},
// bv/av互转,算法见https://www.zhihu.com/question/381784377/answer/1099438784
convertId : (str) => {
let table = 'fZodR9XQDSUm21yCkr6zBqiveYah8bt4xsWpHnJE7jL5VG3guMTKNPAwcF';
let tr = {}, s = [11, 10, 3, 8, 4, 6], xor = 177451812, add = 8728348608;
for (let i = 0; i < 58; i++) tr[table[i]] = i;
if (!(1 * str)) {
// str为字符串(BVxxxxxxxxxx)则转化为aid(纯数字)
let r = 0;
for (let i = 0; i < 6; i++) r += tr[str[s[i]]]*58**i;
return (r-add)^xor;
}
else {
// str为数字(aid)则转化为BV(BVxxxxxxxxxx)
str = (str^xor) + add;
let r = ['B', 'V', 1, '', '', 4, '', 1, '', 7, '', ''];
for (let i = 0; i < 6; i++) r[s[i]] = table[parseInt(str/58**i)%58];
return r.join("");
}
},
// key secret,见https://github.com/Henryhaohao/Bilibili_video_download
sign : () => {
let table = 'rbMCKn@KuamXWlPMoJGsKcbiJKUfkPF_8dABscJntvqhRSETg', str = '';
for (let i = table.length -1; i >= 0; i--) str = str + String.fromCharCode(table[i].charCodeAt() + 2);
return str.split(':')
},
// 对象转url的查询部分
obj2search : (url, obj) => {
if (obj) {
let arr = [],i = 0;
for (let key in obj) {
if(obj[key] !== "" && obj[key] !== "undefined" && obj[key] !== null) {
arr[i] = key + "=" + obj[key];
i++;
}
}
if (url) url = url + "?" + arr.join("&");
else url = arr.join("&");
}
return url;
},
// 提取url查询部分成对象
search2obj : (url) => {
url = url.split('#')[0];
url = url.split('?')[1] ? url.split('?')[1].split('&') : "";
if (!url) return;
let obj = {};
for (let i = 0; i < url.length; i++) obj[url[i].split('=')[0]] = url[i].split('=')[1];
return obj;
},
// cookies对象,通过属性访问键值
getCookies : () => {
let cookies = document.cookie.split('; ');
let obj = cookies.reduce((pre, next) => {
let key = next.split('=')[0];
let val = next.split('=')[1];
pre[key] = val;
return pre;
}, {});
return obj;
},
// 转换解码后的protobuf到xml
toXml : (danmaku, pid) => {
return new Promise(function (resolve) {
//按出现时间排序弹幕,能避免反复插入dom元素,明显提高性能
//排序后40000条弹幕旧播放器能在1秒左右处理完
danmaku.sort(function (a, b) {
return a.progress - b.progress;
});
let dom = (new DOMParser()).parseFromString('<?xml version="1.0" encoding="UTF-8"?><i><chatserver>chat.bilibili.com</chatserver><chatid>' + cid + '</chatid><mission>0</mission><maxlimit>99999</maxlimit><state>0</state><real_name>0</real_name><source>e-r</source></i>', "text/xml");
let root = dom.childNodes[0];
let d, attr, dmk;
for (let i in danmaku) {
dmk = danmaku[i];
d = dom.createElement("d");
attr = [dmk.progress / 1000, dmk.mode, dmk.fontsize, dmk.color, dmk.ctime, 0, dmk.midHash, dmk.id];
d.setAttribute("p", attr.join(","));
d.appendChild(dom.createTextNode(dmk.content));
root.appendChild(d);
}
resolve(new XMLSerializer().serializeToString(dom));
});
},
getSegDanmaku: function (onload) {
let protoSegments = [];
getSegConfig().then(getAllSeg);
function getSegConfig() {
return new Promise(function (resolve) {
let xhr = new XMLHttpRequest();
xhr.addEventListener("load", function () {
let res = protoView.decode(new Uint8Array(xhr.response));
resolve(res);
});
xhr.open("get", "https://api.bilibili.com/x/v2/dm/web/view?type=1&oid=" + cid + "&pid=" + aid);
xhr.responseType = "arraybuffer";
xhr.send();
});
}
// 获得所有分段
function getAllSeg(config) {
let total = config.dmSge.total;
let allrequset = [];
let reqUrl = "https://api.bilibili.com/x/v2/dm/web/seg.so?type=1&oid=" + cid + "&pid=" + aid;
for (let index = 1; index <= total; index++) {
allrequset.push(new Promise(function (resolve) {
let xhr = new XMLHttpRequest();
xhr.addEventListener("load", function () {
protoSegments[index] = xhr.response;
resolve();
});
xhr.open("get", reqUrl + "&segment_index=" + index);
xhr.responseType = "arraybuffer";
xhr.send();
}));
}
// 完成所有的网络请求大概要300ms
return Promise.all(allrequset).then(function () { onload(protoSegments); });
}
},
// 添加全局样式
setGlobalStyle : async () => {
let csss = API.style.uiface;
let style = document.createElement("style");
csss = csss + API.style.online;
csss = csss + API.style.search;
csss = csss + API.style.download;
if (config.reset.playershadow) csss = csss + API.style.playshadow;
if (config.reset.like) csss = csss + API.style.like;
style.setAttribute("type", "text/css");
style.appendChild(document.createTextNode(csss));
setTimeout(() => {if (document.head) document.head.appendChild(style)});
},
// 播放器调试通知
debug : (...msg) => {
let node = document.getElementsByClassName("bilibili-player-video-toast-bottom")[0];
if (!node) {debug.log(...msg); return;}
let warn = msg[1] || "", delay = msg[2] || 3000;
let item = document.createElement("div"),
text = document.createElement("div"),
span = document.createElement("span"),
red = document.createElement("span");
delay = delay ? delay : 3000;
item.setAttribute("class","bilibili-player-video-toast-item bilibili-player-video-toast-msg");
item.appendChild(text);
text.setAttribute("class","bilibili-player-video-toast-item-text");
text.appendChild(span);
if (warn) text.appendChild(red);
span.setAttribute("class","video-float-hint-text");
span.innerText = msg[0];
red.setAttribute("class","video-float-hint-btn hint-red");
red.innerText = warn ? warn : "";
node.children[0] ? node.children[0].replaceWith(item) : node.appendChild(item);
setTimeout(() => item.remove(), delay);
},
// xhr返回json校验
xhrJsonCheck : (data, toast) => {
data = JSON.parse(data);
if ("code" in data && data.code !== 0) {
let msg = data.msg || data.message || "";
if (toast) debug.msg("xhr错误:", data.code + " " + msg);
throw [data.code, msg, data]
}
return data;
},
// 全局变量监听
getVariable : (value) => {
function read(arr){
switch (arr[0]) {
case "aid" : aid = arr[1];
break;
case "cid" : cid = arr[1];
break;
case "__playinfo__" : __playinfo__ = __playinfo__ || arr[1];
break;
}
}
Object.defineProperty(unsafeWindow, "aid",{set: (value) => {read(["aid", value])}, get: () => {return aid}, configurable: true});
Object.defineProperty(unsafeWindow, "cid",{set: (value) => {read(["cid", value])}, get: () => {return cid}, configurable: true});
Object.defineProperty(unsafeWindow, "__playinfo__",{set: (value) => {read(["__playinfo__", value])}, get: () => {return __playinfo__}, configurable: true});
Object.defineProperty(unsafeWindow, "__BILI_CONFIG__",{get: () => {return {"show_bv" : false}}, configurable: true});
if (LOCATION[2] == "live.bilibili.com" && config.reset.roomplay) Object.defineProperty(unsafeWindow, "__NEPTUNE_IS_MY_WAIFU__",{get: () => {return undefined}, configurable: true});
},
// 重写网页
write : (html) => {
document.open();
document.write(html);
document.close();
},
// 重写版面
reSction : async () => {
if (!config.reset.grobalboard) return;
if (!unsafeWindow.$) {
let jq = document.createElement("script");
jq.setAttribute("type", "text/javascript");
jq.setAttribute("src", "//static.hdslb.com/js/jquery.min.js");
document.body.insertBefore(jq,document.body.firstChild);
}
document.getElementById("internationalHeader").setAttribute("style", "visibility:hidden;");
let newh = document.createElement("div");
let script = document.createElement("script");
let foot = document.getElementsByClassName("international-footer");
script.setAttribute("type", "text/javascript");
script.setAttribute("src", "//s1.hdslb.com/bfs/seed/jinkela/header/header.js");
if(document.getElementsByClassName("mini-type")[0]) {
if (location.href.includes("blackboard/topic_list") || location.href.includes("blackboard/x/act_list")) newh.setAttribute("class", "z-top-container has-menu");
else newh.setAttribute("class", "z-top-container");
}
else newh.setAttribute("class", "z-top-container has-menu");
document.body.insertBefore(newh,document.body.firstChild);
document.body.insertBefore(script,document.body.firstChild);
if (foot[0]) {
let div = document.createElement("div");
div.setAttribute("class", "footer bili-footer report-wrap-module");
div.setAttribute("id", "home_footer");
foot[0].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);
}
window.setTimeout(() => {deliver.resetNodes()}, 3000);
},
// 下载视频
download : {
// 创建播放器右键菜单
init : async (node) => {
if (!config.reset.download) return;
let li = document.createElement("li");
let div = document.createElement("a");
li.setAttribute("class", "context-line context-menu-function bili-old-download");
div.setAttribute("class", "context-menu-a js-action");
div.setAttribute("href", "javascript:void(0);");
div.innerText = "下载视频";
li.appendChild(div);
node.firstChild.appendChild(li);
div.onclick = deliver.download.setTable;
},
// 配置下载数据
setTable : async () => {
debug.msg("正在获取视频链接", ">>>");
let qua = {120 : "4K", 116 : "1080P60", 112 : "1080P+", 80 : "1080P", 74 : "720P60", 64 : "720P", 48 : "720P", 32 : "480P", 16 : "360P"};
let bps = {30216 : "64kbps", 30232 : "128kbps", 30280 : "320kbps"}
let path = __playinfo__ ? (__playinfo__.data || (__playinfo__.durl && __playinfo__) || __playinfo__.result) : "";
try {
url = url ? url : ((path && path.durl) ? [await deliver.download.geturl()] : await Promise.all([deliver.download.geturl(), deliver.download.geturl("flv")]));
if (url[1]) {
if (__playinfo__) path.durl = url[1].durl || (url[1].data ? url[1].data.durl : (url[1].result ? url[1].result.durl : ""));
else path = url[1].data || (url[1].durl && url[1]) || url[1].result;
}
}
catch(e) {debug.log("下载获取", url); url = [1]}
try {
// 获取mp4
if (url[0] && url[0].durl) {
url = url[0];
mdf.mp4 = [["1080P", url.durl[0].url.replace("http:", ""), deliver.sizeFormat(url.durl[0].size)]];
navigator.clipboard.writeText(url.durl[0].url);
}
if (path) {
// 获取flv
if (path.durl) {
// durl可能是mp4
if (path.durl[0] && path.durl[0].url.includes("mp4?")) {
if (!mdf.mp4) mdf.mp4 = [];
mdf.mp4.push([qua[path.quality],path.durl[0].url.replace("http:", ""), deliver.sizeFormat(path.durl[0].size)]);
}
else {
mdf.flv = [];
for (let i = 0; i < path.durl.length; i++) mdf.flv.push([qua[path.quality], path.durl[i].url.replace("http:", ""), deliver.sizeFormat(path.durl[i].size)]);
}
}
// 获取DASH
if (path.dash) {
mdf.dash = {}
// 获取视频流
if (path.dash.video) {
for (let i = 0; i < path.dash.video.length; i++) {
if (path.dash.video[i].codecs.startsWith("avc")) {
if (!mdf.dash.avc) mdf.dash.avc = [];
mdf.dash.avc.push([qua[path.dash.video[i].id], path.dash.video[i].baseUrl.replace("http:", ""), deliver.sizeFormat(path.dash.video[i].bandwidth * path.dash.duration / 8)]);
}
else {
if (!mdf.dash.hev) mdf.dash.hev = [];
mdf.dash.hev.push([qua[path.dash.video[i].id], path.dash.video[i].baseUrl.replace("http:", ""), deliver.sizeFormat(path.dash.video[i].bandwidth * path.dash.duration / 8)]);
}
}
}
// 获取音频流
if (path.dash.audio) {
for (let i = 0; i < path.dash.audio.length; i++) {
if (!mdf.dash.aac) mdf.dash.aac = [];
mdf.dash.aac.push([path.dash.audio[i].id, path.dash.audio[i].baseUrl.replace("http:", ""), deliver.sizeFormat(path.dash.audio[i].bandwidth * path.dash.duration / 8)]);
}
// 倒序音频
mdf.dash.aac = deliver.bubbleSort(mdf.dash.aac, true);
// 标注大概码率
for (let i = 0; i < mdf.dash.aac.length; i++) if (mdf.dash.aac[i][0] in bps) mdf.dash.aac[i][0] = bps[mdf.dash.aac[i][0]];
}
}
// 获取弹幕
if (xml) {
let blob = new Blob([xml]);
mdf.xml = [];
bloburl.xml = URL.createObjectURL(blob);
mdf.xml.push(["弹幕", bloburl.xml, deliver.sizeFormat(blob.size)]);
}
else {
mdf.xml = [];
mdf.xml.push(["弹幕", "//api.bilibili.com/x/v1/dm/list.so?oid=" + cid, "--------"]);
}
// 获取其他
if (__INITIAL_STATE__) {
mdf.xml = mdf.xml || [];
mdf.xml.push(["封面", (__INITIAL_STATE__.videoData && __INITIAL_STATE__.videoData.pic || __INITIAL_STATE__.mediaInfo.cover).replace("http:", ""), "--------"]);
if (__INITIAL_STATE__.mediaInfo && __INITIAL_STATE__.mediaInfo.bkg_cover) mdf.xml.push(["海报", __INITIAL_STATE__.mediaInfo.bkg_cover.replace("http:", ""), "--------"]);
if (__INITIAL_STATE__.mediaInfo && __INITIAL_STATE__.mediaInfo.specialCover) mdf.xml.push(["海报", __INITIAL_STATE__.mediaInfo.specialCover.replace("http:", ""), "--------"]);
if (__INITIAL_STATE__.videoData && __INITIAL_STATE__.videoData.subtitle && __INITIAL_STATE__.videoData.subtitle.list) for (let i = 0; i < __INITIAL_STATE__.videoData.subtitle.list.length; i++) mdf.xml.push([__INITIAL_STATE__.videoData.subtitle.list[i].lan_doc, __INITIAL_STATE__.videoData.subtitle.list[i].subtitle_url.replace("http:", ""), "--------"]);
}
}
deliver.download.item();
mdf = {};
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("下载配置", ...e)}
},
// 拉取视频链接
geturl : async (...arg) => {
let url = deliver.download.playurl(...arg);
try {
if (!url) throw url;
let data = await xhr.GM(url);
return deliver.xhrJsonCheck(data);
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("下载拉取", ...e);}
},
// 配置视频链接
playurl : (type, qn) => {
let obj = {}, sign = deliver.sign();
aid = aid || unsafeWindow.aid;
cid = cid || unsafeWindow.cid;
qn = qn || 120;
type = type || "mp4";
if (!cid) return;
switch(type){
case 'dash' : if (pgc) return deliver.obj2search(API.url.pgc, {avid : aid, cid : cid, qn : qn, fourk : 1, otype : 'json', fnver : 0, fnval : 16});
else return deliver.obj2search(API.url.x, {avid : aid, cid : cid, qn : qn, fourk : 1, otype : 'json', fnver : 0, fnval : 16});
break;
case 'flv' : if (pgc) return deliver.obj2search(API.url.pgc, {avid : aid, cid : cid, qn : qn, fourk : 1, otype : 'json'});
else return deliver.obj2search(API.url.x, {avid : aid, cid : cid, qn : qn, fourk : 1, otype : 'json'});
break;
case 'off' : obj = {appkey : sign[0], cid : cid, otype : 'json', qn : qn, quality : qn, type : ''}
obj.sign = md5(deliver.obj2search("", obj) + sign[1]);
return deliver.obj2search(API.url.sign, obj);
break;
case 'mp4' : obj = {appkey : sign[0], cid : cid, otype : 'json', platform : 'android_i', qn : 208}
obj.sign = md5(deliver.obj2search("", obj) + sign[1]);
if (pgc) return deliver.obj2search(API.url.pgcproj, obj);
return deliver.obj2search(API.url.proj, obj);
break;
}
},
// 创建下载面板
item : () => {
let timer, top = document.getElementById("bili-old-download-table");
if (top) {
// 刷新下载面板
top.remove();
// 释放bolb残留
if (bloburl.xml) {
window.URL.revokeObjectURL(bloburl.xml);
bloburl.xml = "";
}
}
if (!mdf.mp4 && !mdf.flv && !mdf.dash) throw (debug.msg("未找到任何视频链接 ಥ_ಥ"), mdf);
function addBox(obj, name, type, quatily){
let box = document.createElement("div");
box.setAttribute("class", "download-box");
let tab = document.createElement("div");
tab.setAttribute("class", "download-type " + type);
tab.innerHTML = name;
box.appendChild(tab);
top.appendChild(box);
switch (name) {
case "mp4" : name = ".mp4"; break;
case "avc" : name = ".m4v"; break;
case "hev" : name = ".m4v"; break;
case "aac" : name = ".m4a"; break;
case "其他" : name = ".xml"; break;
}
let qua = quatily;
for (let i = 0; i < obj.length; i++) {
switch (qua || obj[i][0]) {
case "1080P" : quatily = "quality-1080p"; break;
case "720P" : quatily = "quality-720p"; break;
case "480P" : quatily = "quality-480p"; break;
case "360P" : quatily = "quality-360p"; break;
case "320kbps" : quatily = "quality-720p"; break;
case "128kbps" : quatily = "quality-480p"; break;
case "64kbps" : quatily = "quality-360p"; break;
default : quatily = "quality-high";
}
let a = document.createElement("a");
let q = document.createElement("div");
let s = document.createElement("div");
q.innerHTML = obj[i][0];
obj[i][0] = "弹幕" || "封面" ? "av" + aid : obj[i][0];
name = obj[i][2] == "--------" ? "" : name;
a.setAttribute("download", obj[i][0] + name);
a.setAttribute("href", obj[i][1]);
a.setAttribute("target", "_blank");
q.setAttribute("class", "download-quality " + quatily);
s.setAttribute("class", "download-size");
s.innerHTML = obj[i][2];
a.appendChild(q);
a.appendChild(s);
box.appendChild(a);
}
}
top = document.createElement("div");
top.setAttribute("id", "bili-old-download-table");
if (mdf.mp4) addBox(mdf.mp4, "mp4", "download-mp4");
if (mdf.dash) {
if (mdf.dash.avc) addBox(mdf.dash.avc, "avc", "download-avc");
if (mdf.dash.hev) addBox(mdf.dash.hev, "hev", "download-hev");
if (mdf.dash.aac) addBox(mdf.dash.aac, "aac", "download-aac");
}
if (mdf.flv) addBox(mdf.flv, "flv", "download-flv");
if (mdf.xml) addBox(mdf.xml, "其他", "download-xml", "360P");
document.body.appendChild(top);
debug.msg("右键另存为或右键IDM下载", "详见脚本简介", 3000);
top.onmouseover = () => window.clearTimeout(timer);
top.onmouseout = () => {timer = window.setTimeout(() => {
top.remove();
if (bloburl.xml) {
window.URL.revokeObjectURL(bloburl.xml);
bloburl.xml = "";
}
}, 1000)};
}
},
// 切p相关
switchVideo : async () => {
let title = document.getElementsByTagName("h1")[0] ? document.getElementsByTagName("h1")[0].title : "";
if (config.reset.download) {url = ""; mdf = {}; hash = [];};
if (config.reset.selectdanmu && document.getElementsByClassName("bilibili-player-filter-btn")[1]) document.getElementsByClassName("bilibili-player-filter-btn")[1].click();
if (config.reset.midcrc && !config.reset.danmuku && !hash[0]) {
let data = await xhr.true(deliver.obj2search(API.url.listso, {oid : cid}));
data.match(/d p=".+?"/g).forEach((v) => {hash.push(v.split(",")[6])});
}
setTimeout(()=>{
if (config.reset.viewbofqi && document.querySelector("#bofqi")) document.querySelector("#bofqi").scrollIntoView({behavior: 'smooth', block: 'center'});
if (config.reset.widescreen && document.querySelector(".bilibili-player-iconfont.bilibili-player-iconfont-widescreen.icon-24wideoff")) {
document.querySelector(".bilibili-player-video-btn.bilibili-player-video-btn-widescreen").click();
}
if (config.reset.danmakuoff && !document.querySelector(".bilibili-player-video-btn.bilibili-player-video-btn-danmaku.video-state-danmaku-off")) {
if (document.querySelector(".bilibili-player-video-btn.bilibili-player-video-btn-danmaku")) {
document.querySelector(".bilibili-player-video-btn.bilibili-player-video-btn-danmaku").click();
}
}
});
},
// 付费预览
removePreview : async (node) => {
try {
if (!config.reset.preview) return;
let hint = document.getElementsByClassName("video-float-hint-text")[0];
// 倒计时长度,单位:秒
let i = 10;
let sec = document.createElement("span");
sec.setAttribute("class", "video-float-hint-btn second-cut");
hint.parentNode.appendChild(sec);
function cut(){
sec.innerText = i - 1 + "s";
if (i==0) {
node.remove();
return;
}
i = i - 1;
window.setTimeout(cut,1000);
}
new cut();
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("付费预览", ...e)}
},
// 超链接转化
avdesc : async () => {
if (!config.rewrite.av || !aid || LOCATION[3] != 'video') return;
let desc = document.getElementsByClassName("info");
if (desc[1] && desc[1].outerHTML.match(/BV[A-Za-z0-9]+/i)) {
let paster = desc[1].outerHTML.match(/BV[A-Za-z0-9]+/i);
for (let i = 0; i < paster.length; i++){
let newer = "av" + deliver.convertId(paster[i]);
newer = '<a target="_blank" href="//www.bilibili.com/video/' + newer + '">' + newer + '</a>';
desc[1].innerHTML = desc[1].outerHTML.replace(paster[i], newer);
}
}
},
// 节点监听
resetNodes : async (ext) => {
let remove = (node, type, hidden, index) => {
index ? index : index = 0;
switch(type){
// 一般能移除的就移除,否则隐藏
case "id" : node = document.getElementById(node);break;
case "class" : node = document.getElementsByClassName(node)[index] ? document.getElementsByClassName(node)[index] : "";break;
case "tag" : node = document.getElementsByTagName(node)[index] ? document.getElementsByTagName(node)[index] : "";break;
}
if (!node || node.getAttribute("hidden")) return;
debug.debug("移除节点", node);
hidden ? node.setAttribute("hidden", "hidden") : node.remove();
}
// 隐藏联系客服
remove("contact-help", "class", true);
// 移除新版提示
remove("new-entry", "class");
if (window.recbtn) remove("ver", "class");
remove("trynew-btn", "class");
if (config.reset.panel) remove("bilibili-player-ending-panel", "class");
// 移除app下载浮动框
remove("fixed_app_download", "id");
remove("app-download", "class");
// 移除直播水印
remove("bilibili-live-player-video-logo", "class");
// 移除失效顶栏
remove("bili-header-m report-wrap-module", "class", false, 1);
// 移除主页昨日榜
if (window.recbtn) remove("rec-btn prev", "class");
// 移除主页七日榜
if (window.recbtn) remove("rec-btn next", "class");
// 移除双重视频下载右键菜单
if (document.getElementsByClassName("bili-old-download")[1]) document.getElementsByClassName("bili-old-download")[0].remove();
// 使顶栏透明
if (config.reset.headblur) {
let blur = document.getElementsByClassName("blur-bg");
if (blur[0]) blur[0].removeAttribute("style");
}
},
// 失效分区
fixSort : {
// av
video : async () => {
let timer = window.setInterval(()=>{
let tminfo = document.getElementsByClassName("tm-info");
if (tminfo[0]) {
window.clearInterval(timer);
if (!(tid in API.sort)) return;
let nodes = tminfo[0].childNodes;
// 创建分区信息节点并写入tid对应的分区数据
nodes[1].replaceWith(nodes[0].cloneNode(true));
nodes[2].replaceWith(nodes[0].cloneNode(true));
nodes[2].childNodes[1].remove();
nodes[1].childNodes[0].href = API.sort[API.sort[tid][0]][2];
nodes[1].childNodes[0].innerText = API.sort[API.sort[tid][0]][1];
nodes[2].childNodes[0].href = API.sort[tid][2];
nodes[2].childNodes[0].innerText = API.sort[tid][1];
}
},1000);
},
// 稍后再看
watchlater : async (data) => {
let timer = window.setInterval(async ()=>{
let tminfo = document.getElementsByClassName("tm-info");
// 判断是否是少后再看页面
if (tminfo[0]&&aid) {
window.clearInterval(timer);
let child = tminfo[0].childNodes;
if (child[2].nodeType === 8) {
try {
// 通过链接获取tid
data = await xhr.true(deliver.obj2search(API.url.view, {"aid" : aid}));
tid = deliver.xhrJsonCheck(data).data.tid;
if (!(tid in API.sort)) return;
// 创建分区信息节点并写入tid对应的分区数据
child[2].replaceWith(child[0].cloneNode(true));
child[4].replaceWith(child[0].cloneNode(true));
child[4].childNodes[1].remove();
child[2].childNodes[0].href = API.sort[API.sort[tid][0]][2];
child[2].childNodes[0].innerText = API.sort[API.sort[tid][0]][1];
child[4].childNodes[0].href = API.sort[tid][2];
child[4].childNodes[0].innerText = API.sort[tid][1];
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("分区·稍后再看", ...e)}
}
}
},1000);
},
},
// 点赞功能
setLike : async (data) => {
if (!config.reset.like) return;
let timer = window.setInterval(async () => {
let coin = document.getElementsByClassName("bilibili-player-video-subtitle")[0];
let number = document.getElementsByClassName("number")[0];
let node = document.getElementsByClassName("coin")[0];
// 判断页面渲染进度
if (coin && node) {
window.clearInterval(timer);
let span = document.createElement("span");
let move = document.createElement("i");
let moved = document.createElement("b");
let text = document.createTextNode("点赞 --");
let arg = text;
// 创建点赞数据相关节点并初始化
span.setAttribute("class", "u like");
span.setAttribute("style", "margin-right : 5px;");
span.appendChild(move);
span.appendChild(moved);
span.appendChild(text);
move.setAttribute("class", "l-icon-move");
move.setAttribute("style", "width : 22px;height : 22px;background-position : -660px -2068px;");
moved.setAttribute("class", "l-icon-moved");
moved.setAttribute("style", "width : 22px;height : 22px;background-position : -725px -2068px;display : none;");
try {
move.onclick = async () => {
// 没有点赞过绑定点赞点击事件
if (!uid) {
// 没有登录绑定快捷登录
document.getElementsByClassName("c-icon-move")[0].click();
return;
}
// 构造并请求点赞表单
let msg = "aid=" + aid + "&like=1&csrf=" + deliver.getCookies().bili_jct;
data = await xhr.post(API.url.like, "application/x-www-form-urlencoded", msg);
data = deliver.xhrJsonCheck(data).ttl;
// 点亮点赞图标并修改显示数据
debug.msg("点赞成功!");
document.getElementsByClassName("l-icon-move")[0].setAttribute("style", "width : 22px;height : 22px;background-position : -660px -2068px;display : none;");
document.getElementsByClassName("l-icon-moved")[0].setAttribute("style", "width : 22px;height : 22px;background-position : -725px -2068px;");
if (arg.nodeValue.match("万")) return;
let number = 1 * arg.nodeValue.match(/[0-9]+/) + 1;
text = document.createTextNode(" 点赞 " + number)
arg.replaceWith(text);
arg = text;
}
moved.onclick = async () => {
// 点赞过绑定取消点赞点击事件
// 构造并请求取消点赞表单
let msg = "aid=" + aid + "&like=2&csrf=" + deliver.getCookies().bili_jct;
data = await xhr.post(API.url.like, "application/x-www-form-urlencoded", msg);
data = deliver.xhrJsonCheck(data).ttl;
// 熄灭点赞图标并修改显示数据
debug.msg("点赞撤回!");
document.getElementsByClassName("l-icon-move")[0].setAttribute("style", "width : 22px;height : 22px;background-position : -660px -2068px;");
document.getElementsByClassName("l-icon-moved")[0].setAttribute("style", "width : 22px;height : 22px;background-position : -725px -2068px;display : none;");
if (arg.nodeValue.match("万")) return;
let number = 1 * arg.nodeValue.match(/[0-9]+/) - 1;
text = document.createTextNode(" 点赞 " + number)
arg.replaceWith(text);
arg = text;
}
number.insertBefore(span, node);
// 获取点赞数据
data = await xhr.true(deliver.obj2search(API.url.view, {"aid" : aid}));
data = deliver.xhrJsonCheck(data).data.stat.like;
document.getElementsByClassName("like")[0].setAttribute("title", "点赞人数" + data);
text = document.createTextNode(" 点赞 " + deliver.unitFormat(data));
arg.replaceWith(text);
arg = text;
if (!uid) return;
data = deliver.xhrJsonCheck(await xhr.true(deliver.obj2search(API.url.haslike, {"aid" : aid}))).data;
if (data == 1) {
// 点赞过点亮图标
move.setAttribute("style", "width : 22px;height : 22px;background-position : -660px -2068px;display : none;");
moved.setAttribute("style", "width : 22px;height : 22px;background-position : -725px -2068px;");
}
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("点赞功能", ...e)}
}
},100);
},
// 收藏播放,收藏列表视频过多将导致视频加载及切换缓慢
setMediaList : {
init : async (data) => {
if (!ml) {
// 二次初始化播放器
let timer = setInterval(() => {
if (!unsafeWindow.BilibiliPlayer) return;
clearInterval(timer);
if (__playinfo__.data.accept_quality[0] < 120) return;
let e = "cid=" + unsafeWindow.cid + "&aid=" + unsafeWindow.aid;
unsafeWindow.GrayManager && unsafeWindow.GrayManager.reload(e);
unsafeWindow.BiliCm && unsafeWindow.BiliCm.Core && unsafeWindow.BiliCm.Core.reset();
},100)
return;
}
if (data){
// 以传参data决定处理类型
try {
// 获取首个视频av并跳转
data = await xhr.true(deliver.obj2search(API.url.medialist, {"media_id" : ml, "pn" : 1, "ps":1}));
data = deliver.xhrJsonCheck(data).data;
location.replace("https://www.bilibili.com/video/av" + data.medias[0].id);
}
catch(e) {
// 跳转失败,清理残余
GM_setValue("medialist", 0);
debug.error(e);
}
}
else {
try {
let avs = [], value = [], promises = [];
// 获取收藏列表,这里获取只能获取到aid
data = await xhr.true(deliver.obj2search(API.url.ids4Player, {"media_id" : ml}));
data = deliver.xhrJsonCheck(data).data;
for (let i = 0; i < data.medias.length; i++) {
ids[i] = data.medias[i].id;
avs[i] = "av" + data.medias[i].id;
}
// 同时获取所有aid对应的数据,使用Promise.all对齐,该api会直接忽略失效视频
while (avs.length) {
let i = avs.length > 20 ? 20 : avs.length;
value = avs.splice(0,i);
promises.push(xhr.true(deliver.obj2search(API.url.cards, {"ids" : value.join("%2C")})));
}
value = [];
data = await Promise.all(promises);
// 格式化数据并排序
for (let i = 0; i < data.length; i++) {
data[i] = deliver.xhrJsonCheck(data[i]);
for (let key in data[i].data) avs.push(data[i].data[key]);
}
for (let i = 0; i < ids.length; i++) {
for (let j = 0; j < avs.length; j++) {
if (avs[j].aid == ids[i]) {
value.push(avs[j]);
break;
}
}
}
ids = value;
let timer = window.setInterval(() => {
if (unsafeWindow.BilibiliPlayer) {
clearInterval(timer);
// 将视频列表重构为稍后再看列表
for (let i = 0; i < ids.length; i++) {
ids[i].progress = 0;
ids[i].add_at = ids[i].ctime;
ids[i].pages = [];
ids[i].pages[0] = {};
ids[i].pages[0].cid = ids[i].cid;
ids[i].pages[0].dimension = ids[i].dimension;
ids[i].pages[0].duration = ids[i].duration;
ids[i].pages[0].from = "vupload";
ids[i].pages[0].page = 1;
ids[i].pages[0].part = ids[i].title;
ids[i].pages[0].vid = "";
ids[i].pages[0].weblink = "";
}
let toview = {"code":0, "message" : "0", "ttl" : 1, "data" : {"count" : ids.length, "list" : ids}};
// 保存初始aid,以便判断是否切p
oid = ids[0].aid;
debug.debug("收藏列表", toview);
// 构造初始化参数并重新初始化播放器
obj = {"aid" : ids[0].aid, "cid" : ids[0].cid, "watchlater":encodeURIComponent(JSON.stringify(toview))}; // 重构初始化播放器参数
unsafeWindow.BilibiliPlayer(obj);
let bpui = document.getElementsByClassName("bpui-button-text");
let t = setInterval(() => {
// 更新列表名称
if (bpui[1]) {
clearInterval(t);
bpui[1].firstChild.innerText = "收藏列表";
}
},100);
}
},100);
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("收藏模拟", ...e)}
}
},
// aid变化监听
fixvar : async () => {
if (!aid) aid = unsafeWindow.aid ? unsafeWindow.aid : aid;
if (oid) {
if (oid != unsafeWindow.aid) {
// 收藏播放切p判断
aid = unsafeWindow.aid ? unsafeWindow.aid : aid;
oid = unsafeWindow.aid;
deliver.setMediaList.restore();
}
}
},
// 收藏播放更新
restore : async () => {
let data;
history.replaceState(null,null,"https://www.bilibili.com/video/av" + aid + location.search + location.hash);
for (let i=0;i<ids.length;i++) if (ids[i].aid==aid) data = ids[i];
let video_info = document.getElementById("viewbox_report").childNodes;
let up_info = document.getElementById("v_upinfo").childNodes;
let tag = document.getElementById("v_tag").childNodes;
let desc = document.getElementById("v_desc").childNodes;
let arc_toolbar_report = document.getElementById("arc_toolbar_report").childNodes;
let title = video_info[0];
let info = video_info[1];
let number = video_info[2];
document.title = data.title;
title.title = data.title;
title.childNodes[1].innerText = data.title;
info.childNodes[3].innerText = deliver.timeFormat(data.pubdate*1000);
number.childNodes[0].title = "总播放数" + data.stat.view;
number.childNodes[0].innerText = deliver.unitFormat(data.stat.view);
number.childNodes[1].title = "总弹幕数" + data.stat.danmaku;
number.childNodes[1].innerText = deliver.unitFormat(data.stat.danmaku);
if (data.stat.his_rank > 0) number.childNodes[2].innerText = "最高全站日排行" + data.stat.his_rank + "名";
else try {number.childNodes[2].setAttribute("hidden", "hidden");} catch(e) {}
if (number.childNodes[4].className == "u like") {
number.childNodes[4].title = "点赞人数" + data.stat.like;
number.childNodes[4].childNodes[2].replaceWith(document.createTextNode("点赞 " + deliver.unitFormat(data.stat.like)));
number.childNodes[5].title = "投硬币枚数" + data.stat.coin;
number.childNodes[5].childNodes[2].replaceWith(document.createTextNode("硬币 " + deliver.unitFormat(data.stat.coin)));
number.childNodes[6].title = "收藏人数" + data.stat.favorite;
number.childNodes[6].childNodes[2].replaceWith(document.createTextNode("收藏 " + deliver.unitFormat(data.stat.favorite)));
}
else {
number.childNodes[4].title = "投硬币枚数" + data.stat.coin;
number.childNodes[4].childNodes[2].replaceWith(document.createTextNode("硬币 " + deliver.unitFormat(data.stat.coin)));
number.childNodes[5].title = "收藏人数" + data.stat.favorite;
number.childNodes[5].childNodes[2].replaceWith(document.createTextNode("收藏 " + deliver.unitFormat(data.stat.favorite)));
}
up_info[0].childNodes[1].href = "https://space.bilibili.com/" + data.owner.mid;
up_info[0].childNodes[1].childNodes[0].src = data.owner.face;
up_info[1].childNodes[0].childNodes[0].href = "https://space.bilibili.com/" + data.owner.mid;
up_info[1].childNodes[0].childNodes[0].innerText = data.owner.name;
up_info[1].childNodes[1].childNodes[0].innerText = "up主简介";
up_info[1].childNodes[2].childNodes[0].innerText = "投稿 --";
up_info[1].childNodes[2].childNodes[1].innerText = "粉丝 --";
arc_toolbar_report[0].childNodes[0].title = "分享人数" + data.stat.share;
arc_toolbar_report[0].childNodes[0].childNodes[1].innerText = deliver.unitFormat(data.stat.share);
arc_toolbar_report[2].title = "收藏人数" + data.stat.favorite;
arc_toolbar_report[2].childNodes[0].childNodes[3].innerText = deliver.unitFormat(data.stat.favorite);
arc_toolbar_report[3].title = "投硬币枚数" + data.stat.coin;
arc_toolbar_report[3].childNodes[0].childNodes[3].innerText = deliver.unitFormat(data.stat.coin);
tag[0].setAttribute("hidden", "hidden");
desc[1].innerText = data.desc;
new unsafeWindow.bbComment(".comment",unsafeWindow.aid, 1, unsafeWindow.UserStatus.userInfo, "");
let bpui = document.getElementsByClassName("bpui-button-text");
let t = setInterval(()=>{
// 更新列表名称
if (bpui[1]) {
clearInterval(t);
bpui[1].firstChild.innerText = "收藏列表";
}
},100);
},
},
// 分集数据
setBangumi : {
init : async (data) => {
if (!config.reset.episodedata) return;
// 判断是否有分集数据
if (data.epList[1] && (data.epList[0].aid != data.epList[1].aid)) {
aid = data.epInfo.aid;
let timer = window.setInterval(() => {
if (document.getElementsByClassName("info-sec-av")[0]) {
deliver.setBangumi.episodeData("first");
window.clearInterval(timer);
}
},1000);
// 延时取消操作,10s还未载入完成将不再处理
window.setTimeout(() => window.clearInterval(timer), 10000);
}
},
// 分集数据处理
episodeData : async (data, msg) => {
try {
let views = document.getElementsByClassName("view-count")[0].getElementsByTagName("span")[0];
let danmakus = document.getElementsByClassName("danmu-count")[0].getElementsByTagName("span")[0];
if (data == "first") {
// 判断是否是首次处理
if (views.innerText == "-" && danmakus.innerText == "-") {
window.setTimeout(() => {deliver.setBangumi.episodeData("first")},100);
return;
}
// 备份总播放数和弹幕数
views.setAttribute("title","总播放数 " + views.innerText);
danmakus.setAttribute("title","总弹幕数 " + danmakus.innerText);
debug.debug("总播放数", views.innerText, " 总弹幕数", danmakus.innerText);
data = await xhr.true(deliver.obj2search(API.url.stat, {"aid" : aid}));
}
if (!data) {
aid = msg.relatedNode.innerText.match(/[0-9]+/)[0];
data = await xhr.true(deliver.obj2search(API.url.stat, {"aid" : aid}));
}
data = deliver.xhrJsonCheck(data).data;
let view = data.view;
let danmaku = data.danmaku;
view = deliver.unitFormat(view);
danmaku = deliver.unitFormat(danmaku);
views.innerText = view;
danmakus.innerText = danmaku;
debug.debug("播放", view + " 弹幕", danmaku);
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("分集数据", ...e)}
},
},
// 跳转完后的播单处理
setPlayList : async () => {
window.onload = () => {
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(API.style.bofqi));
}
},
// 在线数据,接口失效
setOnline : async () => {
let timer = window.setInterval(async () => {
let online = document.getElementsByClassName("online")[0];
if (online) {
// 判断主页载入进程
window.clearInterval(timer);
let loop = async () => {
try {
let data = await xhr.true(API.url.online);
data = deliver.xhrJsonCheck(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 ? "在线人数:" + 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 ? "最新投稿:" + all_count : "最新投稿";
}
else {
let count = online.parentNode.getElementsByTagName("a")[1];
count.text = all_count ? "最新投稿:" + all_count : "最新投稿";
}
if (!all_count || !web_online || !play_online) return;
// 60s刷新一次
window.setTimeout(()=> loop(), 60000);
}
catch(e) {e = Array.isArray(e) ? e : [e] ;debug.error("在线数据", ...e)}
}
loop();
}
},1000);
},
// 注册时间
setJoinTime : async () => {
if (!mid && !config.reset.jointime) return;
let data = await xhr.GM(deliver.obj2search(API.url.membercard,{"mid" : mid}));
try {
data = deliver.xhrJsonCheck(data);
// 格式化时间戳,不是13位,主动补位
let jointime = deliver.timeFormat(data.card.regtime * 1000, 1);
let birthdate = data.card.birthday;
debug.log("注册时间", [data.card.name, jointime]);
document.addEventListener("DOMNodeInserted",(msg) => {
let birthday = document.getElementsByClassName("birthday");
if (birthday[0]) {
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(API.style.jointime));
}
}
});
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("注册时间", ...e)}
},
// 失效视频
fixVideoLost : {
// 收藏里的失效视频
favlist : async (msg, data) => {
// src判定是否为频道并取消重复处理
if (!config.reset.lostvideo || window.src) return;
// 获取av号或者将bv转为av
let title, cover, aid = msg.target.getAttribute("data-aid");
if (!(1 * aid)) aid = deliver.convertId(aid);
if (arr.indexOf(aid) != -1) return;
// 记录已经处理过的视频aid
arr.push(aid);
try {
// 尝试读取来自jijidown的数据
data = await xhr.GM(API.url.jijidown + aid);
data.match('window._INIT')[0];
title = data.match(/\<title\>.+?\-哔哩哔哩唧唧/)[0].replace(/<title>/,"").replace(/-哔哩哔哩唧唧/,"");
cover = data.match(/\"img\":\ \".+?\",/)[0].match(/http.+?\",/)[0].replace(/",/,"");
// 判断封面是否有效
cover.match('hdslb')[0];
}
catch(e) {
try {
// 尝试读取来自biliplus数据
data = await xhr.GM(API.url.biliplus + aid);
data.match(/\<title\>.+?\ \-\ AV/)[0];
title = data.match(/\<title\>.+?\ \-\ AV/)[0].replace(/<title>/,"").replace(/ - AV/,"");
cover = data.match(/\<img style=\"display:none\"\ src=\".+?\"\ alt/)[0].replace(/<img style="display:none" src="/,"").replace(/" alt/,"");
}
catch(e) {
// 无有效数据只能把标题改为av号
title = "av" + aid;
}
}
debug.log("失效视频", "av" + aid);
let img = msg.target.getElementsByTagName("img")[0];
let txt = msg.target.getElementsByClassName("title")[0];
if (cover) img.setAttribute("src",cover + "@380w_240h_100Q_1c.webp");
img.setAttribute("alt",title);
txt.setAttribute("href", "//www.bilibili.com/video/av" + aid);
txt.setAttribute("title",title);
txt.setAttribute("style", "text-decoration : line-through;color : #ff0000;");
txt.text = title;
msg.target.setAttribute("class", "small-item");
msg.target.firstChild.setAttribute("href", "//www.bilibili.com/video/av" + aid);
msg.target.firstChild.setAttribute("target", "_blank");
msg.target.firstChild.setAttribute("class", "cover cover-normal");
},
// 频道里的失效视频
channel : async (link) => {
if (!config.reset.lostvideo || !src) return;
try {
let data, obj = deliver.search2obj(link),
cid = obj.cid,
mid = obj.mid,
pn = obj.pn;
let small_item = document.getElementsByClassName("small-item");
if (small_item[0]) for (let i = 0; i < small_item.length; i++) if (small_item[i].getElementsByClassName("title")[0].text == "已失效视频") src = "";
if (src) return;
data = await xhr.true(deliver.obj2search(API.url.channel, {"mid" : mid, "cid" : cid, "pn" : pn, "ps" : 30, "order" : 0}));
data = deliver.xhrJsonCheck(data).data;
for (let i = 0; i < small_item.length; i++) {
let aid = small_item[i].getAttribute("data-aid") * 1;
let title = "av" + aid;
if (data.list.archives[i].title) title = data.list.archives[i].title;
let a = small_item[i].getElementsByClassName("cover")[0];
let img = small_item[i].getElementsByTagName("img")[0];
let txt = small_item[i].getElementsByClassName("title")[0];
if (txt.text == "已失效视频") {
small_item[i].setAttribute("class", "small-item fakeDanmu-item");
if (aid) {
// 修复失效视频av号
debug.log("失效视频", "av" + aid);
txt.setAttribute("href", "//www.bilibili.com/video/av" + aid);
a.setAttribute("href", "//www.bilibili.com/video/av" + aid);
}
else {
// 修复失效视频bv号
aid = small_item[i].getAttribute("data-aid");
debug.log("失效视频", aid);
txt.setAttribute("href", "//www.bilibili.com/video/" + aid);
a.setAttribute("href", "//www.bilibili.com/video/" + aid);
}
a.setAttribute("target", "_blank");
a.setAttribute("class", "cover cover-normal");
img.setAttribute("alt", title);
img.setAttribute("src", data.list.archives[i].pic.replace("http","https") + "@380w_240h_100Q_1c.webp");
txt.setAttribute("target", "_blank");
txt.setAttribute("title", title);
txt.setAttribute("style", "text-decoration : line-through;color : #ff0000;");
txt.text = title;
}
}
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("失效视频·频道", ...e)}
},
// 空间首页展示的失效视频
home : async (msg) => {
if (!config.reset.lostvideo) return;
let channel_item = document.getElementsByClassName("channel-item");
if (channel_item[0]) {
let small_item = document.getElementsByClassName("small-item");
if (small_item[0]) {
for (let i = 0; i < small_item.length; i++) {
if (small_item[i].getAttribute("class") == "small-item disabled") {
small_item[i].setAttribute("class", "small-item fakeDanmu-item");
let aid = small_item[i].getAttribute("data-aid") * 1;
let a = small_item[i].getElementsByClassName("cover")[0];
let img = small_item[i].getElementsByTagName("img")[0].alt;
let txt = small_item[i].getElementsByClassName("title")[0];
if (aid) {
// 修改失效视频av链接
debug.log("失效视频", "av" + aid);
txt.setAttribute("href", "//www.bilibili.com/video/av" + aid);
a.setAttribute("href", "//www.bilibili.com/video/av" + aid);
}
else {
// 修改失效视频bv链接
aid = small_item[i].getAttribute("data-aid");
debug.log("失效视频", aid);
txt.setAttribute("href", "//www.bilibili.com/video/" + aid);
a.setAttribute("href", "//www.bilibili.com/video/" + aid);
}
a.setAttribute("target", "_blank");
a.setAttribute("class", "cover cover-normal");
txt.setAttribute("target", "_blank");
txt.setAttribute("title", img);
txt.setAttribute("style", "text-decoration : line-through;color : #ff0000;");
txt.text = img;
}
}
}
}
// 固定失效视频数据防止被页面改回去
if (msg.relatedNode.text == '已失效视频') msg.relatedNode.text = msg.relatedNode.getAttribute("title");
if (msg.target.className == "small-item disabled") msg.target.className = "small-item";
}
},
// 评论楼层
setReplyFloor : async (link) => {
src = "";
if (!config.reset.replyfloor) return;
try {
let mode, data, obj = deliver.search2obj(link),
oid = obj.oid,
sort = obj.sort,
pn = obj.pn,
root = obj.root,
type = obj.type;
// sort与mode对应转化,sort == 1时暂时处理不了直接退出
// 热门:sort=2 mode=3 时间:sort=0 mode=2 回复:sort=1 默认(热门+时间): mode=1
if (sort == 0) mode = 1;
if (sort == 1) throw ["暂无法处理按回复排列的评论", obj];
if (sort == 2) mode = 3;
let list_item = document.getElementsByClassName("reply-wrap");
let main_floor = document.getElementsByTagName("li");
// 展开楼中楼的楼层号
if (root) {
// 前两页直接获取
if (pn < 2) data = await xhr.true(deliver.obj2search(API.url.replycursor, {"oid" : oid,"root" : root,"type" : type}));
else {
// 3页以上先获取当页首条评论rpid
let dialog;
if (list_item[0]) {
for (let i = 0; i < list_item.length; i++) {
if (list_item[i].getAttribute("data-id") == root) {
list_item = list_item[i].getElementsByClassName("reply-wrap");
if (list_item[0]) {
for (let j = 0; j < list_item.length; j++) {
if (!list_item[j].getElementsByClassName("floor")[0]) {
dialog = list_item[j].getAttribute("data-id");
break;
}
}
}
break;
}
}
}
else if (main_floor[0]) {
for (let i = 0; i < main_floor.length; i++) {
if (main_floor[i].getAttribute("id") && main_floor[i].getAttribute("id").includes(root)) {
main_floor = main_floor[i].getElementsByTagName("li");
if (main_floor[0]) {
for (let j = 0; j < main_floor.length; j++) {
if (main_floor[j].id && main_floor[j].id.includes("l_id") && !main_floor[j].getElementsByClassName("floor-num")[0]) {
dialog = main_floor[j].getAttribute("id").split('_')[2];
break;
}
}
}
break;
}
}
}
// 根据当页首条评论rpid获取min_id
data = await xhr.true(deliver.obj2search(API.url.replydialog, {"oid" : oid,"root" : root,"type" : type, "dialog" : dialog, "size" : 20}));
let min_id = deliver.xhrJsonCheck(data).data.replies;
if (min_id) {for (let i = 0; i < min_id.length; i++) if (min_id[i].rpid == dialog) {min_id = min_id[i].floor; break;}}
else {debug.msg("当前页楼中楼层获取失败 ಥ_ಥ"); return;}
// 根据min_id获取当页数据
data = await xhr.true(deliver.obj2search(API.url.replycursor, {"oid" : oid,"root" : root,"type" : type, "min_id" : min_id}));
}
}
else {
if (sort == 2) data = await xhr.true(deliver.obj2search(API.url.replymain, {"oid" : oid,"next" : pn,"type" : type,"mode" : mode}));
else if (pn == 1) data = await xhr.true(deliver.obj2search(API.url.replymain, {"oid" : oid,"type" : type,"mode" : mode}));
else {
// 时间排序的楼层号需要相对前页判定
pn = pn - 1;
data = await xhr.true(deliver.obj2search(API.url.reply, {"type" : type,"sort" : sort,"oid" : oid,"pn" : pn}));
data = deliver.xhrJsonCheck(data).data;
let i = data.replies.length - 1;
oid = data.replies[0].oid;
let root = data.replies[i].rpid;
data = await xhr.true(deliver.obj2search(API.url.replycursor, {"oid" : oid,"root" : root,"type" : type}));
data = deliver.xhrJsonCheck(data).data;
oid = data.root.oid;
let next = data.root.floor;
data = await xhr.true(deliver.obj2search(API.url.replymain, {"oid" : oid,"next" : next,"type" : type,"mode" : mode}));
}
}
data = deliver.xhrJsonCheck(data).data;
let floor = {}, top = data.top, hots = data.hots, replies = data.replies, froot = data.root;
if (hots && hots[0]) {
for (let i = 0; i < hots.length; i++) {
floor[hots[i].rpid] = hots[i].floor;
if (hots[i].replies) for (let j = 0; j < hots[i].replies.length; j++) floor[hots[i].replies[j].rpid] = hots[i].replies[j].floor;
}
}
if (replies && replies[0]) {
for (let i = 0;i < replies.length; i++) {
floor[replies[i].rpid] = replies[i].floor;
if (replies[i].replies) for (let j = 0; j < replies[i].replies.length; j++) floor[replies[i].replies[j].rpid] = replies[i].replies[j].floor;
}
}
if (top) {
for (let key in top) {
if (top[key]) {
floor[top[key].rpid] = top[key].floor;
if (top[key].replies) for (let i = 0; i < top[key].replies.length; i++) floor[top[key].replies[i].rpid] = top[key].replies[i].floor;
}
}
}
if (froot && froot.replies) for (let i = 0; i < froot.replies.length; i++) floor[froot.replies[i].rpid] = froot.replies[i].floor;
// 旧版评论直接写入楼层号
if (main_floor[0]) {
for (let i = 0; i < main_floor.length; i++) {
if (main_floor[i].id && main_floor[i].id.includes("l_id")) {
let rpid = main_floor[i].getAttribute("id").split('_')[2];
if (rpid in floor) {
try {
main_floor[i].getElementsByClassName("floor-num")[0].innerText = "#" + floor[rpid];
}
catch (e) {
let node = main_floor[i].getElementsByClassName("floor-date")[0].parentNode;
let span = document.createElement("span");
span.setAttribute("class", "floor-num");
span.setAttribute("style", "float : left;color : #aaa;padding-right : 10px;");
span.innerText = "#" + floor[rpid];
node.insertBefore(span,node.firstChild);
}
}
}
}
}
// 新版评论需另外创建楼层号
if (list_item[0]) {
for (let i = 0; i<list_item.length; i++) {
let rpid = list_item[i].getAttribute("data-id");
if (rpid in floor) {
let node = list_item[i].getElementsByClassName("info")[0];
let span = document.createElement("span");
span.setAttribute("class", "floor");
span.innerText = "#" + floor[rpid];
node.insertBefore(span,node.firstChild);
}
}
}
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("评论楼层", ...e)}
},
//修复评论”跳转指定时间“功能
fixVideoSeek: function(node) {
node.querySelectorAll("a.video-seek").forEach(function (v) {
v.addEventListener("click", function (e) {
window.scrollTo(0,170);
unsafeWindow.player.seek(Number(e.target.attributes[2].nodeValue));
});
});
},
// 广告区转资讯区
fixnews : async (node, move) => {
try {
let rank = config.reset.grobalboard ? document.getElementsByClassName("rank-tab")[0] : "";
if (node.id == "bili_ad") {
let sight = node.getElementsByTagName("a");
node = node.getElementsByClassName("name");
if (node[0]) node[0].text = "资讯";
for (let i = 0; i < sight.length; i++) if (sight[i].href.includes("www.bilibili.com/v/ad/ad/")) sight[i].href = "https://www.bilibili.com/v/information/";
let rcon = document.createElement("div");
rcon.setAttribute("class", "r-con");
rcon.innerHTML = '<div class="r-con"><header style="margin-bottom: 14px"><h3 style="font-size: 18px;font-weight: 400;">资讯分区正式上线啦!</h3></header><div class="carousel-module"><div class="panel"><a href="https://www.bilibili.com/v/information" target="_blank"><img src="//i0.hdslb.com/bfs/archive/0747d26dbbc3bbf087d47cff49e598a326b0030c.jpg@320w_330h_1c.webp" width="260" height="280"/></a></div></div></div>';
document.getElementById("ranking_ad").replaceWith(rcon);
}
if (node.className == "report-wrap-module elevator-module") for (let item of node.children[1].children) if (item.innerHTML == "广告") item.innerHTML = "资讯";
if (node.id == "bili-header-m") {
node = node.getElementsByClassName('nav-name');
if (node[0]) {
for (let i = 0; i < node.length; i++) {
if (node[i].textContent == "科技") {
move = node[i].parentNode.parentNode.children[1].lastChild.cloneNode(true);
move.firstChild.href = move.firstChild.href.replace("technology", "life");
node[i].parentNode.parentNode.children[1].lastChild.remove();
}
if (node[i].textContent == "广告") {
node[i].textContent = "资讯";
node[i].parentNode.href = "//www.bilibili.com/v/information/";
}
if (node[i].textContent == "生活") {
let sight = node[i].parentNode.parentNode.children[1];
sight.insertBefore(move, sight.lastChild)
}
if (node[i].textContent == "娱乐") node[i].parentNode.parentNode.children[1].lastChild.remove();
}
}
}
if (rank && rank.children[5]) {
rank.children[5].innerText == "知识" ? rank.children[5].innerText = "科技" : "";
rank.children[6].innerText == "知识" ? rank.children[6].innerText = "科技" : "";
}
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("分区·版面", ...e)}
},
// 跳过充电鸣谢
electricPanelJump : async (node) => {
try {
if (!config.reset.electric) return;
config.reset.electric = 0;
setTimeout(() => {node.click()}, 1);
setTimeout(() => {config.reset.electric = 1}, 5000);
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("充电鸣谢", ...e)}
},
// 修复分区排行
fixrank : async (node) => {
// 这些分区排行榜已全部采用类似番剧排行的模式,故采用相似的节点覆盖
let sort = {
bili_movie : ["ranking_movie", 2, "https://www.bilibili.com/ranking/cinema/23/0/3"],
bili_teleplay : ["ranking_teleplay", 5, "https://www.bilibili.com/ranking/cinema/11/0/3"],
bili_documentary : ["ranking_documentary", 3, "https://www.bilibili.com/ranking/cinema/177/0/3"]
}
sort = sort[node.id];
if (!sort) return;
let section = node.getElementsByClassName("sec-rank report-wrap-module zone-rank")[0];
section.innerHTML = '<header class="rank-head"><h3>排行</h3></header><div class="rank-list-wrap"><ul class="bangumi-rank-list rank-list"></ul></div><a href="' + sort[2] + '" target="_blank" class="more-link">查看更多<i class="icon icon-arrow-r"></i></a>';
try {
let data = await xhr.true(deliver.obj2search(API.url.ranklist, {season_type : sort[1], day : 3}));
data = deliver.xhrJsonCheck(data).data;
node = node.getElementsByClassName("bangumi-rank-list rank-list")[0];
for (let i = 0; i < 8; i++) {
let li = document.createElement("li"),
cl = i < 3 ? "rank-item highlight" : "rank-item",
fw;
li.setAttribute("class", cl);
li.innerHTML = '<i class="ri-num">' + (i + 1) + '</i><a href="'+ data.list[i].url + '" target="_blank" title="' + data.list[i].title + ' 播放:' + data.list[i].stat.view + '" class="ri-info-wrap"><p class="ri-title">' + data.list[i].title + '</p><span class="ri-total">' + data.list[i].new_ep.index_show + '</span></a>';
li.onmouseover = () => {
fw = document.createElement("div");
fw.setAttribute("class", "bangumi-info-module");
fw.setAttribute("style", 'left: ' + li.getBoundingClientRect().left + 'px; top: ' + (getTotalTop(li) - 150) + 'px;');
fw.innerHTML = '<div class="v-preview clearfix"><div class="lazy-img cover"><img alt="' + data.list[i].title + '" src="' + data.list[i].cover + '" /></div><div><p class="title">' + data.list[i].title + '</p><p class="desc">' + data.list[i].new_ep.index_show + '</p></div></div><div class="v-data"><span class="play"><i class="icon"></i>' + deliver.unitFormat(data.list[i].stat.view) + '</span><span class="danmu"><i class="icon"></i>' + deliver.unitFormat(data.list[i].stat.danmaku) + '</span><span class="fav"><i class="icon"></i>' + deliver.unitFormat(data.list[i].stat.follow) + '</span></div>';
document.body.appendChild(fw);
}
li.onmouseout = () => fw.remove();
node.appendChild(li);
}
// 计算节点相对高度
function getTotalTop(node){
var sum = 0;
do{
sum += node.offsetTop;
node = node.offsetParent;
}
while(node);
return sum;
}
}
catch (e) {e = Array.isArray(e) ? e : [e]; debug.error("分区排行", ...e)}
},
// 弹幕哈希反查
danmkuHashId : async (node) => {
if (!config.reset.midcrc || !window.midcrc) return;
let index = 1 * node.getAttribute("dmno");
node.addEventListener("contextmenu", () => {
setTimeout(async (data)=>{
try {
let descipline = document.createElement("li");
let onwer = document.createElement("li");
let mid = window.midcrc(hash[index]);
node = document.getElementsByClassName("bili-old-hash");
if (node[0]) for (let i = 0; i < node.length; i++ ) node[i].remove();
if (document.getElementsByClassName("bilibili-player-icon bilibili-player-icon-arrow-down")[0]) return;
if (document.getElementsByClassName("bilibili-player-icon bilibili-player-icon-arrow-up")[0]) return;
descipline.setAttribute("class", "context-line context-menu-descipline bili-old-hash");
descipline.innerHTML = '<a class="context-menu-a" href="javascript:void(0);"></a>';
onwer.setAttribute("class", "context-line context-menu-function bili-old-hash");
onwer.innerHTML = '<a class="context-menu-a js-action" title="" href="//space.bilibili.com/' + mid + '">hash: ' + hash[index] + " mid: " + mid + '</a>';
node = document.getElementsByClassName("bilibili-player-context-menu-container")[0];
node.firstChild.insertBefore(descipline, node.firstChild.firstChild);
onwer = node.firstChild.insertBefore(onwer, node.firstChild.firstChild);
data = deliver.xhrJsonCheck(await xhr.true(deliver.obj2search(API.url.card, {mid : mid})));
onwer.innerHTML = '<div style="min-height:0px;z-index:-5;" class="bb-comment"><div style="padding-top:10px;" class="comment-list"><div class="list-item"><div class="reply-box"><div style="padding:0px" class="reply-item reply-wrap"><div style="margin-left: 15px;" class="reply-face"><img src="' +
data.data.card.face + '@52w_52h.webp" alt=""></div><div class="reply-con"><div class="user"><a style="display:initial;padding: 0px;" data-usercard-mid="' +
mid + '" href="//space.bilibili.com/' +
mid + '" target="_blank" class="' +
(data.data.card.vip.vipType > 1 ? "name vip-red-name" : "name") + '">' + data.data.card.name + '</a> ' +
data.data.card.sex + '<a style="display:initial;padding: 0px;" href="//www.bilibili.com/blackboard/help.html#%E4%BC%9A%E5%91%98%E7%AD%89%E7%BA%A7%E7%9B%B8%E5%85%B3" target="_blank"><i class="level l' +
data.data.card.level_info.current_level + '"></i></a></div></div></div></div></div></div></div>';
}
catch (e) {e = Array.isArray(e) ? e : [e]; debug.error("弹幕反查", ...e)}
})
})
},
// 会员授权
accesskey : async () => {
if (!config.reset.accesskey) {
if (GM_getValue("access_key")) {
GM_setValue("access_key", "");
GM_setValue("access_date", "");
}
return;
}
if (!GM_getValue("access_key") || (Date.now() - GM_getValue("access_date") > 2160000)) {
try{
if (!uid) return;
let data = deliver.xhrJsonCheck(await xhr.GM("https://passport.bilibili.com/login/app/third?appkey=27eb53fc9058f8c3&api=https%3A%2F%2Fwww.mcbbs.net%2Ftemplate%2Fmcbbs%2Fimage%2Fspecial_photo_bg.png&sign=04224646d1fea004e79606d3b038c84a"));
data = await new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method : "GET",
url : data.data.confirm_uri,
onload : (xhr) => resolve(xhr.finalUrl),
onerror : (xhr) => reject(xhr.statusText || url + " net::ERR_CONNECTION_TIMED_OUT"),
});
})
data = deliver.search2obj(data);
let page = document.createElement("iframe");
page.setAttribute("style", "display: none;");
page.setAttribute("src", deliver.obj2search("https://www.biliplus.com/login", data));
document.body.appendChild(page);
setTimeout(() => {page.remove()},3000);
GM_setValue("access_key", data.access_key);
GM_setValue("access_date", Date.now());
}
catch (e) {e = Array.isArray(e) ? e : [e]; debug.error("登录鉴权", ...e)}
}
}
}
// 设置界面
const UI = {
// 设置入口
init : async () => {
let ui_face = document.createElement("div");
let enter = document.createElement("span");
let icon = document.createElement("i");
ui_face.setAttribute("class", "bili-old ui-face");
ui_face.setAttribute("id", "ui-face");
ui_face.setAttribute("style", "right : -54px;");
ui_face.onmouseover = () => ui_face.setAttribute("style", "right : 0px;box-shadow : rgba(0, 85, 255, 0.098) 0px 0px 20px 0px;border : 1px solid rgb(233, 234, 236);");
ui_face.onmouseout = () => ui_face.setAttribute("style", "right : -54px;");
ui_face.onclick = () => {
let table = document.getElementsByClassName("ui-table");
if (!table[0]) UI.table();
else if (table[0].getAttribute("hidden")) table[0].removeAttribute("hidden");
}
ui_face.appendChild(icon);
ui_face.appendChild(enter);
enter.innerText = "设置";
let timer = window.setInterval(() => {
// 等待body载入再进行设置绘制
if (document.body) {
window.clearInterval(timer);
document.body.appendChild(ui_face);
}
}, 1000);
},
// 设置面板
table : async () => {
let table = document.getElementsByClassName("ui-table")[0];
let timer;
if (!table) {
table = document.createElement("div");
table.setAttribute("class", "bili-old ui-table");
table.setAttribute("id", "ui-table");
let info = document.createElement("span");
let rec = document.createElement("span");
info.setAttribute("style", "color : rgb(0,0,0);font-size : 14px;");
info.innerText = "BilibiliOld 设置";
table.appendChild(info);
rec.setAttribute("style", "color : blue;float : right;font-size : 12px;");
rec.innerText = "恢复默认";
rec.onclick = () => {
for (let key in defig.rewrite) if (key in config.rewrite) config.rewrite[key] = defig.rewrite[key];
for (let key in defig.reset) if (key in config.reset) config.reset[key] = defig.reset[key];
GM_setValue("config",config);
debug.msg("恢复默认设置");
table.remove();
}
table.appendChild(rec);
for (let key in config.rewrite) UI.setTable(table, UI.menu[key], config.rewrite[key], key);
for (let key in config.reset) UI.setTable(table, UI.menu[key], config.reset[key], key);
document.body.appendChild(table);
}
// 设置失去焦点时消失时间
table.onmouseover = () => window.clearTimeout(timer);
table.onmouseout = () => {
timer = window.setTimeout(() => {
table.setAttribute("hidden", "hidden");
GM_setValue("config", config);
debug.msg("设置已保存");
}, 500);
}
},
// 设置选项
setTable : async (ele, name, check, key) => {
let div = document.createElement("div");
let span = document.createElement("span");
let input = document.createElement("input");
ele.appendChild(div);
div.setAttribute("style", "padding : 4px 4px 0px 4px;clear : both;");
if (document.getElementsByClassName("checke")[0]) div.setAttribute("style", "padding : 0px 4px 0px 4px;clear : both;");
div.appendChild(span);
div.appendChild(input);
span.setAttribute("style", "float : left;display : inline-block;color : rgb(0,0,0);font-size : 14px;");
span.innerText = name[0];
input.setAttribute("type", "checkbox");
input.setAttribute("class", "checke");
if (check) input.checked = true;
input.onclick = () => {
// 开关响应
if (input.checked) {
if (key in config.rewrite) config.rewrite[key] = 1;
else config.reset[key] = 1;
if (!config.reset.xhrhook && key != "xhrhook" && UI.menu[key][1].includes("xhrhook")) debug.msg("启用失败!xhrhook已关闭!", UI.menu[key][0])
}
else {
if (key in config.rewrite) config.rewrite[key] = 0;
else config.reset[key] = 0;
if (key == "xhrhook") debug.msg("xhrhook已关闭,部分功能无法生效!")
}
}
// 鼠标移出隐藏并保存设置
div.onmouseover = () => {
let div = document.createElement("div");
div.setAttribute("class","bili-old ui-state");
div.setAttribute("id","ui-state");
div.innerHTML = name[1];
document.body.appendChild(div);
}
div.onmouseout = () => document.getElementById("ui-state") ? document.getElementById("ui-state").remove() : "";
},
// 设置内容及说明,基本与config一一对应
menu : {
av : ["av(BV)", "启用旧版av页面,基于旧版网页框架"],
bangumi : ["Bangumi", "启用旧版番剧页面,基于旧版网页框架"],
watchlater : ["稍后再看", "启用旧版稍后再看页面,基于旧版网页框架"],
frame : ["嵌入", "替换嵌入式播放器,不会单独适配被嵌入页面的其他功能"],
home : ["主页", "启用旧版主页,,基于旧版网页框架,旧版主页失效内容过多,已进行一定程度处理满足日常使用"],
playlist : ["播单", "恢复播单页,基于旧版网页框架"],
medialist : ["收藏", "模拟收藏列表播放页面,收藏播放页是新版专属页面,只能先跳转av页再模拟收藏列表<br>依赖旧版av页<br>切P时up主简介等少数信息不会另外请求<br>※播放列表视频太多将导致视频载入及切换速度变慢"],
rank : ["排行", "启用旧版排行,基于旧版网页框架"],
danmuku : ["新版弹幕", "尝试换用新版弹幕接口,弹幕上限将变为两倍,对加载速度影响不明显<br>※依赖WebWorker hook"],
livechat : ["实时弹幕", "尝试修复实时弹幕聊天功能,使旧播放器能继续实时接收最新弹幕<br>※依赖WebSocket hook"],
limit : ["区域限制", "尝试解除B站区域限制(包括部分仅限APP限制),用于观看港澳台番剧<br>※只适配旧版播放器<br>※功能不及专门的脚本,同时使用请关闭本选项<br>※依赖xhrhook<br>※参看“会员授权”"],
accesskey : ["会员授权", "“区域限制”的高级功能,大会员的用户可以授权使用B站账户登录代理服务器,以观看会员专享视频。<br>※使用B站官方授权接口,完全不涉及密码等隐私信息,敬请放心<br>※不能解除大会员限制,只是让本身是大会员的用户能在区域限制视频继承大会员身份<br>※非大会员或者不需要观看大会员区域限制番剧开启本功能毫无意义"],
grobalboard : ["顶栏底栏", "识别并替换所有新版顶栏为旧版顶栏,旧版失效广告区替换为资讯区"],
replyfloor : ["评论楼层", "恢复评论区楼层号,上古“按评论数”排列的评论除外<br>添加了楼中楼层号显示,但若楼中楼当页第一条评论是回复别人则该页都无法获取"],
headblur : ["顶栏透明", "使旧版顶栏全透明"],
preview : ["付费预览", "去除播放器左下角付费预览框"],
jointime : ["注册时间", "在个人空间显示B站账号注册时间,依赖主人开放个人资料"],
lostvideo : ["失效视频", "借助第三方接口修复失效视频的封面和标题,将标题标红并添加删除线,无数据时只修改标题为av号"],
bvid2av : ["BV⇒av", "让所有页面能使用av号的地方尽量使用av号(未能完全覆盖)<br>进入bv页面自动跳转到av页面(不会重载页面)"],
selectdanmu : ["弹幕优先", "让旧版播放器优先展示弹幕列表而不是推荐视频"],
episodedata : ["分集数据", "让番剧显示分集的播放数和弹幕数,原来总计数据显示在鼠标焦点的浮动信息上"],
like : ["点赞功能", "为旧版播放页面添加点赞功能,点赞是新版页面专属功能,功能简陋,不支持一键三联"],
static : ["静态页面", "将静态页面跳转到普通页面以启用旧版页面,静态页面是新版新增页面,页面大部分信息都内置于页面中以加快载入速度"],
download : ["下载视频", "播放器右键菜单>>>下载视频>>>选择文件>>>右键另存为/右键IDM下载<br>!!!复制无效/左键点击无效!!!<br>※详见脚本简介"],
heartbeat : ["视频心跳", "替换被其他广告屏蔽插件拦截的视频心跳,若出现播放视频但不记录历史的情况可以尝试启用"],
carousel : ["播放信息", "填充旧版播放器顶部缺失的通知信息"],
adloc : ["主页广告", "去除旧版主页直接写在网页里的广告的内容,如滚动图、推荐位、横幅……"],
roomplay : ["直播拦截", "拦截直播视频及轮播视频以节约流量<br>受浏览器缓存影响注入没有载入直播快则会失败,此种情况硬刷新可以解决"],
history : ["视频历史", "去掉历史记录页面的直播、专栏,只显示视频播放历史"],
xhrhook : ["xhrhook", "hook xhr的send方法,副作用是所有xhr的initiator都会变成本脚本,强迫症可以选择关闭除非需要启用以下功能:<br>※新版弹幕<br>※区域限制"],
electric : ["充电鸣谢", "自动跳过充电鸣谢<br>※动作再快还是会一闪而过"],
panel : ["最后一帧", "使视频播放结束后画面停留在最后一帧,不再展示功能窗口"],
midcrc : ["弹幕反查" , "在旧版播放器弹幕列表上右键将显示发送者信息,鼠标移动到发送者名字上展示详细信息页<br>※原理是通过crc哈希值暴力逆推出mid,再通过mid获取发送者信息,由于哈希函数特性二者不一定一一对应,所以结果仅供参考<br>※不支持嵌入式旧版播放器<br>※修改弹幕排序后由于无法获取哈希值故无法查询<br>※出错时说明逆推出的mid不正确,但不出错也不代表一定正确"],
viewbofqi : ["播放居中", "自动滚动到播放器,使播放器位于网页可视区域正中"],
widescreen : ["自动宽屏", "默认启用网页宽屏"],
danmakuoff : ["关闭弹幕", "默认关闭弹幕,开启后切p也会主动关闭弹幕"]
}
}
// 页面分离
const thread = {
// av/BV
video : () => {
try {
// 判断是否收藏跳转而来
ml = GM_getValue("medialist");
GM_setValue("medialist", 0);
// bv转av
if (config.reset.bvid2av && LOCATION[4].toLowerCase().startsWith('bv')) {
aid = deliver.convertId(LOCATION[4]);
history.replaceState(null, null, "https://www.bilibili.com/video/av" + aid + location.search + location.hash);
}
if (!config.rewrite.av) throw ["未启用旧版av页", location.href];
aid = aid || LOCATION[4].match(/[0-9]+/)[0];
DOCUMENT = xhr.false(deliver.obj2search(API.url.detail, {aid : aid}));
__INITIAL_STATE__ = INITIAL_STATE.av(DOCUMENT);
if (!__INITIAL_STATE__) throw "av/BV号可能无效!";
if (__INITIAL_STATE__.videoData.redirect_url) throw ["番剧重定向:", __INITIAL_STATE__.videoData.redirect_url];
if (__INITIAL_STATE__.videoData.stein_guide_cid) throw ["忽略互动视频:", "av" + aid];
// 写入全局变量
aid = __INITIAL_STATE__.aid ? __INITIAL_STATE__.aid : aid;
tid = __INITIAL_STATE__.videoData.tid ? __INITIAL_STATE__.videoData.tid : tid;
unsafeWindow.__INITIAL_STATE__ = __INITIAL_STATE__;
// 重写网页框架并进行调用后续处理
deliver.write(API.pageframe.video);
document.title = __INITIAL_STATE__.videoData.title + "_哔哩哔哩 (゜-゜)つロ 干杯~-bilibili";
deliver.fixSort.video()
deliver.setLike();
deliver.setMediaList.init();
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("框架·av/BV", ...e)}
},
// 稍后再看
watchlater : () => {
try {
if (!config.rewrite.watchlater) throw ["未启用旧版稍后再看", location.href];
if (!uid) throw ["未登录", "无法启用旧版稍后再看"];
// 重写网页框架并调用后续处理
deliver.write(API.pageframe.watchlater);
deliver.setLike();
deliver.fixSort.watchlater();
// bv转av
if (LOCATION[5]) {
aid = LOCATION[5].match(/[0-9]+/) ? LOCATION[5].match(/[0-9]+/)[0] : aid;
if (LOCATION[5].toLowerCase().startsWith('bv')){
aid = deliver.convertId(LOCATION[5]);
LOCATION[5] = "av" + aid;
history.replaceState(null,null,LOCATION.join("/"));
}
}
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("框架·稍后再看", ...e)}
},
// 番剧
bangumi : () => {
try {
if (!config.rewrite.bangumi) throw ["未启用旧版Bangumi", location.href];
// 指定playurl类型
pgc = true;
// 获取网页源代码
DOCUMENT = xhr.false(location.href);
__INITIAL_STATE__ = DOCUMENT.includes("__INITIAL_STATE__=") ? JSON.parse(DOCUMENT.match(/INITIAL_STATE__=.+?\;\(function/)[0].replace(/INITIAL_STATE__=/, "").replace(/;\(function/, "")) : ""; // 继承__INITIAL_STATE__
// 判断页面是否404
if (!__INITIAL_STATE__){
// 尝试获取备用数据源
if (LOCATION[5].startsWith('ss')) DOCUMENT = xhr.false(deliver.obj2search(API.url.season, {season_id : location.href.match(/[0-9]+/)[0]}));
else if (LOCATION[5].startsWith('ep')) DOCUMENT = xhr.false(deliver.obj2search(API.url.season, {ep_id : location.href.match(/[0-9]+/)[0]}));
}
let id = LOCATION[5].startsWith('ep') ? location.href.match(/[0-9]+/)[0] : "";
// 获取__INITIAL_STATE__
__INITIAL_STATE__ = INITIAL_STATE.bangumi(id);
if (__INITIAL_STATE__ && __INITIAL_STATE__.epInfo && __INITIAL_STATE__.epInfo.badge === "互动") throw ["忽略互动视频:", location.href];
// 重写网页框架并调用后续处理,按是否有特殊背景分别处理
unsafeWindow.__INITIAL_STATE__ = __INITIAL_STATE__;
if (DOCUMENT.match('"specialCover":""') || !__INITIAL_STATE__.special) deliver.write(API.pageframe.bangumi); else deliver.write(API.pageframe.cinema);
document.title = DOCUMENT.match(/<title.*?>.+?<\/title>/) ? DOCUMENT.match(/<title.*?>.+?<\/title>/)[0].replace(/<title.*?>/, "").replace(/<\/title>/, "") : __INITIAL_STATE__.mediaInfo.title;
if (__INITIAL_STATE__) deliver.setBangumi.init(__INITIAL_STATE__);
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("框架·Bangumi", ...e)}
},
// 嵌入
blackboard : () => {
try {
// 修复HTML5播放器帮助页视频cid错误
if (LOCATION[4].startsWith('html5player')) if (LOCATION[4].includes("3521416") && LOCATION[4].includes("6041635")) location.replace(deliver.obj2search(API.playerframe.html5player,{"aid":3521416,"cid":192446449}));
if (!config.rewrite.frame) throw ["未启用旧版嵌入播放器", location.href];
if (LOCATION[4].startsWith('newplayer')) {
let obj = deliver.search2obj(location.href),
season_type = obj.season_type || "",
player_type = obj.player_type || "";
aid = 1 * obj.aid || (obj.aid ? deliver.convertId(obj.aid) : undefined) || (obj.bvid ? deliver.convertId(obj.bvid) : undefined);
cid = obj.cid || "";
try {
cid = cid || deliver.xhrJsonCheck(xhr.false(deliver.obj2search(API.url.pagelist,{"aid" : aid}))).data[0].cid
}
catch (e) {e = Array.isArray(e) ? e : [e]; debug.error("框架·嵌入", ...e)}
// 重定向到旧版播放器
location.replace(deliver.obj2search(API.playerframe.html5player,{"aid" : aid,"cid" : cid,"season_type" : season_type,"player_type" : player_type,"as_wide" : 1,}));
debug.log("嵌入播放器", "aid=", aid, " cid=", cid);
}
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("框架·嵌入", ...e)}
},
// 播单
playlist : () => {
try {
if (!config.rewrite.playlist) return;
if (LOCATION[4] == "video") {
deliver.write(API.pageframe.playlist);
deliver.setPlayList();
}
if (LOCATION[4] == "detail") {
__INITIAL_STATE__ = {mid : "", pid : "", plinfoData : {}, pllistData : {}}
DOCUMENT = deliver.xhrJsonCheck(xhr.false(deliver.obj2search(API.url.playlist, {pid : LOCATION[5].match(/[0-9]+/)[0]}))).data;
__INITIAL_STATE__.mid = DOCUMENT.mid;
__INITIAL_STATE__.pid = DOCUMENT.pid;
__INITIAL_STATE__.plinfoData = {attr : DOCUMENT.attr, count : DOCUMENT.count, cover : DOCUMENT.cover, ctime : DOCUMENT.ctime, description : DOCUMENT.description, favored : DOCUMENT.favored, id : DOCUMENT.id, is_favorite : DOCUMENT.is_favorite, mid : DOCUMENT.mid, mtime : DOCUMENT.mtime, owner : DOCUMENT.owner, pid : DOCUMENT.pid, stat : DOCUMENT.stat, state : DOCUMENT.state, type : DOCUMENT.type,};
__INITIAL_STATE__.pllistData = DOCUMENT.list;
unsafeWindow.__INITIAL_STATE__ = __INITIAL_STATE__;
deliver.write(API.pageframe.detail);
}
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("播单", ...e)}
},
// 收藏
medialist : () => {
if (LOCATION[5].startsWith("ml")) {
ml = 1 * LOCATION[5].match(/[0-9]+/)[0];
// 保存收藏号并调用av跳转
if (!config.rewrite.medialist) return;
GM_setValue("medialist", ml);
deliver.setMediaList.init(ml);
}
// 新版稍后再看跳转到旧版稍后再看
if (LOCATION[5].startsWith("watchlater") && config.rewrite.watchlater) location.replace("https://www.bilibili.com/watchlater/#/");
},
// 静态av
svideo : () => {
// 直接跳转回普通av
if (!config.reset.static) return;
location.replace(location.href.replace("s/video", "video"));
},
// 空间
space : () => {
// 调用注册时间处理
mid = LOCATION[3] ? 1 * LOCATION[3] : mid;
deliver.setJoinTime();
},
// 主页
home : () => {
try {
if (!config.rewrite.home) throw ["未启用旧版主页", location.href];
window.recbtn = 1;
if (!unsafeWindow.__INITIAL_STATE__) {
DOCUMENT = xhr.false(location.href);
__INITIAL_STATE__ = DOCUMENT.includes("__INITIAL_STATE__=") ? DOCUMENT.match(/INITIAL_STATE__=.+?\;\(function/)[0].replace(/INITIAL_STATE__=/, "").replace(/;\(function/, "") : ""; // 继承__INITIAL_STATE__
}
else __INITIAL_STATE__ = JSON.stringify(unsafeWindow.__INITIAL_STATE__);
// 新旧__INITIAL_STATE__不兼容,进行重构
unsafeWindow.__INITIAL_STATE__ = __INITIAL_STATE__ = INITIAL_STATE.home(__INITIAL_STATE__);
// 重写网页框架
deliver.write(API.pageframe.home);
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("框架·主页", ...e)}
// 调用在线数据处理
deliver.setOnline();
},
// 排行榜
rank : () => {
try {
if (!config.rewrite.rank) throw ["未启用排行", location.href];
let refer = document.referrer.split("/");
if (refer && refer[4] && refer[4] == "all") DOCUMENT = deliver.xhrJsonCheck(xhr.false(deliver.obj2search(API.url.ranking, {rid : refer[5], day : 3, type : 1, arc_type : 0})));
else DOCUMENT = deliver.xhrJsonCheck(xhr.false(deliver.obj2search(API.url.ranking, {rid : 0, day : 3, type : 1, arc_type : 0})));
__INITIAL_STATE__ = {loading : false, rankRouteParams : {arc_type:0,day:3,rankTab:"all",rid: 1 * refer[5] || 0,season_type:1}, showTypes: true, times : [{name:"日排行",value:1},{name:"三日排行",value:3},{name:"周排行",value:7},{name:"月排行",value:30}], typeList : [{name:"全部投稿",value:0},{name:"近期投稿",value:1}]};
__INITIAL_STATE__.channels = [{name:"全站",tid:0},{name:"动画",tid:1},{name:"国创相关",tid:168},{name:"音乐",tid:3},{name:"舞蹈",tid:129},{name:"游戏",tid:4},{name:"科技",tid:36},{name:"数码",tid:188},{name:"生活",tid:160},{name:"美食",tid:211},{name:"鬼畜",tid:119},{name:"时尚",tid:155},{name:"娱乐",tid:5},{name:"影视",tid:181}];
__INITIAL_STATE__.rankList = DOCUMENT.data.list;
__INITIAL_STATE__.note = DOCUMENT.data.note;
unsafeWindow.__INITIAL_STATE__ = __INITIAL_STATE__;
deliver.write(API.pageframe.rank);
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("框架·排行", ...e)}
}
}
// 初始化设置
defig = JSON.parse(JSON.stringify(config));
let data = GM_getValue("config");
if (data) {
// 读取脚本管理器中的修改过的设置
for (let key in data.rewrite) if (key in config.rewrite) config.rewrite[key] = data.rewrite[key];
for (let key in data.reset) if (key in config.reset) config.reset[key] = data.reset[key];
}
else GM_setValue("config",config);
try {
// 监听拦截全局变量
deliver.getVariable();
// uid用于判断是否登录
uid = deliver.getCookies().DedeUserID;
// 维护旧版播放器设置
let bilibili_player_settings = localStorage.getItem("bilibili_player_settings");
if (bilibili_player_settings) {
bilibili_player_settings = JSON.parse(bilibili_player_settings);
if (bilibili_player_settings.video_status.autopart !== "") GM_setValue("bilibili_player_settings", bilibili_player_settings);
else if (GM_getValue("bilibili_player_settings")) localStorage.setItem("bilibili_player_settings", JSON.stringify(GM_getValue("bilibili_player_settings")));
}
else if (LOCATION[2] == 'www.bilibili.com' && GM_getValue("bilibili_player_settings")) localStorage.setItem("bilibili_player_settings", JSON.stringify(GM_getValue("bilibili_player_settings")));
// 维护旧版动态状态
if (uid) {
let offset = deliver.getCookies()["bp_video_offset_"+ uid];
if (offset) document.cookie = "bp_t_offset_" + uid + "=" + offset + "; domain=bilibili.com; expires=Aug, 18 Dec 2038 18:00:00 GMT; path=/";
}
}
catch(e) {e = Array.isArray(e) ? e : [e]; debug.error("初始化", ...e)}
// 分离页面单独调用
if (LOCATION[3]) {
if (LOCATION[3] == 'video' && (LOCATION[4].toLowerCase().startsWith('av') || LOCATION[4].toLowerCase().startsWith('bv'))) thread.video();
if (LOCATION[3] == 'watchlater') thread.watchlater();
if (LOCATION[3] == 'bangumi' && LOCATION[4] == 'play') thread.bangumi();
if (LOCATION[3] == 'blackboard' && LOCATION[4]) thread.blackboard();
if (LOCATION[3] == 'playlist' && LOCATION[5].startsWith('pl')) thread.playlist();
if (LOCATION[3] == 'medialist' && LOCATION[4] == 'play') thread.medialist();
if (LOCATION[3] == 's' && (LOCATION[5].toLowerCase().startsWith('av') || LOCATION[5].toLowerCase().startsWith('bv'))) thread.svideo();
if (LOCATION[2] == 'space.bilibili.com') thread.space();
if (LOCATION[2] == 'www.bilibili.com' && (LOCATION[3].startsWith('\?') || LOCATION[3].startsWith('\#') || LOCATION[3].startsWith('index.'))) thread.home();
if (LOCATION[3] == 'v' && LOCATION[4] == "popular") thread.rank();
}
else if (LOCATION[2] == 'www.bilibili.com') thread.home();
// 全局调用
// 绘制设置入口
if (window.self == window.top) UI.init();
// 创建全局样式
deliver.setGlobalStyle();
// 启用xhr hook
intercept.init();
// 启用弹幕哈希引擎
if (window.self == window.top && config.reset.midcrc) window.midcrc = new BiliBili_midcrc();
// 登录鉴权
if (window.self == window.top) deliver.accesskey();
// DOM修改监听调用
document.addEventListener("DOMNodeInserted",(msg) => {
// 去除预览提示框
if (/bilibili-player-video-toast-pay/.test(msg.target.className)) deliver.removePreview(msg.target);
// 版面替换
if (msg.target.id == "internationalHeader") deliver.reSction();
if (msg.target.id == "bili-header-m") if (document.getElementById("internationalHeader")) document.getElementById("internationalHeader").remove();
// 切p监听
if (/bilibili-player-video-btn-start/.test(msg.target.className)) deliver.switchVideo();
// 创建播放器右键下载菜单
if (/bilibili-player-context-menu-container/.test(msg.target.className)) deliver.download.init(msg.target);
// 捕获评论链接
if (msg.target.src && msg.target.src.startsWith('https://api.bilibili.com/x/v2/reply') && msg.target.src.includes("oid")) src = msg.target.src;
// 捕获频道视频链接
if (msg.target.src && msg.target.src.includes("//api.bilibili.com/x/space/channel/video?")) window.src = src = msg.target.src;
// 修复失效频道视频
if (msg.relatedNode.getAttribute("class") == "row video-list clearfix") deliver.fixVideoLost.channel(src);
// 修复失效收藏视频
if (msg.target.className == "small-item disabled") deliver.fixVideoLost.favlist(msg);
// 刷新番剧分集数据
if (msg.relatedNode.className == "info-sec-av") deliver.setBangumi.episodeData("", msg);
// 失效分区转换
if (msg.target.id == "bili_ad" || msg.target.className == "report-wrap-module elevator-module" || msg.target.id == "bili-header-m" || msg.target.className == "no-data loading") deliver.fixnews(msg.target);
// 修复评论楼层&修复评论空降坐标
if (src && (/l_id/.test(msg.target.id) || /reply-wrap/.test(msg.target.className))) { deliver.setReplyFloor(src); deliver.fixVideoSeek(msg.target.parentNode); }
// 跳过充电鸣谢
if (/bilibili-player-electric-panel-jump/.test(msg.relatedNode.className)) deliver.electricPanelJump(msg.relatedNode);
// 修复分区排行
if (msg.target.id == "bili_movie" || msg.target.id == "bili_teleplay" || msg.target.id == "bili_documentary") deliver.fixrank(msg.target);
// 弹幕哈希反查
if (/danmaku-info-row/.test(msg.target.className)) deliver.danmkuHashId(msg.target);
// 其他节点监听
deliver.resetNodes();
// 收藏页切p监听
deliver.setMediaList.fixvar();
// 修复空间主页失效视频
deliver.fixVideoLost.home(msg);
// bv号转超链接
deliver.avdesc();
});
})();