b站直播徽章切换增强

临时版本

目前為 2022-06-01 提交的版本,檢視 最新版本

// ==UserScript==
// @name         b站直播徽章切换增强
// @version      1.0.3
// @description  临时版本
// @author       Pronax
// @include      /https:\/\/live\.bilibili\.com\/(blanc\/)?\d+/
// @icon         http://bilibili.com/favicon.ico
// @grant        GM_addStyle
// @grant        GM_getValue
// @grant        GM_setValue
// @require		 https://lib.baomitu.com/vue/2.6.14/vue.js
// @require      https://greasyfork.org/scripts/439903-blive-room-info-api/code/blive_room_info_api.js?version=1037039
// @namespace http://tampermonkey.net/
// ==/UserScript==

(function init() {
    'use strict';

    if (!document.cookie.match(/bili_jct=(\w*); /)) { return; }

    let controlPanelCtnrBox = document.querySelector(".medal-section");
    if (controlPanelCtnrBox && Object.keys(controlPanelCtnrBox.dataset).length) {
        // 浅拷贝一次,用于抹掉子元素的事件
        controlPanelCtnrBox.innerHTML = `<span id="medal-selector" class="dp-i-block medal"><span class="action-item medal get-medal"></span></span>`;
        let tempElement = document.createElement("div");
        tempElement.id = "medel_switch_box";
        document.querySelector(".bottom-actions").after(tempElement);
    } else {
        requestAnimationFrame(function () {
            init();
        });
        return;
    }

    // body内的条目css
    GM_addStyle(`
        .medal-list-move {
            transition: transform .5s !important;
        }

        .medal-wear-body{
            height: 335px;
            margin-top: 5px;
            padding-right: 2px;
            overflow:auto;
            scrollbar-width: thin;
        }
        .medal-wear-body::-webkit-scrollbar{
            width:6px
        }
        .medal-wear-body::-webkit-scrollbar-thumb{
            background-color:#aaa
        }
        
        .medal-item-content {
            display: flex;
            justify-content: space-between;
        }
        
        .medal-wear-body .medal-item {
            cursor:pointer;
            padding: 5px 5px 3px;
            background: 0;
            border: 1px solid transparent;
            border-radius:5px;
            width:calc(100% - 12px);
            text-align:left;
            transition: border,background .2s;
        }
        
        .medal-wear-body .medal-item:hover {
            border: 1px solid #d7d7d7;
            background-color: #f5f5f5;
        }

        .medal-item .face,
        .medal-item .search-user-avatar{
            width: auto;
            height: 35px;
            margin-right: 5px;
            padding: 1px;
            position: relative;
            transition: filter .3s;
        }

        .medal-item .face:hover,
        .medal-item .search-user-avatar:hover {
            filter: drop-shadow(0px 0px 3px #FB7299);
        }

        .medal-item .face>img {
            height: 35px;
            border-radius: 50%;
        }

        .medal-wear-body .medal-item .name{
            color: #666;
            position: relative;
            max-width: calc(100% - 78px);
            font-size: 14px;
            line-height: 18px;
            overflow: hidden;
            text-overflow: ellipsis;
            white-space: nowrap;
        }
        
        .medal-wear-body .medal-item .living-gif {
            background-image: url(//s1.hdslb.com/bfs/static/blive/live-fansmedal-wall/static/img/icon-online.fd4254c1.gif);
            background-size: cover;
            width: 16px;
            height: 16px;
            transform: rotateY(180deg);
        }

        .medal-item .wear-icon {
            background-color: #fb7299;
            padding: 0 2px;
            color: #fff;
            height: 16px;
            border: 1px solid #fb7299;
            border-radius: 4px;
            line-height: 16px;
            font-size: 14px;
        }

        .medal-item .room-icon {
            padding: 0 2px;
            color: #fea249;
            height: 16px;
            border: 1px solid #fea249;
            border-radius: 4px;
            line-height: 16px;
            font-size: 14px;
        }
        
        .medal-item .content-icon {
            padding: 0 2px;
            color: #40bf55;
            height: 16px;
            border: 1px solid #40bf55;
            border-radius: 4px;
            line-height: 16px;
            font-size: 14px;
        }
        
        .medal-wear-body .medal-item .text{
            color: #888;
            position: relative;
            line-height: 18px;
            font-size: 14px;
        }

        .medal-wear-body .medal-item .left{
            color: #2cbce7;
            line-height: 18px;
            font-size: 14px;
            margin-right: 5px;
        }

        .medal-item-content .medal-content-head{
            height: 18px
        }

        .medal-item-content .medal-content-footer{
            height: 18px;
            padding-top: 1px;
            width: 100%;
        }

        .medal-wear-body .medal-item .progress-level-div {
            margin-top: 3px;
            width:100%;
            text-align:center;
            display: flex;
            justify-content: space-between; 
            font-size: 13px;
        }

        .medal-wear-body .medal-item .progress-level-div .level-span-left {
            text-align: right !important;
        }
        
        .medal-wear-body .medal-item .progress-level-div .level-span {
            width: 33px;
            color: #999;
            padding-top: 1px;
        }

        .medal-wear-body .medal-item .progress-level-div .progress-div {
            line-height: 16px;
            height: 14px;
            width: 70%;
            background-color: #e2e8ec;
            border-radius: 2px;
            margin: 0 2px;
            position: relative;
            overflow: hidden;
        }

        .medal-wear-body .medal-item .progress-level-div .progress-div-cover {
            position: absolute;
            left: 0;
            top: 0;
            overflow: hidden;
            background-color: #23ade5;
        }

        .medal-wear-body .medal-item .progress-level-div .progress-div .progress-num-span {
            color: #23ade5;
        }

        .medal-wear-body .medal-item .progress-level-div .progress-div-cover .progress-num-span-cover {
            width: 174px;
            position: relative;
            z-index: 1000;
            color: #fff;
        }
    `);

    // 面板css
    GM_addStyle(`
        .chat-input-ctnr .medal-section{
            position: static;
            display: flex;
            align-items: center;
            justify-content: center;
            flex-shrink: 0;
            padding: 0 12px;
            min-width: 70px;
            height: 56px;
            border-right: 1px solid #e9eaec;
            box-sizing: border-box;
        }

        .medal-section .action-item.medal.get-medal,
        .medal-section .action-item.medal.wear-medal {
            width: 41px;
            height: 24px;
            background-image: url()
        }

        .medal-section .action-item.medal {
            background-size: cover;
            border: 0
        }

        .medal-section .action-item {
            display: inline-block;
            margin: 0 2px;
            font-size: 12px;
            color: #fff;
            line-height: 14px;
            text-align: center;
            border-radius: 2px;
            cursor: pointer;
            -webkit-user-select: none;
            -moz-user-select: none;
            -ms-user-select: none;
            user-select: none;
        }

        .dialog-ctnr.medal {
            z-index: 999;
            padding: 10px 14px 10px 16px;
            position: absolute;
            bottom: 100px;
            left: -1px;
            width: 302px;
        }
        
        .dialog-ctnr {
            padding: 16px;
            z-index: 699;
        }
        .medal-ctnr {
            width: 268px;
        }
        .title {
            font-weight: 400;
            font-size: 18px;
            margin: 0;
            color: #23ade5;
        }
        .des {
            cursor:pointer;
            color: #666;
            line-height: 20px;
        }
        .des>.svg-icon{
            width: 13px;
            height: 13px;
            font-size: 13px;
            background-position: 0 -6em;
        }
        .des>.svg-icon.checkbox-selected{
            background-position: 0 -7em;
        }
        .qs-icon {
            width: 14px;
            height: 14px;
            background-size: 100%;
            background-image: url();
            cursor: pointer;
            margin-left: 5px;
            position: relative;
            top: 0;
        }
        .link-radio-button-ctnr {
            display: inline-block;
            cursor: default;
            vertical-align: middle;
            font-size: 0;
        }
        .footer-line {
            position: relative;
            left: -16px;
            width: 300px;
            border-top: 1px solid #f0f0f0;
            margin-top: 3px;
        }
        .medal-wear-footer {
            margin-top: 10px;
            font-size: 14px;
            color: #23ade5;
        }
        .medal-wear-footer .cancel-wear {
            cursor: pointer;
        }
        .medal-wear-footer a {
            color: #23ade5;
        }
        
        .medal-wear-footer .right-span {
            float: right;
        }
        .medal-wear-footer .arrow-box{
            width: 10px;
            height: 10px;
            font-size: 10px;
            position: relative;
            top: 2px;
        }
    `);

    GM_addStyle(`

    .search-user-avatar.avatar-small .avatar-wrap {
        transform: scale(.8);
    }

    .search-user-avatar .avatar-wrap {
        width: 100%;
        height: 35px;
    }

    .bili-avatar {
        display: block;
        position: relative;
        background-image: url();
        background-size: cover;
        border-radius: 50%;
        margin: 0;
        padding: 0;
        width:35px;
        height:35px;
    }

    .bili-avatar .bili-avatar-img {
        border: 1px solid var(--line_light);
    }

    .bili-avatar-img-radius {
        border-radius: 50%;
    }

    .bili-avatar-img {
        border: none;
        display: block;
        -o-object-fit: cover;
        object-fit: cover;
        image-rendering: -webkit-optimize-contrast;
    }

    .bili-avatar-face {
        position: absolute;
        top: 50%;
        left: 50%;
        -webkit-transform: translate(-50%, -50%);
        -moz-transform: translate(-50%, -50%);
        -ms-transform: translate(-50%, -50%);
        -o-transform: translate(-50%, -50%);
        transform: translate(-50%, -50%);
        width: 100%;
        height: 100%;
    }

    .bili-avatar * {
        margin: 0;
        padding: 0;
    }

    .bili-avatar-right-icon {
        width: 27.5%;
        height: 27.5%;
        position: absolute;
        right: 0;
        bottom: -1px;
        background-size: cover;
        image-rendering: -webkit-optimize-contrast;
        background-image: url();
    }

    .search-user-avatar .avatar-wrap.live-ani .a-cycle {
        position: absolute;
        top: 50%;
        left: 50%;
        transform: translate(-50%, -50%);
        width: 35px;
        height: 35px;
        border: 1px solid #ff6699;
        border-radius: 50%;
        z-index: 1;
        opacity: 0;
        animation: scaleUpCircle 1.5s linear;
        animation-iteration-count: infinite;
    }

    .search-user-avatar .avatar-wrap.live-ani .a-cycle-1 {
        animation-delay: 0s;
    }

    .search-user-avatar .avatar-wrap.live-ani .a-cycle-2 {
        animation-delay: .5s;
    }

    .search-user-avatar .avatar-wrap.live-ani .a-cycle-3 {
        animation-delay: 1s;
    }

    @keyframes scaleUpCircle{
        0% {
            transform: translate(-50%, -50%) scale(1);
            opacity: 1;
        }

        100% {
            transform: translate(-50%, -50%) scale(1.5);
            opacity: 0;
        }
    }
    `);

    unsafeWindow.vm = new Vue({
        el: '#medel_switch_box',
        async created() {
            this.fansMedalInfo = Object.assign({}, this.fansMedalInfo, await this.getFansMedalInfo());
            this.refreshMedalList().then(() => {
                if (this.autoSwitch && this.fansMedalInfo.has_fans_medal && this.fansMedalInfo.my_fans_medal.medal_id != this.currentlyWearing.medal.medal_id) {
                    this.switchBadge(this.fansMedalInfo.my_fans_medal.medal_id);
                }
            });
        },
        mounted: function () {
            document.querySelector("#medal-selector").onclick = () => {
                this.togglePanel();
            };
            window.addEventListener('click', e => {
                if (e.target.closest(".medal") == null && this.panelStatus) {
                    this.panelStatus = false;
                    document.querySelector(".medal-wear-body").scrollTop = 0;
                }
            });
            window.addEventListener('focus', e => {
                let wearing = GM_getValue("currentlyWearing");
                if (this.currentlyWearing.medal.medal_id != wearing.medal.medal_id) {
                    this.currentlyWearing = wearing;
                }
                if (this.name != GM_getValue("operator") && this.fansMedalInfo.my_fans_medal.medal_id != wearing.medal.medal_id) {
                    this.needSwitch = true;
                } else {
                    this.needSwitch = false;
                }
            });
            document.querySelector("#gift-control-vm .section").onmouseenter = document.querySelector("#control-panel-ctnr-box").onmouseenter = () => {
                if (this.autoSwitch && this.needSwitch && this.fansMedalInfo.my_fans_medal.medal_id != 0) {
                    this.switchBadge(this.fansMedalInfo.my_fans_medal.medal_id);
                    this.needSwitch = false;
                }
            };
        },
        computed: {
        },
        data() {
            return {
                name: Date.now().toString(16) + "-" + btoa(location.host),
                jct: document.cookie.match(/bili_jct=(\w*); /)[1],
                fansMedalInfo: {
                    "has_fans_medal": false,
                    "my_fans_medal": {
                        "target_id": 0,
                        "medal_id": 0
                    }
                },
                currentlyWearing: GM_getValue("currentlyWearing", {
                    medal: {
                        medal_id: 0
                    }
                }),
                autoSwitch: GM_getValue("autoSwitch", false),
                needSwitch: false,
                panelStatus: false,
                medalWall: GM_getValue("medalWall", []),
            }
        },
        watch: {
            currentlyWearing: {
                handler(val) {
                    this.refreshMedal();
                },
                immediate: true
            },
            autoSwitch(val) {
                GM_setValue("autoSwitch", val);
            }
        },
        methods: {
            async getFansMedalInfo() {
                let uid = await ROOM_INFO_API.getUid();
                let res = await fetch(`https://api.live.bilibili.com/xlive/app-ucenter/v1/fansMedal/fans_medal_info?target_id=${uid}`, { credentials: 'include', });
                let json = await res.json();
                if (json.code == json.message) {
                    return json.data;
                }
                alert("徽章初始化失败:", json.message);
            },
            async refreshMedalList(page = 1) {
                return new Promise((resolve, reject) => {
                    fetch(`https://api.live.bilibili.com/xlive/app-ucenter/v1/fansMedal/panel?page=${page}&page_size=50`, { credentials: 'include', })
                        .then(res => res.json())
                        .then(json => {
                            if (json.code == json.message) {
                                let list = [].concat(json.data.list, json.data.special_list);
                                if (list.length == 1 && list[0].medal.wearing_status) {
                                    this.currentlyWearing = list[0];
                                }
                                list.sort(this.sort);
                                // if (page == 1) {
                                this.medalWall = list;
                                // } else {
                                //     this.medalWall = this.medalWall.concat(json.data.list, json.data.special_list);
                                // }
                                if (json.data.page_info.has_more) {
                                    console.log("has_more");
                                    // setTimeout(() => {
                                    //     this.refreshMedalList(json.data.page_info.next_page);
                                    // }, 2000);
                                }
                                resolve();
                            } else {
                                reject();
                            }
                        })
                        .catch(err => {
                            reject();
                        });
                });
            },
            switchBadge(badgeId, index) {
                let params = new URLSearchParams();
                params.set("medal_id", badgeId);
                params.set("csrf_token", this.jct);
                params.set("csrf", this.jct);
                fetch("https://api.live.bilibili.com/xlive/web-room/v1/fansMedal/wear", {
                    credentials: 'include',
                    method: 'POST',
                    body: params
                });
                // .then(res => res.json())
                // .then(json => {
                //     if (json.code == 0) {
                //     }
                // });
                if (index) {
                    this.currentlyWearing = this.medalWall[index];
                } else {
                    let result = this.medalWall.find(item => {
                        return badgeId == item.medal.medal_id;
                    });
                    if (result) {
                        this.currentlyWearing = result;
                    } else {
                        console.warn("徽章列表内找不到对应的徽章");
                        this.refreshMedalList();
                    }
                }
                GM_setValue("currentlyWearing", this.currentlyWearing);
                GM_setValue("operator", this.name);
            },
            takeOff() {
                this.currentlyWearing = { medal: { medal_id: 0 } };
                let params = new URLSearchParams();
                params.set("visit_id", '');
                params.set("csrf_token", this.jct);
                params.set("csrf", this.jct);
                fetch("https://api.live.bilibili.com/xlive/web-room/v1/fansMedal/take_off", {
                    "method": "POST",
                    "credentials": "include",
                    "body": params,
                });
                GM_setValue("currentlyWearing", this.currentlyWearing);
                GM_setValue("operator", this.name);
            },
            open: (uid) => {
                window.open(`//space.bilibili.com/${uid}`);
            },
            togglePanel() {
                this.panelStatus = !this.panelStatus;
                if (!this.panelStatus) {
                    document.querySelector(".medal-wear-body").scrollTop = 0;
                } else {
                    this.$nextTick(() => {
                        this.refreshMedalList();
                    });
                }
            },
            refreshMedal() {
                let selector = document.querySelector("#medal-selector");
                if (this.currentlyWearing.medal.medal_id != 0) {
                    selector.innerHTML = `
                        <div class="v-middle fans-medal-item medal-item-margin"
                            style="border-color:#${this.currentlyWearing.medal.medal_color_border.toString(16)}">
                            <div class="fans-medal-label"
                                style="background-image:linear-gradient(45deg,#${this.currentlyWearing.medal.medal_color_start.toString(16)},#${this.currentlyWearing.medal.medal_color_end.toString(16)})">
                                <span class="fans-medal-content">${this.currentlyWearing.medal.medal_name}</span>
                            </div>
                            <div class="fans-medal-level" style="color:#${this.currentlyWearing.medal.medal_color_start.toString(16)}">${this.currentlyWearing.medal.level}</div>
                        </div>
                    `;
                } else {
                    selector.innerHTML = `<span class="action-item medal get-medal"></span>`;
                }
            },
            sort(a, b) {
                if (a.medal.wearing_status == 1) {
                    this.currentlyWearing = a;
                }
                let count_a = a.medal.wearing_status * 600000000 + a.medal.level * 15000000 + a.medal.intimacy;
                let count_b = b.medal.wearing_status * 600000000 + b.medal.level * 15000000 + b.medal.intimacy;
                if (a.medal.target_id == this.fansMedalInfo.my_fans_medal.target_id) {
                    count_a = Number.MAX_VALUE;
                } else if (b.medal.target_id == this.fansMedalInfo.my_fans_medal.target_id) {
                    count_b = Number.MAX_VALUE;
                } else if (a.medal.is_lighted == 0) {
                    count_a = +a.medal.level;
                } else if (b.medal.is_lighted == 0) {
                    count_b = +b.medal.level;
                }
                return count_b - count_a;
            }
        },
        template: `
            <div class="border-box dialog-ctnr common-popup-wrap medal a-scale-in" v-show="panelStatus" @mouseleave="togglePanel">
                <div class="medal-ctnr none-select">
                    <div class="medal-wear-component">
                        <h1 class="dp-i-block title">
                            我的粉丝勋章
                        </h1>
                        <a href="http://link.bilibili.com/p/help/index#/audience-fans-medal" target="_blank"
                            class="dp-i-block qs-icon"></a>
                        <div class="dp-i-block des f-right" @click="autoSwitch = !autoSwitch">
                            <span class="cb-icon svg-icon v-middle" :class="{'checkbox-selected':autoSwitch}"></span>
                            <span class="pointer dp-i-block v-middle">自动更换</span>
                        </div>
                        <transition-group name="medal-list" tag="div" class="medal-wear-body">
                            <div class="medal-item" v-for="(item,index) in medalWall" :key="item.medal.medal_id"
                                @click="currentlyWearing.medal.medal_id == item.medal.medal_id ? takeOff() : switchBadge(item.medal.medal_id,index)">
                                <div class="medal-item-content">
                                    <template v-if="item.room_info.living_status == 1">
                                        <div class="search-user-avatar p_relative avatar-small mr_md cs_pointer"  @click.stop="open(item.medal.target_id)">
                                            <div class="avatar-wrap p_relative live-ani">
                                                <div class="avatar-inner">
                                                    <div class="bili-avatar" style="width: 35px;height:35px;">
                                                        <img class="bili-avatar-img bili-avatar-face bili-avatar-img-radius"
                                                            :src="item.anchor_info.avatar">
                                                        <span class="bili-avatar-right-icon"
                                                            v-if="item.anchor_info.verify == 0"></span>
                                                    </div>
                                                </div>
                                                <div class="a-cycle a-cycle-1"></div>
                                                <div class="a-cycle a-cycle-2"></div>
                                                <div class="a-cycle a-cycle-3"></div>
                                            </div>
                                        </div>
                                    </template>
                                    <template v-else>
                                        <div class="face" @click.stop="open(item.medal.target_id)">
                                            <img :src="item.anchor_info.avatar">
                                            <span class="bili-avatar-right-icon" v-if="item.anchor_info.verify == 0"></span>
                                        </div>
                                    </template>
                                    <div class="dp-i-block v-bottom w-100 p-relative">
                                        <div class="medal-content-head">
                                            <div class="fans-medal-item f-right"
                                                :style="'border-color:#'+(item.medal.medal_color_border.toString(16))">
                                                <div class="fans-medal-label"
                                                    :style="'background-image:linear-gradient(45deg,#'+(item.medal.medal_color_start.toString(16))+',#'+(item.medal.medal_color_end.toString(16))+')'">
                                                    <span class="fans-medal-content">{{item.medal.medal_name}}</span>
                                                </div>
                                                <div class="fans-medal-level"
                                                    :style="'color:#'+(item.medal.medal_color_start.toString(16))">
                                                    {{item.medal.level}}
                                                </div>
                                            </div>
                                            <div class="name dp-i-block">{{item.anchor_info.nick_name}}</div>
                                        </div>
                                        <div class="medal-content-footer">
                                            <transition enter-active-class="a-scale-in" leave-active-class="a-scale-out"
                                                mode="out-in">
                                                <div class="wear-icon dp-i-block" :key="'wear'"
                                                    v-if="item.medal.medal_id == currentlyWearing.medal.medal_id">
                                                    佩戴中
                                                </div>
                                                <div class="room-icon dp-i-block" :key="'room'"
                                                    v-else-if="item.medal.medal_id == fansMedalInfo.my_fans_medal.medal_id">
                                                    当前房间
                                                </div>
                                                <div class="content-icon dp-i-block" :key="'content'"
                                                    v-else-if="item.superscript != null">
                                                    {{item.superscript.content}}
                                                </div>
                                            </transition>
                                            <span class="text f-right dp-i-block">{{item.medal.today_feed}}/{{item.medal.day_limit}}</span>
                                            <span v-if="false" class="left f-right dp-i-block">{{item.medal.next_intimacy - item.medal.intimacy}}</span>
                                        </div>
                                    </div>
                                </div>
                                <div class="progress-level-div">
                                    <span class="dp-i-block level-span">Lv.{{item.medal.level}}</span>
                                    <div class="dp-i-block progress-div">
                                        <span
                                            class="dp-i-block progress-num-span">{{item.medal.intimacy}}/{{item.medal.next_intimacy}}</span>
                                        <div class="dp-i-block progress-div-cover"
                                            :style="'width:'+(item.medal.intimacy / item.medal.next_intimacy * 100) + '%'">
                                            <span class="dp-i-block progress-num-span-cover">
                                                {{item.medal.intimacy}}/{{item.medal.next_intimacy}}
                                            </span>
                                        </div>
                                    </div>
                                    <span class="dp-i-block level-span">Lv.{{item.medal.level + 1}}</span>
                                </div>
                            </div>
                        </transition-group>
                        <div class="footer-line"></div>
                        <div class="dp-block medal-wear-footer">
                            <span class="dp-i-block cancel-wear" @click="takeOff">
                                不佩戴勋章
                            </span>
                            <a href="https://link.bilibili.com/p/center/index#/user-center/wearing-center/my-medal" target="_blank"
                                class="dp-i-block right-span">
                                装扮中心
                                <span class="dp-i-block icon-font icon-arrow-right arrow-box"></span>
                            </a>
                        </div>
                    </div>
                </div>
            </div>
        `,
    });

    // function toast() {
    //     let id = Math.random() * 1000 >> 1;
    //     let temp = document.createElement("div");
    //     temp.innerHTML = `<div id="badgeSwitcher-${id}" class="link-toast info badgeToast" style="left: 16px;bottom:360px"><span class="toast-text">佩戴成功!信仰爆表!&lt;(▰˘◡˘▰)&gt;</span></div>`;
    //     document.querySelector(".medal-section").append(temp.firstElementChild);
    //     let toast = document.querySelector(`#badgeSwitcher-${id}`);
    //     toast.style.opacity = 1;
    //     setTimeout(() => {
    //         fadeOut();
    //     }, 1000);

})();