// ==UserScript==
// @name B站 乃琳
// @namespace https://space.bilibili.com/449968879
// @version 1.1
// @description B站首页添加乃琳分区🍧🍧
// @author mjtlyzbsy
// @match https://www.bilibili.com/
// @match https://www.bilibili.com/?spm_id_from=*
// @icon 
// @grant GM_addStyle
// @license MIT
// ==/UserScript==
;(function () {
'use strict'
GM_addStyle(`
#bili_Eileen .popover-video-card {
display: none;
}
#bili_Eileen a:hover+.popover-video-card {
display: block;
}
@media (max-width: 1099.9px)
#newbili_Eileen {
grid-template-columns: repeat(4,1fr);
position: relative;
margin-bottom: 56px;
width: 100%;
display: grid;
grid-gap: 20px 12px;
}
.svg-icon {
margin-right: 6px;
overflow: hidden;
width: 36px;
height: 36px;
vertical-align: middle;
fill: currentColor;
}
img {
vertical-align: middle;
border-style: none;
}
.bili-rank-list-video .bili-rank-list-video__item--wrap {
display: -webkit-flex;
display: flex;
align-items: center;
justify-content: flex-start;
width: 100%;
height: 100%;
}
.bili-rank-list-video__grid>.bili-rank-list-video__list .rank-video-card {
padding-top: var(--rank-cover, 15px);
padding-bottom: var(--rank-cover, 15px);
}
.bili-rank-list-video .rank-video-card {
display: -webkit-flex;
display: flex;
align-items: center;
justify-content: flex-start;
align-items: stretch;
min-width: 0;
flex: 1;
padding: 15px 0;
width: 100%;
color: var(--text1);
}
.bili-rank-list-video a {
text-decoration: none;
color: inherit;
transition: color .2s linear;
}
.bili-rank-list-video .rank-video-card__info {
display: -webkit-flex;
display: flex;
align-items: flex-start;
justify-content: space-between;
flex-direction: column;
min-width: 0;
flex: 1;
}
`)
const ICON =''
const KEY_WORDS = ['ASOUL', '向晚', '贝拉', '珈乐', '乃琳', '嘉然']
const CHANNEL_ID_LIST = [17532493, 17532491, 17532492, 17532487, 17532495]
const CHANNEL_NAME_LIST = ['向晚大魔王', '贝拉kira', '珈乐Carol', '嘉然今天吃什么', '乃琳Queen']
const CHANNEL_ID = CHANNEL_ID_LIST[4]
const CHANNEL_NAME = CHANNEL_NAME_LIST[4]
const TITLE = CHANNEL_NAME
let videoList = []
let rankList = []
let knd
let OFFSET
async function getDetail(bvid) {
let res = await fetch(
`https://api.bilibili.com/x/web-interface/view?bvid=${bvid}`,
)
return (await res.json()).data
}
function getBv(str) {
str = str.slice(31,43);
return str;
}
async function getInitVideo() {
let res = await fetch(
`https://api.bilibili.com/x/web-interface/web/channel/multiple/list?channel_id=${CHANNEL_ID}&sort_type=hot&page_size=10`,
)
res = (await res.json()).data
OFFSET = res.offset
rankList = res.list[0].items
videoList = res.list.slice(1, 11)
}
async function getNewVideo() {
let res = await fetch(
`https://api.bilibili.com/x/web-interface/web/channel/multiple/list?channel_id=${CHANNEL_ID}&sort_type=hot&offset=${OFFSET}&page_size=10`,
)
res = (await res.json()).data
OFFSET = res.offset
videoList = res.list
}
function bigNumber(num) {
return num > 10000 ? `${(num / 10000).toFixed(2)}万` : num
}
function s2d(string) {
return new DOMParser().parseFromString(string, 'text/html').body
.childNodes[0]
}
function removeUrlPrefix(str) {
return str.slice(5,str.length);
}
async function refresh() {
await getNewVideo()
if(knd){
drawVideosNew()
} else {
drawVideos()
}
}
function timeFormat(time) {
let res = []
let [s = 0, m = 0, h = 0] = time.split(':').reverse()
res.unshift(String(s).padStart(2, '0'))
res.unshift(String(m % 60).padStart(2, '0'))
res.unshift(String(h % 60).padStart(2, '0'))
return res.join(':')
}
function drawVideos() {
const VIDEO_DOM = document.querySelector('#bili_Eileen .zone-list-box')
VIDEO_DOM.innerHTML = ''
videoList.forEach((item) => {
let title = item.name.replace(/<em class="keyword">(.*?)<\/em>/g, '$1')
let DOM = s2d(`
<div class="video-card-common">
<div class="card-pic card-pic-hover"><a href="//www.bilibili.com/video/${item.bvid}" target="_blank"><img src="${item.cover}" alt="">
<div class="count">
<div class="left"><span><i class="bilifont bili-icon_shipin_bofangshu"></i>${bigNumber(item.view_count)}</span>
<span><i class="bilifont bili-icon_shipin_dianzanshu"></i>${bigNumber(item.like_count)}</span>
</div>
<div class="right"><span>${timeFormat(item.duration)}</span></div>
</div><i class="crown"></i>
</a>
<div class="watch-later-video van-watchlater black">
<span class="wl-tips" style="display: none;"></span>
</div>
</div>
<a href="//www.bilibili.com/video/${item.bvid}" target="_blank" title="${title}" class="title">${title}</a><a href="//space.bilibili.com/${item.author_id}/" target="_blank" class="up"><i class="bilifont bili-icon_xinxi_UPzhu"></i>${item.author_name}</a>
</div>
`)
VIDEO_DOM.append(DOM)
})
}
function drawVideosNew() {
const VIDEO_DOM = document.getElementById('newbili_Eileen').getElementsByClassName('video-card-body')[0]
VIDEO_DOM.innerHTML = ''
videoList.forEach((item) => {
let title = item.name.replace(/<em class="keyword">(.*?)<\/em>/g, '$1')
let DOM = s2d(`
<div class="bili-video-card" data-report="partition_recommend.content">
<div class="bili-video-card__skeleton hide">
<div class="bili-video-card__skeleton--cover"></div>
<div class="bili-video-card__skeleton--info">
<div class="bili-video-card__skeleton--face"></div>
<div class="bili-video-card__skeleton--right">
<p class="bili-video-card__skeleton--text"></p>
<p class="bili-video-card__skeleton--text short"></p>
<p class="bili-video-card__skeleton--light"></p>
</div>
</div>
</div>
<div class="bili-video-card__wrap __scale-wrap">
<a href="//www.bilibili.com/video/${item.bvid}" target="_blank" data-mod="partition_recommend" data-idx="content" data-ext="click">
<div class="bili-video-card__image __scale-player-wrap">
<div class="bili-video-card__image--wrap">
<div class="bili-watch-later" style="display: none;"><svg class="bili-watch-later__icon">
<use xlink:href="#widget-watch-later"></use>
</svg><span class="bili-watch-later__tip" style="display: none;"></span>
</div>
<picture class="v-img bili-video-card__cover">
<!---->
<source srcset="${removeUrlPrefix(item.cover)}" type="image/webp"><img src="${removeUrlPrefix(item.cover)}" alt=${title} loading="lazy" onload="">
</picture>
<div class="v-inline-player"></div>
</div>
<div class="bili-video-card__mask">
<div class="bili-video-card__stats">
<div class="bili-video-card__stats--left"><span class="bili-video-card__stats--item"><svg
class="bili-video-card__stats--icon">
<use xlink:href="#widget-play-count"></use>
</svg><span class="bili-video-card__stats--text">${bigNumber(item.view_count)}</span></span><span
class="bili-video-card__stats--item"><svg class="bili-video-card__stats--icon">
<use xlink:href="#widget-agree"></use>
</svg><span class="bili-video-card__stats--text">${bigNumber(item.like_count)}</span></span></div><span
class="bili-video-card__stats__duration">${timeFormat(item.duration)}</span>
</div>
</div>
</div>
</a>
<div class="bili-video-card__info __scale-disable">
<!---->
<div class="bili-video-card__info--right">
<a href="//www.bilibili.com/video/${item.bvid}" target="_blank" data-mod="partition_recommend" data-idx="content" data-ext="click">
<h3 class="bili-video-card__info--tit" title=${title}>${item.name}</h3>
</a>
<p class="bili-video-card__info--bottom">
<!---->
<a class="bili-video-card__info--owner" href="//space.bilibili.com/${item.author_id}/" target="_blank"
data-mod="partition_recommend" data-idx="content" data-ext="click"><svg
class="bili-video-card__info--owner__up">
<use xlink:href="#widget-up"></use>
</svg>
<span class="bili-video-card__info--author">${item.author_name}</span>
</a>
</p>
</div>
</div>
</div>
</div>
`)
VIDEO_DOM.append(DOM)
})
}
async function drawFirst(item) {
const RANK_DOM = document.querySelector('#bili_Eileen .rank-list')
let firstDetail = await getDetail(item.bvid)
let firstTitle = item.name.replace(/<em class="keyword">(.*?)<\/em>/g, '$1')
let first = s2d(`
<div class="rank-wrap"><span class="number on">1</span>
<div class="preview">
<div class="pic">
<a href="//www.bilibili.com/video/${item.bvid}" target="_blank" class="link">
<img src="${item.cover}" alt="${firstTitle}">
</a>
<div class="watch-later-video van-watchlater black"><span class="wl-tips" style="display: none;"></span>
</div>
</div>
<div class="txt">
<a href="//www.bilibili.com/video/${item.bvid}" target="_blank" class="link">
<p title="${firstTitle}">${firstTitle}</p>
</a><span>播放次数:${bigNumber(firstDetail.stat.view)}</span></div>
</div>
<div class="popover-video-card pvc" style="display: none;">
<div class="content"><img src="${item.cover}" alt="">
<div class="info">
<p class="f-title">${firstTitle}</p>
<p class="subtitle"><span class="name">${item.author_name}</span>
<span class="point">·</span><span class="time">2021-11-22</span></p>
</div>
</div>
<div class="count">
<ul>
<li><i class="bilifont bili-icon_shipin_bofangshu"></i><span>${bigNumber(
firstDetail.stat.view,
)}</span></li>
<li><i class="bilifont bili-icon_shipin_danmushu"></i><span>${bigNumber(
firstDetail.stat.danmaku,
)}</span></li>
<li><i class="bilifont bili-icon_shipin_shoucangshu"></i><span>${bigNumber(
firstDetail.stat.favorite,
)}</span></li>
<li><i class="bilifont bili-icon_shipin_yingbishu"></i><span>${bigNumber(
firstDetail.stat.coin,
)}</span></li>
</ul>
</div>
</div>
</div>
`)
RANK_DOM.append(first)
}
async function drawFirstNew(item) {
const RANK_DOM = document.querySelector('#newbili_Eileen .Eileen-rank-list')
let firstTitle = item.name.replace(/<em class="keyword">(.*?)<\/em>/g, '$1')
let first = s2d(`
<li class="Eileen_rank" class="bili-rank-list-video__item">
<div class="bili-rank-list-video__item--wrap">
<span class="bili-rank-list-video__item--index" data-index="1">1</span>
<a href="//www.bilibili.com/video/${item.bvid}" class="rank-video-card" target="_blank" data-mod="partition_rank" data-idx="content" data-ext="click">
<div class="rank-video-card__image">
<picture class="v-img rank-video-card__cover">
<!---->
<source srcset="${removeUrlPrefix(item.cover+"@.webp")}" type="image/jpg">
<img src="${removeUrlPrefix(item.cover+"@.webp")}" alt=${firstTitle} loading="lazy" onload="">
</picture>
</div>
<div class="rank-video-card__info">
<h3 class="rank-video-card__info--tit" title=${firstTitle}> ${firstTitle}</h3>
</div>
</a>
</div>
</li>
`)
RANK_DOM.append(first)
}
async function drawHot() {
const RANK_DOM = document.querySelector('#bili_Eileen .rank-list')
//let rankList = await getHotVideo()
await drawFirst(rankList.shift())
let index = 1
for(let index = 0 ; index < rankList.length ; index++){
let title = rankList[index].name.replace(/<em class="keyword">(.*?)<\/em>/g, '$1')
let item = await getDetail(rankList[index].bvid)
let DOM = s2d(`
<div class="rank-wrap"><span class="number ${index < 2 && 'on'}">${index + 2}</span>
<a href="//www.bilibili.com/video/${
item.bvid
}" target="_blank" class="link">
<p title="${title}" class="title">${title}</p>
</a>
<div class="popover-video-card pvc">
<div class="content"><img
src="${item.pic}" alt="">
<div class="info">
<p class="f-title">${title}</p>
<p class="subtitle"><span class="name">${item.owner.name}</span><span class="point">·</span><span
class="time">${timeFormat(rankList[index].duration)}</span></p>
</div>
</div>
<div class="count">
<ul>
<li><i class="bilifont bili-icon_shipin_bofangshu"></i><span>${bigNumber(item.stat.view)}</span></li>
<li><i class="bilifont bili-icon_shipin_danmushu"></i><span>${bigNumber(item.stat.danmaku)}</span></li>
<li><i class="bilifont bili-icon_shipin_shoucangshu"></i><span>${bigNumber(item.stat.favorite)}</span></li>
<li><i class="bilifont bili-icon_shipin_yingbishu"></i><span>${bigNumber(item.stat.coin)}</span></li>
</ul>
</div>
</div>
</div>
`)
RANK_DOM.append(DOM)
}
}
async function drawFlex() {
let lst = document.querySelectorAll('.win .ASoul_flex')
for(let i of lst){
i.remove()
}
const RANK_DOM = document.querySelector('.win')
let item =await getDetail(getBv(this.querySelector('.rank-video-card').href))
let posl = 117
let post = 0
let y = this
while(y != document){
posl = posl + y.offsetLeft + y.clientLeft;
post = post + y.offsetTop + y.clientTop;
y = y.parentNode;
}
let firstTitle = item.title.replace(/<em class="keyword">(.*?)<\/em>/g, '$1')
let first = s2d(`
<div class="ASoul_flex" style="position: absolute; z-index: 2000; top: ${post}px; left: ${posl}px;">
<div class="v-popover is-top" style="padding-bottom: 0px; margin-left: 0px; pointer-events: none;">
<div class="v-popover-content bili-rank-list-video">
<div class="rank-video-card__popover">
<div class="rank-video-card__popover--top">
<div class="rank-video-card__image rank-video-card__popover--image">
<picture class="v-img rank-video-card__cover rank-video-card__popover--cover">
<source srcset="${removeUrlPrefix(item.pic+"@.webp")}" type="image/webp"><img src="${removeUrlPrefix(item.pic+"@.webp")}" alt=${firstTitle} loading="lazy" onload="">
</picture>
</div>
<div class="rank-video-card__info rank-video-card__popover--info">
<h3 class="rank-video-card__popover--tit" title="${firstTitle}">${firstTitle}</h3>
<p class="rank-video-card__popover--author">
<span>${item.owner.name}</span>
<span></span></p>
</div>
</div>
<ul class="rank-video-card__popover--stats">
<li class="rank-video-card__popover--stats__item"><svg class="rank-video-card__popover--icon">
<use xlink:href="#widget-play-count"></use>
</svg><span>${bigNumber(item.stat.view)}</span></li>
<li class="rank-video-card__popover--stats__item"><svg class="rank-video-card__popover--icon">
<use xlink:href="#widget-danmaku"></use>
</svg><span>${bigNumber(item.stat.danmaku)}</span></li>
<li class="rank-video-card__popover--stats__item"><svg class="rank-video-card__popover--icon">
<use xlink:href="#widget-favorite"></use>
</svg><span>${bigNumber(item.stat.favorite)}</span></li>
<li class="rank-video-card__popover--stats__item"><svg class="rank-video-card__popover--icon">
<use xlink:href="#widget-coin"></use>
</svg><span>${bigNumber(item.stat.coin)}</span></li>
</ul>
</div>
</div>
</div>
</div>
`)
RANK_DOM.append(first)
}
function delFlex() {
let lst = document.querySelectorAll('.win .ASoul_flex')
for(let i of lst){
i.remove()
}
}
async function drawHotNew() {
const RANK_DOM = document.querySelector('#newbili_Eileen .Eileen-rank-list')
//let rankList = await getHotVideo()
await drawFirstNew(rankList.shift())
rankList.forEach((item, index) => {
let title = item.name.replace(/<em class="keyword">(.*?)<\/em>/g, '$1')
let DOM = s2d(`
<li class="Eileen_rank" class="bili-rank-list-video__item">
<div class="bili-rank-list-video__item--wrap">
<span class="bili-rank-list-video__item--index" data-index="${index + 2}">${index + 2}</span>
<a href="//www.bilibili.com/video/${item.bvid}" class="rank-video-card rank-video-card__concise" target="_blank" data-mod="partition_rank" data-idx="content" data-ext="click">
<!---->
<div class="rank-video-card__info">
<h3 class="rank-video-card__info--tit" title=${title}>${item.name}</h3>
</div>
</a>
</div>
</li>
`)
RANK_DOM.append(DOM)
})
let lst = document.getElementsByClassName('Eileen_rank')
for(let i of lst){
i.onmouseenter = drawFlex
i.onmouseleave = delFlex
}
}
const ASOULDOM = `
<div id="bili_Eileen">
<div class="space-between report-wrap-module report-scroll-module" id="bili_report_douga" scrollshow="true">
<div class="card-list">
<header class="storey-title">
<div class="l-con">
<img class="svg-icon" src="${ICON}" />
<a href="https://www.bilibili.com/v/channel/${CHANNEL_ID}" target="_blank" class="name">${TITLE}</a>
</div>
<div class="exchange-btn">
<div class="btn btn-change Eileen-refresh">
<i class="bilifont bili-icon_caozuo_huanyihuan"></i>
换一换
</div>
<a href="https://www.bilibili.com/v/channel/${CHANNEL_ID}" target="_blank" class="btn more">更多
<i class="bilifont bili-icon_caozuo_qianwang"></i>
</a>
</div>
</header>
<div class="zone-list-box"> </div>
</div>
<div class="rank-list">
<header class="rank-header"><span class="name">排行榜</span><a
href="https://www.bilibili.com/v/channel/${CHANNEL_ID}?tab=multiple" target="_blank"
class="more">更多<i class="bilifont bili-icon_caozuo_qianwang"></i></a></header>
</div>
</div>
</div>
`
const ASOULNEWDOM = `
<section id="newbili_Eileen" class="bili-grid">
<div class="video-card-list is-main">
<div class="area-header">
<div class="left">
<a id="${TITLE}" class="the-world area-anchor" data-id="0"></a>
<img class="svg-icon" src="${ICON}" />
<a class="title" href="https://www.bilibili.com/v/channel/${CHANNEL_ID}" target="_blank"><span>${TITLE}</span></a>
</div>
<div class="right"><button class="primary-btn roll-btn Eileen-refresh">
<svg style="transform: rotate(0deg);"><use xlink:href="#widget-roll"></use></svg><span>换一换</span></button>
<a class="primary-btn see-more" href="https://www.bilibili.com/v/channel/${CHANNEL_ID}" target="_blank"><span>查看更多</span><svg><use xlink:href="#widget-arrow"></use></svg></a>
</div>
</div>
<div class="video-card-body"></div>
</div>
<aside>
<div class="aside-wrap">
<div class="aside-head">
<div class="area-header">
<div class="left">
<!---->
<!---->
<a class="title rank-title" href="https://www.bilibili.com/v/channel/${CHANNEL_ID}?tab=multiple" target="_blank"><span>排行榜</span></a></div>
<div class="right">
<a class="primary-btn see-more" href="https://www.bilibili.com/v/channel/${CHANNEL_ID}?tab=multiple" target="_blank">
<span>更多</span>
<svg><use xlink:href="#widget-arrow"></use></svg>
</a>
</div>
</div>
</div>
<div class="aside-body">
<div class="aside-core">
<div class="bili-rank-list-video bili-rank-list-video__grid" data-report="partition_rank.content">
<ul class="bili-rank-list-video__list video-rank-list Eileen-rank-list">
</ul>
</div>
<!---->
<!---->
</div>
</div>
</div>
</aside>
</section>
`
window.addEventListener(
'load',
async function () {
await getInitVideo()
let bdy = document.querySelector('.win')
if(bdy === null){
knd = false
let content = document.querySelector('.first-screen')
let anchor = document.querySelector('#reportFirst3')
let init = s2d(ASOULDOM)
// 插入初始模版
content.insertBefore(init, anchor)
//点击事件
document
.querySelector('.Eileen-refresh')
.addEventListener('click', refresh)
// 插入最新视频
//await getNewVideo()
drawVideos()
// 插入热门视频
drawHot()
//删掉叔叔的广告分区
document.querySelector('#reportFirst2').remove()
let lst = document.querySelectorAll('.banner-card')
for(let i of lst){
i.remove()
}
} else {
knd = true
let content = document.getElementsByClassName('bili-layout')[0]
let anchor = document.getElementsByClassName('bili-grid')[1]
let init = s2d(ASOULNEWDOM)
// 插入初始模版
content.insertBefore(init, anchor)
//点击事件
document
.querySelector('.Eileen-refresh')
.addEventListener('click', refresh)
// 插入最新视频
//await getNewVideo()
drawVideosNew()
// 插入热门视频
drawHotNew()
//删掉叔叔的广告分区
document.getElementsByClassName('bili-grid')[3].remove()
let lst = document.querySelectorAll('.win .eva-banner')
for(let i of lst){
i.remove()
}
}
},
false,
)
})()