Greasy Fork is available in English.

B站 珈乐

B站首页添加珈乐分区🎤🎤

// ==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_Carol .popover-video-card {
            display: none;
        }

        #bili_Carol a:hover+.popover-video-card {
            display: block;
        }

        @media (max-width: 1099.9px)
            #newbili_Carol {
                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[2]
    const CHANNEL_NAME = CHANNEL_NAME_LIST[2]
    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_Carol .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_Carol').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_Carol .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_Carol .Carol-rank-list')
        let firstTitle = item.name.replace(/<em class="keyword">(.*?)<\/em>/g, '$1')
        let first = s2d(`
            <li class="Carol_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_Carol .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_Carol .Carol-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="Carol_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('Carol_rank')
        for(let i of lst){
            i.onmouseenter = drawFlex
            i.onmouseleave = delFlex
        }
    }

    const ASOULDOM = `
        <div id="bili_Carol">
            <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 Carol-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_Carol" 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 Carol-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 Carol-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('.Carol-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('.Carol-refresh')
                    .addEventListener('click', refresh)

                // 插入最新视频
                //await getNewVideo()
                // console.log(typeof(videoList))
                drawVideosNew()

                // 插入热门视频
                drawHotNew()

                //删掉叔叔的广告分区
                document.getElementsByClassName('bili-grid')[3].remove()
                let lst = document.querySelectorAll('.win .eva-banner')
                for(let i of lst){
                    i.remove()
                }

            }
        },
        false,
    )

})()