Greasy Fork is available in English.

b站直播徽章切换增强

展示全部徽章,展示更多信息,更方便切换,可以自动切换徽章

Instalirajte ovu skriptu?
Autorov prijedlog skripta

Možda ti se također svidi b站自动续牌.

Instalirajte ovu skriptu
// ==UserScript==
// @name         b站直播徽章切换增强
// @version      1.2.9
// @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://lf3-cdn-tos.bytecdntp.com/cdn/expire-1-M/vue/2.6.14/vue.min.js
// @require      https://greasyfork.org/scripts/439903-blive-room-info-api/code/blive_room_info_api.js?version=1037039
// @namespace http://tampermonkey.net/
// ==/UserScript==

// ! csrf过期后调用原生徽章按钮不会刷新状态
// ! bug: 有一个最近获得的粉丝牌置顶,然后获取了新粉丝牌,且新粉丝牌为当前房间(最新获得被当前房间顶掉)

function main() {
    'use strict';

    // 设置是否支持使用拼音查找粉丝牌/ID
    // 为false时关闭此功能
    let pinyinSwitch = true;



    if (document.querySelector(".medal-section:not(.scripted)")) {
        let controlPanelCtnrBox = document.querySelector(".medal-section:not(.scripted)");
        let template = document.createElement("div");
        template.className = "medal-section scripted";
        template.innerHTML = `<span id="medal-selector" class="dp-i-block medal"><span class="action-item medal get-medal"></span></span>`;
        for (let key in controlPanelCtnrBox.dataset) {
            template.dataset[key] = controlPanelCtnrBox.dataset[key];
        }
        controlPanelCtnrBox.after(template);
        // controlPanelCtnrBox.classList.add("origin"); // 加了没用,这个元素每次聚焦都会消失
        controlPanelCtnrBox.style.display = "none";
    }

    // 活动直播间的CSS调整
    setTimeout(() => {
        let svgIconList = document.querySelectorAll('.svg-icon');
        for (let index = 0; index < svgIconList.length && index < 10; index++) {
            const element = svgIconList[index];
            if (element) {
                let computedStyle = getComputedStyle(element);
                let backgroundImage = computedStyle.getPropertyValue('background-image');
                // 普通页面用base64覆盖了http的,活动页面还是http
                if (backgroundImage && backgroundImage.includes("http")) {
                    GM_addStyle(".des>.svg-icon{background-position:0 -6em !important}.des>.svg-icon.checkbox-selected{background-position:0 -7em !important}");
                }
                break;
            }
        }
    }, 3000);
    // 加载动画
    GM_addStyle(".medal-loading{height:30px;color:#bbb;font-size:13px;display:flex;align-items:center;justify-content:center}.medal-loading>i.icon-link-world{font-size:12px;margin-left:5px;animation:medal-loading-rotate 2s infinite}.medal-loading>i.icon-info{margin-right:5px}@keyframes medal-loading-rotate{from{transform:rotate(45deg)}to{transform:rotate(405deg)}}");
    // 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{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 0 3px #fb7299);cursor:alias}.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;cursor:pointer}.medal-wear-body .medal-item .name:hover{color:#00aeec}.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}.medal-item.outdated{opacity:.5;filter:grayscale(0.5);}");
    // 面板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}.medal-ctnr{width:268px}.medal-wear-component>.title{font-weight:400;font-size:18px;margin:0;color:#23ade5;line-height:20px}.medal-search{width:160px;margin-left:10px;position:relative;line-height:20px;top:-1px;font-size:14px;font-weight:100;border:0;padding:0;color:var(--Pi4)}.medal-search::placeholder{color:#dcdcdc}.des{cursor:pointer;color:#666;height:20px;display:flex;align-items:center}.des>.svg-icon{width:14px;height:14px;font-size:14px;background-position:0-8em;margin-right:5px}.des>span.pointer{line-height:14px}.des>.svg-icon.checkbox-selected{background-position:0-9em}.qs-icon{width:14px;height:14px;background-size:100%;background-image:url();cursor:pointer;position:relative;top:1px}.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;justify-content:space-between}.medal-wear-footer>*{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}.medal-item .bili-avatar{display:block;position:relative;background-image:url();background-size:cover;border-radius:50%;margin:0;padding:0;width:35px;height:35px}.medal-item .bili-avatar .bili-avatar-img{border:1px solid var(--line_light)}.medal-item .bili-avatar-img-radius{border-radius:50%}.medal-item .bili-avatar-img{border:0;display:block;-o-object-fit:cover;object-fit:cover;image-rendering:-webkit-optimize-contrast}.medal-item .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%}.medal-item .bili-avatar *{margin:0;padding:0}.medal-item .bili-avatar-right-icon{width:27.5%;height:27.5%;position:absolute;z-index:2;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 #f69;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}}");
    // 旧的粉丝牌样式
    GM_addStyle(".old-style.fans-medal-item{border:1px solid #fff;border-radius:2px;padding:0;height:16px}.old-style .fans-medal-label{height:100%;border-radius:0;padding:0 3px}.old-style .fans-medal-label .fans-medal-content{transform:none}.old-style .fans-medal-level{width:17px;height:14px;border-radius:0}");
    // 调用原始按钮更新粉丝牌,暂时隐藏弹窗的css
    GM_addStyle(".panel-hide .medalAb{display:none}");

    // 公用对象
    let pinyinPro = undefined;
    let my_id = document.cookie.match(/DedeUserID=(\d*); /)[1];
    let originMedalSelectorDebounce = null;
    let pinyin = pinyinSwitch;
    // 默认粉丝牌UID   如果拥有该用户的粉丝牌会在退出直播间时切换到这个粉丝牌
    // 废弃功能,不要用
    let defaultMedalUid = false;

    new Vue({
        el: '#medel_switch_box',
        async created() {
            let json = await this.getFansMedalInfo();
            let assignMedal = true;
            if (this.autoSwitch && json.has_fans_medal) {
                assignMedal = false;
                this.switchBadge(json.my_fans_medal.medal_id);
            }
            this.refreshMedalList(1, assignMedal);
            if (pinyin) {
                fetch("https://unpkg.com/pinyin-pro@3.13.0/dist/index.js")
                    .then(res => res.text(), err => { })
                    .then(js => {
                        if (!js) { console.warn("徽章切换增强-启动拼音组件失败"); return; }
                        try {
                            js = js.replace("pinyinPro", "biliSwitchBoostPinyinPro");
                            eval(js);
                            pinyinPro = window.biliSwitchBoostPinyinPro;
                            Reflect.deleteProperty(window, 'biliSwitchBoostPinyinPro');
                            this.backUpMedalWall.forEach(item => {
                                // 用户名
                                item.anchor_info.nick_pinyin = pinyinPro.pinyin(item.anchor_info.nick_name, { toneType: 'none', nonZh: 'consecutive', v: true }).replaceAll(" ", "").toLowerCase();
                                // 粉丝牌
                                item.medal.medal_pinyin = pinyinPro.pinyin(item.medal.medal_name, { toneType: 'none', nonZh: 'consecutive', v: true }).replaceAll(" ", "").toLowerCase();
                            });
                        } catch (error) {
                            console.warn("徽章切换增强-启动拼音组件错误", error);
                        }
                    });
            }
        },
        mounted: function () {
            document.querySelector("#medal-selector").onclick = () => {
                this.togglePanel();
            };
            // 如果有默认粉丝牌,则退出直播间时换上默认粉丝牌
            if (defaultMedalUid) {
                this.getFansMedalInfo(defaultMedalUid, (json, context) => {
                    if (json.has_fans_medal == false) { return; }
                    let index = this.medalWallIndex.indexOf(json.my_fans_medal.medal_id);
                    window.addEventListener('beforeunload', () => {
                        context.switchBadge(json.my_fans_medal.medal_id, index);
                    });
                });
            }
            window.addEventListener('click', e => {
                if (e.target.closest(".medal") == null && this.panelStatus) {
                    this.panelStatus = false;
                    document.querySelector(".medal-wear-body").scrollTop = 0;
                }
            });
            window.addEventListener('blur', () => { this.isTabBlur = true; });
            window.addEventListener('focus', this.refreshScriptInfo);
            // // 鼠标移动到礼物栏、弹幕输入区域时进行自动切换
            // let initMouseEventDeadLine = Date.now() + 15000;
            // (function initMouseEvent(vueInstance) {
            //     let giftDom = document.querySelector("#gift-control-vm .gift-panel");
            //     let inputDom = document.querySelector("#control-panel-ctnr-box");
            //     if (giftDom && inputDom) {
            //         giftDom.onmouseenter = inputDom.onmouseenter = () => {
            //             // console.log("徽章更换检测1", vueInstance.isTabBlur);
            //             if (vueInstance.isTabBlur) {
            //                 vueInstance.refreshScriptInfo();
            //             }
            //             // console.log("徽章更换检测2", vueInstance.isTabBlur);
            //             let cRoomMedal = vueInstance.fansMedalInfo.my_fans_medal.medal_id;
            //             if (vueInstance.autoSwitch && vueInstance.needSwitch && cRoomMedal != 0) {
            //                 // console.log("徽章触发更换");
            //                 vueInstance.switchBadge(cRoomMedal, vueInstance.medalWallIndex.indexOf(cRoomMedal));
            //                 vueInstance.needSwitch = false;
            //             }
            //         };
            //     } else if (Date.now() < initMouseEventDeadLine) {
            //         requestIdleCallback(() => {
            //             initMouseEvent();
            //         }, { timeout: 3000 });
            //     } else {
            //         console.log("切换模块初始化失败");
            //     }
            // })(this);
            this.$refs.medalList.$el.addEventListener('scroll', () => {
                let medalList = this.$refs.medalList.$el;
                let baseline = medalList.scrollHeight - medalList.offsetHeight;
                // 滚动条到最后10%的时候开始加载下一页
                if (this.pageInfo.isLastPage == false && this.pageInfo.loading == false && medalList.scrollTop > baseline - baseline * .1) {
                    this.pageInfo.loading = true;
                    this.refreshMedalList(this.pageInfo.cPage + 1);
                }
            }, false);
        },
        computed: {
            medalWallIndex: function () {
                let indexList = [];
                this.medalWall.forEach(item => {
                    indexList.push(item.medal.medal_id);
                });
                return indexList;
            },
            backUpMedalWallIndex: function () {
                let indexList = [];
                this.backUpMedalWall.forEach(item => {
                    indexList.push(item.medal.medal_id);
                });
                return indexList;
            }
        },
        data() {
            return {
                name: Date.now().toString(16) + "-" + btoa(location.host),
                fansMedalInfo: {
                    "has_fans_medal": false,
                    "my_fans_medal": {
                        "target_id": 0,
                        "medal_id": 0
                    }
                },
                currentlyWearing: {
                    medal: {
                        medal_id: 0
                    }
                },
                recentAward: {
                    medal: {
                        medal_id: 0
                    }
                },
                autoSwitch: false,
                // b站自己做了自动切换功能,所以我就不做了,这里改一下默认值防止有些人开着然后永远关不了了
                // autoSwitch: GM_getValue(`autoSwitch-${my_id}`, false),
                needSwitch: false,
                panelStatus: false,
                pageInfo: {
                    loading: false,
                    cPage: 1,
                    isLastPage: true
                },
                /* 
                    本来是用于展示的,但是牌子多加载需要翻页的情况显示的全是脏数据 
                    现在仅用于缓存存在的牌子提速自动换牌的速度
                */
                backUpMedalWall: GM_getValue(`medalWall-${my_id}`, []),
                medalWall: [],
                debounce: {},
                search: "",
                label: "",
                isTabBlur: false,   // 标识网页是否失焦
            }
        },
        watch: {
            currentlyWearing: {
                handler(val, oldVal) {
                    // 持久化用于从其他tab取出信息
                    GM_setValue(`currentlyWearing-${my_id}`, val);
                    clearTimeout(originMedalSelectorDebounce);
                    this.refreshMedal();
                    /* 
                        新旧ID相同的情况下也刷新牌子显示,因为牌子的数据可能会有变化
                        但是牌子相同的情况下不需要调用网页刷新
                    */
                    if (oldVal && val.medal.medal_id == oldVal.medal.medal_id) {
                        return;
                    }
                    // 借用原始徽章按钮来刷新当前页面上的徽章缓存
                    // 用于其他原版消息发送时展示最新的牌子(比如发表情弹幕)
                    originMedalSelectorDebounce = setTimeout(() => {
                        let originMedelBtn = document.querySelector(".medal-section:not(.scripted)>span");
                        let medelPanelParent = document.querySelector(".control-panel-ctnr-new");
                        if (!originMedelBtn || !medelPanelParent) {
                            console.log("徽章刷新失败-找不到原版按钮");
                            return;
                        }
                        originMedelBtn.click();
                        medelPanelParent.classList.add("panel-hide");
                        // 检测窗口
                        let deadLine = Date.now() + 1000;
                        let interval = setInterval(() => {
                            if (Date.now() > deadLine) {
                                clearInterval(interval);
                                medelPanelParent.classList.remove("panel-hide");
                                console.log("徽章刷新失败-无法抓取到原版徽章页面");
                            }
                            // 弹出后,点击展示设置
                            let originMedelBox = document.querySelector(".medalAb");
                            if (originMedelBox) {
                                clearInterval(interval);
                                originMedelBox.dispatchEvent(new Event('mouseleave'));
                                // 立刻执行会闪一下,观感很不好
                                setTimeout(() => { medelPanelParent.classList.remove("panel-hide"); }, 500);
                            }
                        }, 100);
                    }, 1000);
                },
                immediate: false
            },
            autoSwitch(val) {
                GM_setValue(`autoSwitch-${my_id}`, val);
                if (val) {
                    let cRoomMedal = this.fansMedalInfo.my_fans_medal.medal_id;
                    if (cRoomMedal != 0) {
                        this.switchBadge(cRoomMedal, this.medalWallIndex.indexOf(cRoomMedal));
                        this.needSwitch = false;
                    }
                }
            },
            search(val) {
                clearTimeout(this.debounce["search"]);
                let vm = this;
                this.debounce["search"] = setTimeout(function () {
                    if (val) {
                        val = val.toLowerCase().trim();
                        vm.medalWall = vm.backUpMedalWall.filter((item) => {
                            if (item.medal.medal_name.toLowerCase().includes(val) || item.anchor_info.nick_name.toLowerCase().includes(val)) {
                                item.score = 2;
                                return true;
                            }
                            if (!pinyinPro) { return false; }
                            if (item.medal.medal_pinyin.includes(val) || item.anchor_info.nick_pinyin.includes(val)) {
                                item.score = 1;
                                return true;
                            }
                            return false;
                        });
                        vm.medalWall.sort(vm.sort);
                        vm.pageInfo.isLastPage = true;
                    } else {
                        vm.medalWall = vm.backUpMedalWall.slice(0, 50);
                        vm.pageInfo.isLastPage = vm.backUpMedalWall.length < 50;
                    }
                }, 200);
            }
        },
        methods: {
            /* 将10进制数字转为16进制字符串,不足6位时自动补充 */
            padToHex(str) {
                return str.toString(16).padStart(6, 0);
            },
            async sleep(ms) {
                return new Promise(r => {
                    setTimeout(() => {
                        r(true);
                    }, ms);
                });
            },
            async getCurrentWear() {    // 获取当前佩戴粉丝牌
                let res = await fetch(`https://api.live.bilibili.com/xlive/app-ucenter/v1/fansMedal/panel?page=1&page_size=1`, { credentials: 'include', });
                let json = await res.json();
                if (json.code == json.message) {
                    for (const item of json.data.special_list) {
                        if (item.superscript == null) {
                            this.currentlyWearing = item;
                            break;
                        }
                    }
                    return;
                }
                warn("获取当前佩戴失败:", json.message);
            },
            async getFansMedalInfo(uid, callback) {  // 用来获取是否拥有指定用户的粉丝牌
                let muid = undefined;
                if (!uid) {
                    muid = 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) {
                    // 仅在获取当前房间信息时赋值
                    if (muid == uid) {
                        this.fansMedalInfo = json.data;
                    }
                    if (callback) {
                        // 存在回调的情况下异步执行
                        (async () => {
                            callback(json.data, this);
                        })();
                    }
                    return json.data;
                }
                alert("徽章初始化失败:", json.message);
            },
            async refreshMedalList(page = 1, assignMedal = true) {
                let uid = await ROOM_INFO_API.getUid();
                return new Promise((resolve, reject) => {
                    fetch(`https://api.live.bilibili.com/xlive/app-ucenter/v1/fansMedal/panel?page=${page}&page_size=50&target_id=${uid}`, { credentials: 'include', })
                        .then(res => res.json())
                        .then(json => {
                            if (json.code == json.message) {
                                /* 
                                    刷新当前佩戴的徽章
                                    special_list的内容不会超过3条,所以两次循环无所谓
                                */
                                if (page == 1 && assignMedal) {     // 只有第一页special_list才会有值
                                    // 防止在其他地方取消牌子后插件无反应
                                    this.currentlyWearing = { medal: { medal_id: 0 } };
                                }
                                for (let item of json.data.special_list) {
                                    // 抓取当前佩戴
                                    // 2023年7月26日 item.medal.wearing_status不总是准确的
                                    if (assignMedal && item.medal.wearing_status) {
                                        this.currentlyWearing = item;
                                        continue;
                                    }
                                    // 抓取最近获取
                                    if (item.superscript && item.superscript.type == 2) {
                                        this.recentAward = item;
                                        continue;
                                    }
                                }

                                if (page == 1) {
                                    this.label = Math.floor(Math.random() * 2 ** 32);
                                }
                                // 合并列表并排序
                                let list = [].concat(json.data.list, json.data.special_list);
                                list.forEach((item) => {
                                    // 添加标识
                                    item.label = this.label;
                                    // 解析拼音
                                    if (pinyinPro) {
                                        // 用户名
                                        item.anchor_info.nick_pinyin = pinyinPro.pinyin(item.anchor_info.nick_name, { toneType: 'none', nonZh: 'consecutive', v: true }).replaceAll(" ", "").toLowerCase();
                                        // 粉丝牌
                                        item.medal.medal_pinyin = pinyinPro.pinyin(item.medal.medal_name, { toneType: 'none', nonZh: 'consecutive', v: true }).replaceAll(" ", "").toLowerCase();
                                    }
                                    let index = this.medalWallIndex.indexOf(item.medal.medal_id);
                                    if (index >= 0) {
                                        this.$set(this.medalWall, index, item);
                                    } else {
                                        this.medalWall.push(item);
                                    }
                                    // backUpMedalWall
                                    index = this.backUpMedalWallIndex.indexOf(item.medal.medal_id);
                                    item.superscript = null;    // 防止搜索出现脏数据
                                    if (index >= 0) {
                                        this.$set(this.backUpMedalWall, index, item);
                                    } else {
                                        this.backUpMedalWall.push(item);
                                    }
                                });
                                this.medalWall.sort(this.sort);
                                this.backUpMedalWall.sort(this.sort);
                                // 保存页面数据
                                this.pageInfo.loading = false;
                                this.pageInfo.cPage = +json.data.page_info.current_page;
                                this.pageInfo.isLastPage = page >= json.data.page_info.total_page;
                                // 获取完整列表后替换旧列表
                                if (this.pageInfo.isLastPage) {
                                    this.backUpMedalWall = this.medalWall;
                                }
                                GM_setValue(`medalWall-${my_id}`, this.backUpMedalWall);
                                // 返回是否有下一页
                                resolve(json.data.page_info.has_more && page < json.data.page_info.total_page);
                            } else {
                                reject(false);
                            }
                        })
                        .catch(err => {
                            reject();
                        });
                });
            },
            async switchBadge(badgeId, index) {
                let jct = document.cookie.match(/bili_jct=(\w*); /)[1]
                let params = new URLSearchParams();
                params.set("medal_id", badgeId);
                params.set("csrf_token", jct);
                params.set("csrf", 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 >= 0) {
                    this.currentlyWearing = this.medalWall[index];
                } else {
                    let result = this.backUpMedalWall.find(item => {
                        return badgeId == item.medal.medal_id;
                    });
                    if (result) {
                        this.currentlyWearing = result;
                    } else {
                        console.warn("徽章列表内找不到对应的徽章");
                        await this.getCurrentWear();
                    }
                }
                // 佩戴时更新为最新状态
                // todo 现有接口不能更新 头像、直播状态
                // if (this.currentlyWearing.label != this.label) {
                //     this.getFansMedalInfo(this.currentlyWearing.medal.target_id, (data) => {
                //         this.medalWall[index].label = this.label;
                //         this.$set(this.medalWall[index], "medal", data.my_fans_medal);
                //     });
                // }
                // 仅主动切换才保存操作人
                GM_setValue(`operator-${my_id}`, this.name);
            },
            takeOff() {
                this.currentlyWearing = { medal: { medal_id: 0 } };
                let jct = document.cookie.match(/bili_jct=(\w*); /)[1]
                let params = new URLSearchParams();
                params.set("visit_id", '');
                params.set("csrf_token", jct);
                params.set("csrf", jct);
                fetch("https://api.live.bilibili.com/xlive/web-room/v1/fansMedal/take_off", {
                    "method": "POST",
                    "credentials": "include",
                    "body": params,
                });
                // 仅主动切换才保存操作人
                GM_setValue(`operator-${my_id}`, this.name);
            },
            openConfig() {
                // 点击原版
                let originMedelBtn = document.querySelector(".medal-section:not(.scripted)>span");
                if (!originMedelBtn) {
                    console.log("展示设置打开失败-找不到原版按钮");
                    return;
                }
                originMedelBtn.click();
                // 检测窗口
                let deadLine = Date.now() + 1000;
                let interval = setInterval(() => {
                    if (Date.now() > deadLine) {
                        clearInterval(interval);
                        console.log("展示设置打开失败-无法抓取到原版徽章页面");
                    }
                    // 弹出后,点击展示设置
                    let originMedelBox = document.querySelector(".medalAb");
                    if (originMedelBox) {
                        clearInterval(interval);
                        let configBtn = document.querySelector(".medalAb .cancel-wear");
                        if (configBtn) {
                            configBtn.click();
                        } else {
                            console.log("展示设置打开失败-找不到设置按钮");
                        }
                    }
                }, 100);
            },
            openSpace: (uid) => {
                window.open(`//space.bilibili.com/${uid}`);
            },
            openRoom: (rid) => {
                window.open(`//live.bilibili.com/${rid}`);
            },
            togglePanel() {
                clearTimeout(this.debounce["panel"]);
                this.panelStatus = !this.panelStatus;
                if (!this.panelStatus) {
                    this.debounce["panel"] = setTimeout(() => {
                        this.debounce["panel"] = null;
                        // 临时处理,防止第二次打开后未翻页脏数据
                        this.medalWall = this.backUpMedalWall.slice(0, 50);
                        this.search = "";
                    }, 2000);
                } else {
                    if (this.debounce["panel"]) { return; }
                    // 刷新本房间粉丝牌状态
                    if (this.fansMedalInfo.has_fans_medal == false) {
                        this.getFansMedalInfo();
                    }
                    this.$nextTick(() => {
                        // 只能在nexttick里面不然元素处于display:none时无法起作用
                        document.querySelector(".medal-wear-body").scrollTop = 0;
                        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 none-select old-style medal-item-margin"
                            style="border-color:#${this.padToHex(this.currentlyWearing.medal.medal_color_border)}">
                            <div class="fans-medal-label"
                                style="background-image:linear-gradient(45deg,#${this.padToHex(this.currentlyWearing.medal.medal_color_start)},#${this.padToHex(this.currentlyWearing.medal.medal_color_end)})">
                                <span class="fans-medal-content">${this.currentlyWearing.medal.medal_name}</span>
                            </div>
                            <div class="fans-medal-level" style="color:#${this.padToHex(this.currentlyWearing.medal.medal_color_start)}">${this.currentlyWearing.medal.level}</div>
                        </div>
                    `;
                } else {
                    selector.innerHTML = `<span class="action-item medal get-medal"></span>`;
                }
            },
            sort(a, b) {
                // 搜索匹配度
                // if (this.search && (a.score || b.score) && a.score != b.score) {
                //     return a.score > b.score ? -1 : 1;
                // }
                // 当前房间
                if (a.medal.target_id == this.fansMedalInfo.my_fans_medal.target_id) {
                    return -1;
                } else if (b.medal.target_id == this.fansMedalInfo.my_fans_medal.target_id) {
                    return 1;
                }
                // 当前佩戴 
                if (a.medal.medal_id == this.currentlyWearing.medal.medal_id) {
                    return -1;
                } else if (b.medal.medal_id == this.currentlyWearing.medal.medal_id) {
                    return 1;
                }
                // 最近获得、其他特殊情况
                if (a.medal.medal_id == this.recentAward.medal.medal_id) {
                    return -1;
                } else if (b.medal.medal_id == this.recentAward.medal.medal_id) {
                    return 1;
                }
                // 如果不是此次获取的牌子,向后靠
                if (a.label != this.label) {
                    return 1;
                } else if (b.label != this.label) {
                    return -1;
                }
                // 灰色牌子
                if (a.medal.is_lighted == 0) {
                    return 1;
                } else if (b.medal.is_lighted == 0) {
                    return -1;
                }
                // 等级排序
                if (a.medal.level != b.medal.level) {
                    return b.medal.level - a.medal.level;
                }
                // 经验排序
                return b.medal.intimacy - a.medal.intimacy;
            },
            refreshScriptInfo(event) {
                // console.log("插件状态更新", this.isTabBlur);
                this.isTabBlur = false;
                // 获取最新状态
                // this.autoSwitch = GM_getValue(`autoSwitch-${my_id}`, false);
                // 如果当前页面展示的粉丝牌不是实际佩戴的粉丝牌,那么更新显示
                let wearing = GM_getValue(`currentlyWearing-${my_id}`);
                if (wearing && this.currentlyWearing.medal.medal_id != wearing.medal.medal_id) {
                    this.currentlyWearing = wearing;
                }
                if (wearing && this.name != GM_getValue(`operator-${my_id}`) && this.fansMedalInfo.my_fans_medal.medal_id != wearing.medal.medal_id) {
                    this.needSwitch = true;
                } else {
                    this.needSwitch = false;
                }
            }
        },
        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>
                        <input class="medal-search" placeholder="搜索粉丝牌" v-model="search">
                        <div class="dp-i-block des f-right" @click="autoSwitch = !autoSwitch" style="display:none">
                            <span class="cb-icon svg-icon v-middle" :class="{'checkbox-selected':autoSwitch}"></span>
                            <span class="pointer v-middle">自动更换</span>
                        </div>
                        <transition-group name="medal-list" tag="div" class="medal-wear-body" ref="medalList">
                            <div class="medal-item" v-for="(item,index) in medalWall" :class="{ outdated: item.label != label }"
                                :key="item.medal.medal_id" :data-uid="item.medal.target_id" :data-rid="item.room_info.room_id" 
                                :data-uname="item.anchor_info.nick_name.toLowerCase()" :data-mname="item.medal.medal_name.toLowerCase()"
                                @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">
                                        <a :href="'//live.bilibili.com/' + item.room_info.room_id" target="blank" @click.stop="" class="search-user-avatar p_relative avatar-small mr_md cs_pointer">
                                            <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>
                                        </a>
                                    </template>
                                    <template v-else>
                                        <a :href="'//live.bilibili.com/' + item.room_info.room_id" target="blank" class="face" @click.stop="">
                                            <img :src="item.anchor_info.avatar">
                                            <span class="bili-avatar-right-icon" v-if="item.anchor_info.verify == 0"></span>
                                        </a>
                                    </template>
                                    <div class="dp-i-block v-bottom w-100 p-relative">
                                        <div class="medal-content-head">
                                            <div class="fans-medal-item none-select old-style f-right"
                                                :style="'border-color:#'+(padToHex(item.medal.medal_color_border))">
                                                <div class="fans-medal-label"
                                                    :style="'background-image:linear-gradient(45deg,#'+(padToHex(item.medal.medal_color_start))+',#'+(padToHex(item.medal.medal_color_end))+')'">
                                                    <span class="fans-medal-content">{{item.medal.medal_name}}</span>
                                                </div>
                                                <div class="fans-medal-level"
                                                    :style="'color:#'+(padToHex(item.medal.medal_color_start))">
                                                    {{item.medal.level}}
                                                </div>
                                            </div>
                                            <a :href="'//space.bilibili.com/' + item.medal.target_id" target="blank" @click.stop="" class="name dp-i-block">{{item.anchor_info.nick_name}}</a>
                                        </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.medal.medal_id == recentAward.medal.medal_id">
                                                    最近获得
                                                    <!-- {{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>
                            <div class="medal-loading" key="medal-loading">
                                <template v-if="!pageInfo.isLastPage">
                                    正在加载<i class="v-middle icon-font icon-link-world"></i>
                                </template>
                                <template v-else>
                                    <i class="v-middle icon-font icon-info"></i>没有了
                                </template>
                            </div>
                        </transition-group>
                        <div class="footer-line"></div>
                        <div class="dp-flex medal-wear-footer">
                            <span class="dp-i-block cancel-wear" @click="takeOff">
                                不佩戴勋章
                            </span>
                            <span class="dp-i-block display-config" @click="openConfig">
                                展示设置
                            </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 spinInit(loadHook, timeout) {
    return new Promise((resolve, reject) => {
        let startTime = Date.now();
        let checkInterval = setInterval(() => {
            if (Date.now() - startTime >= timeout) {
                clearInterval(checkInterval);
                reject("徽章切换增强-无法找到加载点");
            }
            // console.log(`徽章切换增强-正在寻找加载点`);
            let hooked = false;
            for (const prop in loadHook) {
                // console.log(`徽章切换增强-加载点[${prop}]`);
                if (typeof loadHook[prop] === 'function') {
                    hooked = loadHook[prop]();
                    if (hooked) {
                        console.log(`徽章切换增强-加载成功[${prop}]`);
                        clearInterval(checkInterval);
                        resolve();
                        break;
                    }
                }
            }
        }, 300);
    });
}

if (!document.cookie.match(/bili_jct=(\w*); /)) { return; } // 未登录就撤

spinInit({
    ">0.0.1": function () {  // 旧版锚点
        let bottomBox = document.querySelector(".bottom-actions");
        if (bottomBox) {
            // 列表元素
            let tempElement = document.createElement("div");
            tempElement.id = "medel_switch_box";
            bottomBox.after(tempElement);
            // 旧版适用的插件粉丝牌框框CSS
            GM_addStyle(".dialog-ctnr.medal{bottom:100px;left:-1px}");
            return true;
        }
        return false;
    },
    ">1.2.8": function () {  // B站新版前端(SC和点赞独立大按钮)锚点
        let bottomBox = document.querySelector(".icon-left-part-new");
        if (bottomBox) {
            // 列表元素
            let tempElement = document.createElement("div");
            tempElement.id = "medel_switch_box";
            bottomBox.after(tempElement);
            // 原版的按钮CSS调整
            GM_addStyle(".medalAb .close{display:none}.medal-section:not(.scripted){display:none !important;}.chat-input-ctnr-new .medal-section{padding: 5px !important;}.chat-input-focus .medal-section{display:none !important}");
            GM_addStyle(".dialog-ctnr.medal{bottom:-20px;left:-8px}");  // 插件粉丝牌框框CSS
            return true;
        }
        return false;
    }
}, 7000)
    .then((result) => main());