B站首页添加A-Soul分区

B站首页看A-Soul

// ==UserScript==
// @name               B站首页添加A-Soul分区
// @namespace          https://space.bilibili.com/449968879
// @version            1.2
// @description        B站首页看A-Soul
// @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_asoul .popover-video-card {
            display: none;
        }
 
        #bili_asoul a:hover+.popover-video-card {
            display: block;
        }
 
        @media (max-width: 1099.9px)
            #newbili_asoul {
                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 TITLE = 'A-SOUL'
    const KEY_WORDS = ['ASOUL', '向晚', '贝拉', '珈乐', '乃琳', '嘉然']
    const CHANNEL_ID = 4429874
    const ICON =
        ''
    const ICON_DARK =
        ''
    let currentPage = 1
    let page = 0
    let videoList = []
    let knd
 
    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 getNewVideo() {
        let res = await fetch(
            `https://api.bilibili.com/x/web-interface/search/type?context=&order=pubdate&keyword=${KEY_WORDS}&search_type=video&page=${currentPage++}`,
        )
        videoList = videoList.concat((await res.json()).data.result)
    }
 
    async function getHotVideo() {
        let res = await fetch(
            `https://api.bilibili.com/x/web-interface/web/channel/multiple/list?channel_id=${CHANNEL_ID}&sort_type=hot&offset=&page_size=10`,
        )
        return (await res.json()).data.list[0].items
    }
 
    function bigNumber(num) {
        return num > 10000 ? `${(num / 10000).toFixed(2)}万` : num
    }
 
    function s2d(string) {
        return new DOMParser().parseFromString(string, 'text/html').body
            .childNodes[0]
    }
 
    async function refresh() {
        page++
        if (videoList.length < page * 10 + 10) {
            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_asoul .zone-list-box')
        VIDEO_DOM.innerHTML = ''
 
        videoList.slice(page * 10, page * 10 + 10).forEach((item) => {
            let title = item.title.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.pic}" alt="">
                            <div class="count">
                                <div class="left"><span><i class="bilifont bili-icon_shipin_bofangshu"></i>${bigNumber(item.play)}</span>
                                <span><i class="bilifont bili-icon_shipin_dianzanshu"></i>${bigNumber(item.favorites)}</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.mid}/" target="_blank" class="up"><i class="bilifont bili-icon_xinxi_UPzhu"></i>${item.author}</a>
                </div>
            `)
            VIDEO_DOM.append(DOM)
        })
    }
 
    function drawVideosNew() {
        const VIDEO_DOM = document.getElementById('newbili_asoul').getElementsByClassName('video-card-body')[0]
        VIDEO_DOM.innerHTML = ''
 
        videoList.slice(page * 10, page * 10 + 10).forEach((item) => {
            let title = item.title.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="${item.pic}" type="image/webp"><img src="${item.pic}" 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.play)}</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.favorites)}</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.title}</h3>
                                </a>
                                <p class="bili-video-card__info--bottom">
                                    <!---->
                                    <a class="bili-video-card__info--owner" href="//space.bilibili.com/${item.mid}/" 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}</span>
                                        <span class="bili-video-card__info--date">· ${timeFormat(item.duration)}</span>
                                    </a>
                                </p>
                            </div>
                        </div>
                    </div>
                </div>
            `)
            VIDEO_DOM.append(DOM)
        })
    }
 
    async function drawFirst(item) {
        const RANK_DOM = document.querySelector('#bili_asoul .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)
    }
 
    function removeUrlPrefix(str) {
        return str.slice(5,str.length);
    }
 
    async function drawFirstNew(item) {
        const RANK_DOM = document.querySelector('#newbili_asoul .asoul-rank-list')
        let firstTitle = item.name.replace(/<em class="keyword">(.*?)<\/em>/g, '$1')
        let first = s2d(`
            <li class="asoul_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_asoul .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_asoul .asoul-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="asoul_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('asoul_rank')
        for(let i of lst){
            i.onmouseenter = drawFlex
            i.onmouseleave = delFlex
        }
    }
 
    const ASOULDOM = `
        <div id="bili_asoul">
            <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://search.bilibili.com/all?keyword=${KEY_WORDS}" target="_blank"    class="name">${TITLE}</a>
                        </div>
                        <div class="exchange-btn">
                            <div class="btn btn-change A-Soul-refresh">
                                <i class="bilifont bili-icon_caozuo_huanyihuan"></i>
                                换一换
                            </div>
                            <a href="https://search.bilibili.com/all?keyword=${KEY_WORDS}&order=totalrank" 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_asoul" 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://search.bilibili.com/all?keyword=${KEY_WORDS}" target="_blank"><span>${TITLE}</span></a>
                    </div>
                    <div class="right"><button class="primary-btn roll-btn A-Soul-refresh">
                        <svg style="transform: rotate(0deg);"><use xlink:href="#widget-roll"></use></svg><span>换一换</span></button>
                        <a class="primary-btn see-more" href="https://search.bilibili.com/all?keyword=${KEY_WORDS}" 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 asoul-rank-list">
                                </ul>
                            </div>
                            <!---->
                            <!---->
                        </div>
                    </div>
                </div>
            </aside>
        </section>
    `
 
    window.addEventListener(
        'load',
        async function () {
            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('.A-Soul-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()
                }
 
                let bdy_ = document.querySelector('.dark')
                if(bdy_ != null) {
                    let drk = document.querySelector('#bili_asoul .svg-icon')
                    drk.src = ICON_DARK
                }
            } 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('.A-Soul-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()
                }
 
                let bdy_ = document.querySelector('.dark')
                if(bdy_ != null) {
                    let drk = document.querySelector('#newbili_asoul .svg-icon')
                    drk.src = ICON_DARK
                }
            }
        },
        false,
    )
 
})()