Greasy Fork

Greasy Fork is available in English.

天选红包

某站天选红包,只抽关注无要求

// ==UserScript==
// @name         天选红包
// @version      1.05
// @description  某站天选红包,只抽关注无要求
// @author       无夏不春风orz
// @iconURL      https://www.bilibili.com/favicon.ico
// @match        https://live.bilibili.com/*
// @match        https://www.bilibili.com/blackboard/live/*
// @connect      bilibili.com
// @connect      150.158.35.26
// @require      https://unpkg.com/[email protected]/dist/jquery.js
// @require      https://unpkg.com/[email protected]/crypto-js.js
// @grant        unsafeWindow
// @grant        GM_xmlhttpRequest
// @run-at       document-idle
// @license      MIT
// @namespace    http://tampermonkey.net/
// ==/UserScript==


(function (){
    var tipping_pic = ""
    var Toast_out = false
    var NAME
    var BAPI
    var Live_info = {
        coin: undefined,
        room_id: undefined,
        uid: undefined,
        csrf_token: undefined,
        rnd: undefined,
        ruid: undefined,
        uname: undefined,
        user_level: undefined,
        Blever: undefined,
        vipType: undefined,
        face_url: undefined,
        vipTypetext: undefined,
        cost: undefined,
        regtime: undefined,
        identification: undefined,
        img_key:undefined,
        sub_key:undefined,
    };
    const prettyLog = (...args) => {
        const styles = 'color: blue; font-weight: bold; background: #ddd';
        console.log(`%c${args.join(' ')}`, styles);
    };

    const dateNow = () => Date.now();
    const ts_ms = () => Date.now();
    const ts_s = () => Math.round(ts_ms() / 1000);
    function sleep(ms){
        return new Promise(resolve => setTimeout(() => resolve('sleep'), ms));
    }
    String.prototype.replaceAll = function(oldSubStr, newSubStr){
        return this.replace(new RegExp(oldSubStr, 'gm'), newSubStr)
    }
    const getCookie = (name) => {
        let arr = document.cookie.match(new RegExp("(^| )" + name + "=([^;]*)(;|$)"));
        if (arr != null)
            return unescape(arr[2]);
        return false;
    }
    const newWindow = {
        init: () => {
            return newWindow.Toast.init();
        },
        Toast: {
            init: () => {
                try {
                    const list = [];
                    window.toast = (msg, type = 'info', timeout = 5e3) => {
                        if(msg.indexOf("当前操作异常") > -1){
                            msg = msg.replaceAll("请升级至最新版本后重试","使用手机端通过验证码后再试")
                        }
                        if(msg.indexOf("需要进行验证码校验") > -1){
                            msg = msg.replaceAll("需要进行验证码校验","进入抽奖直播间完成验证码校验后可恢复正常")
                        }
                        prettyLog(`【${new Date(ts_s()*1000).toLocaleString()}】${msg}`)
                        switch (type){
                            case 'success':
                            case 'info':
                            case 'error':
                                break;
                            default:
                                type = 'info';
                        }
                        const a = $(`<div class="link-toast ${type} fixed" style="z-index:2001;"><span class="toast-text">${msg}</span></div>`)[0];
                        if(!Toast_out)document.body.appendChild(a);
                        if(Toast_out){
                            list.forEach((v) => {
                                $(v).remove();
                            });
                            return $.Deferred().resolve();
                        }
                        a.style.top = (document.body.scrollTop + list.length * 40 + 50) + 'px';
                        a.style.left = 10 + 'px';
                        list.push(a);
                        setTimeout(() => {
                            a.className += ' out';
                            setTimeout(() => {
                                list.shift();
                                list.forEach((v) => {
                                    v.style.top = (parseInt(v.style.top, 10) - 40) + 'px';
                                });
                                $(a).remove();
                            }, 200);
                        }, timeout);
                    };
                    return $.Deferred().resolve();
                } catch (err){
                    return $.Deferred().reject();
                }
            }
        }
    }
    $(function () { //DOM完毕,等待弹幕加载完成
        if((typeof BilibiliLive) == "undefined"){
            BilibiliLive = undefined;
        }
        let loadInfo = (delay) => {
            setTimeout(async function () {
                GM_xmlhttpRequest({
                    method: "GET",
                    url: "https://api.bilibili.com/x/web-interface/nav",
                    onload: async function(response) {
                        let data = JSON.parse(response.response);
                        if (!data.data.isLogin) {
                            loadInfo(5000);
                            window.toast('无账号登陆信息,请先登录或检查网络','error',8000);
                        } else {
                            if(BilibiliLive == undefined || BilibiliLive.ROOMID == undefined) return loadInfo(5000);
                            Live_info.room_id = BilibiliLive.ROOMID;
                            Live_info.ruid = BilibiliLive.ANCHOR_UID
                            Live_info.uid = data.data.mid
                            Live_info.coin = parseInt(data.data.money)
                            Live_info.Blever = data.data.level_info.current_level
                            Live_info.vipType = data.data.vipType
                            Live_info.vipStatus = data.data.vipStatus
                            Live_info.uname = data.data.uname
                            Live_info.face_url = data.data.face
                            Live_info.vipTypetext = data.data.vip_label.text
                            if(Live_info.vipTypetext=='')Live_info.vipTypetext = '普通用户'
                            let img_url = data.data.wbi_img.img_url
                            let sub_url = data.data.wbi_img.sub_url
                            let img_key = img_url.slice(img_url.lastIndexOf('/') + 1,img_url.lastIndexOf('.'))
                            let sub_key = sub_url.slice(sub_url.lastIndexOf('/') + 1,sub_url.lastIndexOf('.'))
                            Live_info.img_key = img_key
                            Live_info.sub_key = sub_key
                            NAME = Live_info.uid
                            Live_info.csrf_token = getCookie('bili_jct');
                            Live_info.buvid3 = getCookie('buvid3');
                            window.toast('登陆信息获取成功','success');
                            init();
                        }
                    },
                    onerror : function(err){
                        loadInfo(5000);
                        window.toast('无账号登陆信息,请先登录或检查网络','error',8000);
                    }
                });
            }, delay);
        };
        newWindow.init();
        loadInfo(5000);
    });

    function init(){ //API初始化
        BAPI = BilibiliAPI;
        const MY_API = {
            CONFIG_DEFAULT: {
                auto_like: true, //点赞直播间
                check_newday_ts:0,
                js_running_mark: 0, //多开标记
                auto_gift_anchor: false, //礼物天选
                popularity_red_pocket_join_switch:false,//人气红包
                done_id_list:[],
                popularity_red_pocket_done_id_list:[],
                gift_total_price_min:30,//电池
                popularity_red_pocket_flash:60,//道具红包抽奖间隔
                gift_anchor_flash:30,//礼物天选抽奖间隔
                popularity_red_pocket_onlineNum_switch:true,//红包在线人数开关
                popularity_red_pocket_onlineNum:50,//红包在线人数上限
                bagsendonekey_room:0,
                Toast_out: false, //弹出信息
                gift_anchor_pass_giftname:[],
                join_code_check: false, //出验证码暂停
            },

            CONFIG: {},
            init: async function () {
                let p = $.Deferred();
                try {
                    MY_API.loadConfig().then(async function () {
                        window.toast('正在初始化脚本....', 'success');
                        window.toast('脚本参数配置完成!', 'success');
                        p.resolve();
                    });
                } catch (e){
                    prettyLog('API初始化出错', e);
                    window.toast('脚本初始化出错', 'error');
                    p.reject()
                }
                return p
            },
            loadConfig: async function () {
                let p = $.Deferred();
                try {
                    let config = JSON.parse(localStorage.getItem(`${NAME}_TXHBCONFIG`));
                    $.extend(true, MY_API.CONFIG, MY_API.CONFIG_DEFAULT);
                    for(let item in MY_API.CONFIG){
                        if(!MY_API.CONFIG.hasOwnProperty(item))
                            continue;
                        if(config[item] !== undefined && config[item] !== null)
                            MY_API.CONFIG[item] = config[item];
                    }
                    p.resolve()
                } catch (e){
                    prettyLog('API载入配置失败,加载默认配置', e);
                    MY_API.setDefaults();
                    p.reject()
                }
                return p
            },

            saveConfig: function () {
                try {
                    localStorage.setItem(`${NAME}_TXHBCONFIG`, JSON.stringify(MY_API.CONFIG));
                    //prettyLog('配置已保存', MY_API.CONFIG);
                    return true
                } catch (e){
                    prettyLog('API保存出错', e);
                    return false
                }
            },
            setDefaults: async function () {
                window.toast(`未检测到配置信息,正在载入脚本默认配置!`, 'error');
                MY_API.CONFIG = MY_API.CONFIG_DEFAULT;
                MY_API.saveConfig();
                window.toast(`将在5秒后自动刷新!`, 'error');
                setTimeout(() => {
                    window.location.reload()
                }, 5000);
            },
            creatSetBox: function() {
                // Create settings box with the requested UI style
                const heightmax = $('.chat-history-panel').height();
                const bagsendbox = $("<div class='bagsendbox'>").css({
                    'width': '480px',
                    'min-width': '480px',
                    'height': '580px',
                    'max-height': `${heightmax}px`,
                    'position': 'absolute',
                    'top': '10px',
                    'right': '10px',
                    'background': '#fff',
                    'padding': '15px',
                    'z-index': '99',
                    'border-radius': '8px',
                    'box-shadow': '0 2px 10px rgba(0,0,0,0.1)',
                    'transition': 'all .3s ease',
                    'overflow': 'auto',
                    'line-height': '1.2',
                    'font-family': 'system-ui, -apple-system, sans-serif',
                    'color': '#333'
                })

                // HTML template with the requested styling
                bagsendbox.append(`
<div data-toggle="tipping">
     <button class="setting-button toggle-button" data-action="save">点击微信赞赏报销监控服务器费用</button>
</div>
<div id="circular-image">
     <img src="${tipping_pic}" alt="圆形图片">
</div>
<fieldset class="settings-fieldset" style="margin-top:10px;">
    <legend class="settings-legend" style="color:#06aed5;font-weight:bold;">当前包裹状态</legend>
    <div class="user-details">
        <div class="setting-label" id="gift_count">包裹电池数:0</div>
        <div class="setting-label" id="gift_num">包裹礼物数:0</div>
    </div>
</fieldset>

<fieldset class="settings-fieldset">
    <legend class="settings-legend" style="color:#06aed5;font-weight:bold;">礼物天选抽奖</legend>
    <div class="setting-item" data-toggle="auto_gift_anchor">
        <label class="checkbox-container">
            <input type="checkbox">
            <span class="checkmark"></span>
            <span class="setting-label">开启礼物天选抽奖</span>
        </label>
        <div style="margin-top:10px;">
            <span class="setting-label">抽奖间隔:</span>
            <input class="setting-input num" type="text">秒
            <button class="setting-button" data-action="save">保存</button>
        </div>

        <div style="margin-top:10px;">
            <span class="setting-label">屏蔽天选礼物:</span>
            <input class="setting-input keyword num1" type="text">
            <button class="setting-button" data-action="save1">保存</button>
        </div>
   </div>
        <div class="setting-item" data-toggle="join_code_check" style="margin-top:10px;">
            <label class="checkbox-container">
                <input type="checkbox">
                <span class="checkmark"></span>
                <span class="setting-label">天选出验证码暂停抽奖</span>
            </label>
        </div>
</fieldset>

<fieldset class="settings-fieldset">
    <legend class="settings-legend" style="color:#06aed5;font-weight:bold;">人气红包抽奖</legend>
    <div class="setting-item" data-toggle="popularity_red_pocket_join_switch">
        <label class="checkbox-container">
            <input type="checkbox">
            <span class="checkmark"></span>
            <span class="setting-label">开启人气红包抽奖</span>
        </label>

        <div  style="margin-top:10px;">
            <span class="setting-label">红包参与电池下限:</span>
            <input class="setting-input num1" type="text">电池
            <button class="setting-button" data-action="save1">保存</button>
        </div>

        <div style="margin-top:10px;">
            <span class="setting-label">抽奖间隔:</span>
            <input class="setting-input num" type="text">秒
            <button class="setting-button" data-action="save">保存</button>
        </div>
    </div>

    <div class="setting-item" data-toggle="popularity_red_pocket_onlineNum_switch">
        <label class="checkbox-container" style="margin-top:10px;">
            <input type="checkbox">
            <span class="checkmark"></span>
            <span class="setting-label">开启在线人数过滤</span>
        </label>

        <div  style="margin-top:10px;">
            在线人数超过<input class="setting-input num" type="text">时不参加
            <button class="setting-button" data-action="save">保存</button>
        </div>
    </div>
</fieldset>

<fieldset class="settings-fieldset">
    <legend class="settings-legend">其他设置</legend>
    <div class="setting-item" data-toggle="auto_like" style="margin-top:10px;">
        <label class="checkbox-container">
            <input type="checkbox" title="抽奖直播间自动点赞">
            <span class="checkmark"></span>
            <span class="setting-label">抽奖直播间自动点赞</span>
        </label>
    </div>
    <div class="setting-item" data-toggle="bagsendonekey" style="margin-top:10px;">
        <div style="margin-top:10px;">
            <span class="setting-label">直播间号:</span>
            <input class="setting-input num" type="text" style="width: 150px;">
            <button class="setting-button" data-action="save">一键投喂</button>
        </div>

        <div class="setting-item" style="margin-top:10px;">
            <label class="checkbox-container">
                <input type="checkbox">
                <span class="checkmark"></span>
                <span class="setting-label">隐藏弹出信息</span>
            </label>
        </div>

        <div class="settings-note" style="margin-top:10px;">
            <ul>
                <li>可按F12点Console查看历史信息</li>
                <li>提示异常或多日不中奖,大概率是触发抽奖风控了</li>
                <li>可休养一段时间后再抽,具体休养时间各不相同</li>
            </ul>
        </div>
    </div>
</fieldset>
`)

                // 添加CSS样式
                $('head').append(`
        <style>
        #circular-image {
            width: 200px;
            height: 200px;
            border-radius: 50%;
            overflow: hidden;
            display: block;
            margin: 0 auto;
        }
        #circular-image img {
            width: 100%;
            height: 100%;
            object-fit: cover;
        }
            .bagsendbox {
                scrollbar-width: thin;
                scrollbar-color: #06aed5 rgba(0,0,0,0.1);
            }
            .bagsendbox::-webkit-scrollbar {
                width: 6px;
            }
            .bagsendbox::-webkit-scrollbar-thumb {
                background-color: #06aed5;
                border-radius: 3px;
            }
            .settings-fieldset {
                border: 1px solid #ddd;
                border-radius: 8px;
                margin-bottom: 5px;
                padding: 10px;
            }
            .settings-legend {
                color: #06aed5;
                font-weight: bold;
                padding: 0 8px;
                font-size: 14px;
            }
            .user-info {
                display: flex;
                gap: 10px;
                align-items: flex-start;
            }
            .user-avatar {
                border-radius: 50%;
                border: 2px solid #06aed5;
            }
            .user-details {
                display: grid;
                grid-template-columns: 1fr 1fr;
                gap: 4px;
                font-size: 12px;
            }
            .detail-label {
                color: #666;
                font-weight: bold;
            }
            .setting-item {
                margin-bottom: 2px;
                padding: 0px 0;
                border-bottom: 1px dashed #eee;
            }
            .setting-item:last-child {
                border-bottom: none;
            }
            .checkbox-container {
                display: flex;
                align-items: center;
                cursor: pointer;
                position: relative;
                user-select: none;
            }
            .checkbox-container input {
                position: absolute;
                opacity: 0;
                cursor: pointer;
                height: 0;
                width: 0;
            }
            .checkmark {
                position: relative;
                height: 18px;
                width: 18px;
                background-color: #fff;
                border: 1px solid #06aed5;
                border-radius: 4px;
                margin-right: 8px;
            }
            .checkbox-container input:checked ~ .checkmark {
                background-color: #06aed5;
            }
            .checkmark:after {
                content: "";
                position: absolute;
                display: none;
            }
            .checkbox-container input:checked ~ .checkmark:after {
                display: block;
            }
            .checkbox-container .checkmark:after {
                left: 6px;
                top: 2px;
                width: 5px;
                height: 10px;
                border: solid white;
                border-width: 0 2px 2px 0;
                transform: rotate(45deg);
            }
            .setting-label {
                font-size: 14px;
                color: #333;
            }
            .setting-input {
                padding: 6px 8px;
                border: 1px solid #ddd;
                border-radius: 4px;
                font-size: 14px;
                margin: 0 5px;
                width: 50px;
            }
            .setting-input.keyword {
                width: 220px;
                margin-right: 8px;
            }
            .setting-button {
                background-color: #06aed5;
                color: white;
                border: none;
                padding: 6px 12px;
                border-radius: 4px;
                cursor: pointer;
                font-size: 13px;
                transition: all 0.2s;
            }
            .setting-button:hover {
                background-color: #0589b0;
            }
            .settings-note {
                font-size: 12px;
                color: #666;
                margin-top: 15px;
                background: #f8f8f8;
                padding: 0px;
                border-radius: 6px;
            }
            .settings-note ul {
                margin: 5px 0 0 15px;
                padding: 0;
            }
            .settings-note li {
                margin-bottom: 3px;
            }
            .toggle-button {
                display: block;
                width: 100%;
                margin-top: 10px;
            }
            .toggle-button:hover {
                background-color: #0589b0;
            }
        </style>
    `);
                $('.player-section.p-relative.border-box.none-select.z-player-section').append(bagsendbox);
                $('.bagsendbox').hide();

                // 隐藏/显示按钮事件
                document.getElementById('circular-image').style.display = 'none'
                bagsendbox.find('div[data-toggle="tipping"] [data-action="save"]').click(function () {
                    const image = document.getElementById('circular-image');
                    if (image.style.display === 'none') {
                        image.style.display = 'block';
                    } else {
                        image.style.display = 'none';
                    }
                });
                // Initialize checkbox states
                const initCheckbox = (toggle, configKey) => {
                    if (MY_API.CONFIG[configKey]) {
                        bagsendbox.find(`div[data-toggle="${toggle}"] input`).prop('checked', true);
                    }
                    bagsendbox.find(`div[data-toggle="${toggle}"] input:checkbox`).change(function() {
                        MY_API.CONFIG[configKey] = $(this).prop('checked');
                        MY_API.saveConfig();
                        window.toast(`设置${MY_API.CONFIG[configKey] ? '已开启' : '已关闭'}`);
                    });
                };

                // Initialize checkboxes
                initCheckbox('join_code_check', 'join_code_check');
                initCheckbox('auto_like', 'auto_like');
                initCheckbox('auto_gift_anchor', 'auto_gift_anchor');
                initCheckbox('popularity_red_pocket_join_switch', 'popularity_red_pocket_join_switch');
                initCheckbox('popularity_red_pocket_onlineNum_switch', 'popularity_red_pocket_onlineNum_switch');

                // Initialize input values
                bagsendbox.find('div[data-toggle="auto_gift_anchor"] .num').val(parseInt(MY_API.CONFIG.gift_anchor_flash));
                bagsendbox.find('div[data-toggle="auto_gift_anchor"] .num1').val(MY_API.CONFIG.gift_anchor_pass_giftname.join(', '));
                bagsendbox.find('div[data-toggle="popularity_red_pocket_join_switch"] .num1').val(parseInt(MY_API.CONFIG.gift_total_price_min));
                bagsendbox.find('div[data-toggle="popularity_red_pocket_join_switch"] .num').val(parseInt(MY_API.CONFIG.popularity_red_pocket_flash));
                bagsendbox.find('div[data-toggle="popularity_red_pocket_onlineNum_switch"] .num').val(parseInt(MY_API.CONFIG.popularity_red_pocket_onlineNum));
                bagsendbox.find('div[data-toggle="bagsendonekey"] .num').val(parseInt(MY_API.CONFIG.bagsendonekey_room));

                // Toast out checkbox
                if (MY_API.CONFIG.Toast_out) {
                    bagsendbox.find('div[data-toggle="bagsendonekey"] input').prop('checked', true);
                }
                Toast_out = MY_API.CONFIG.Toast_out;
                bagsendbox.find('div[data-toggle="bagsendonekey"] input:checkbox').change(function() {
                    MY_API.CONFIG.Toast_out = $(this).prop('checked');
                    MY_API.saveConfig();
                    Toast_out = MY_API.CONFIG.Toast_out;
                    window.toast(`提示信息显示:${MY_API.CONFIG.Toast_out ? '已隐藏' : '已开启'}`);
                });

                // Save button handlers
                bagsendbox.find('[data-action="save"]').click(function() {
                    const parentDiv = $(this).closest('div[data-toggle]');
                    const toggle = parentDiv.attr('data-toggle');

                    switch (toggle) {
                        case 'auto_gift_anchor':
                            MY_API.CONFIG.gift_anchor_flash = parseInt(parentDiv.find('.num').val());
                            window.toast(`礼物天选抽奖间隔设置为:${MY_API.CONFIG.gift_anchor_flash}秒`);
                            break;
                        case 'popularity_red_pocket_join_switch':
                            MY_API.CONFIG.popularity_red_pocket_flash = parseInt(parentDiv.find('.num').val());
                            window.toast(`人气红包抽奖间隔设置为:${MY_API.CONFIG.popularity_red_pocket_flash}秒`);
                            break;
                        case 'popularity_red_pocket_onlineNum_switch':
                            MY_API.CONFIG.popularity_red_pocket_onlineNum = parseInt(parentDiv.find('.num').val());
                            window.toast(`人气红包抽奖在线人数限制设置为:${MY_API.CONFIG.popularity_red_pocket_onlineNum}人`);
                            break;
                        case 'bagsendonekey':
                            MY_API.CONFIG.bagsendonekey_room = parseInt(parentDiv.find('.num').val());
                            window.toast(`快捷送礼房间号设置为:${MY_API.CONFIG.bagsendonekey_room}`);
                            break;
                    }
                    MY_API.saveConfig();
                });

                bagsendbox.find('[data-action="save1"]').click(function() {
                    const parentDiv = $(this).closest('div[data-toggle]');
                    const toggle = parentDiv.attr('data-toggle');

                    if (toggle === 'auto_gift_anchor') {
                        let val = parentDiv.find('.num1').val().replaceAll(' ', '').replaceAll(',', ',');
                        MY_API.CONFIG.gift_anchor_pass_giftname = val.split(',').filter(item => item);
                        parentDiv.find('.num1').val(MY_API.CONFIG.gift_anchor_pass_giftname.join(', '));
                        window.toast(`已屏蔽 ${MY_API.CONFIG.gift_anchor_pass_giftname.length} 种礼物`);
                        MY_API.saveConfig();
                    } else if (toggle === 'popularity_red_pocket_join_switch') {
                        MY_API.CONFIG.gift_total_price_min = parseInt(parentDiv.find('.num1').val());
                        window.toast(`红包参与电池下限设置为:${MY_API.CONFIG.gift_total_price_min}电池`);
                        MY_API.saveConfig();
                    }
                });

                // Quick send gifts handler
                let bagsendonekey_mark = false;
                bagsendbox.find('div[data-toggle="bagsendonekey"] [data-action="save"]').click(async function() {
                    MY_API.CONFIG.bagsendonekey_room = parseInt(bagsendbox.find('div[data-toggle="bagsendonekey"] .num').val());
                    MY_API.saveConfig();
                    window.toast(`快捷送礼房间号:${MY_API.CONFIG.bagsendonekey_room}`);
                    if (bagsendonekey_mark) return window.toast('正在执行快捷送礼,请稍候...');
                    bagsendonekey_mark = true;
                    await sleep(1000);
                    const gift_list = ["情书", "打call", "这个好诶", "星愿水晶球",
                                       "告白花束", "花式夸夸", "撒花", "小花花",
                                       "人气票", "星轨列车", "次元之城", "小电视飞船", "粉丝团灯牌",
                                       "棒棒糖", "你真好看", "爱心小熊", "告白气球"];

                    const r = confirm(`投喂直播间号:${MY_API.CONFIG.bagsendonekey_room}\n投喂包裹礼物范围:${gift_list.join(", ")}\n点击确认开始投喂`);

                    if (r) {
                        await BAPI.get_anchor_in_room(MY_API.CONFIG.bagsendonekey_room).then(async(data) => {
                            if(data.code==0 && data.data.info && data.data.info.uid){
                                let ruid = data.data.info.uid;
                                window.toast(`开始投喂直播间 ${MY_API.CONFIG.bagsendonekey_room}...`);
                                await BAPI.bag_list().then(async function(bagResult) {
                                    if (bagResult.data.list) {
                                        for (const item of bagResult.data.list) {
                                            if (gift_list.includes(item.gift_name) && item.expire_at !== 0 && item.gift_type !== 5) {
                                                await sleep(2000);
                                                await BAPI.bag_send(//(uid, gift_id, ruid, gift_num, bag_id, room_id, csrf_token)
                                                    Live_info.uid,
                                                    item.gift_id,
                                                    ruid,
                                                    item.gift_num,
                                                    item.bag_id,
                                                    MY_API.CONFIG.bagsendonekey_room,
                                                    Live_info.csrf_token
                                                ).then(async function(data) {
                                                    if (data.code === 0) {
                                                        window.toast(
                                                            `投喂成功 ${item.gift_name} × ${item.gift_num}`
                                                        );
                                                    } else {
                                                        window.toast(`投喂失败: ${data.message}`);
                                                    }
                                                });
                                            }
                                        }
                                    }
                                });
                            }
                        })
                    }
                    bagsendonekey_mark = false;
                })
            },
            check_bag_gift: async function() {
                let dianchi = 0;
                let dianchi_gift_num = 0;
                const gift_data = [
                    {"gift_id": 30758, "price": 100, "gift_name": "这个好诶"},
                    {"gift_id": 31164, "price": 100, "gift_name": "粉丝团灯牌"},
                    {"gift_id": 31212, "price": 500, "gift_name": "打call"},
                    {"gift_id": 31213, "price": 1000, "gift_name": "这个好诶"},
                    {"gift_id": 31216, "price": 100, "gift_name": "小花花"},
                    {"gift_id": 31250, "price": 5200, "gift_name": "情书"},
                    {"gift_id": 31738, "price": 1000, "gift_name": "粉丝团灯牌"},
                    {"gift_id": 32075, "price": 5200, "gift_name": "情书"},
                    {"gift_id": 32698, "price": 1500, "gift_name": "小蛋糕"},
                    {"gift_id": 33987, "price": 0, "gift_name": "人气票"},
                    {"gift_id": 34003, "price": 100, "gift_name": "人气票"},
                    {"gift_id": 34016, "price": 100, "gift_name": "小花花"},
                    {"gift_id": 34017, "price": 1000, "gift_name": "这个好诶"},
                    {"gift_id": 34358, "price": 100, "gift_name": "粉丝团灯牌"},
                    {"gift_id": 34758, "price": 200, "gift_name": "棒棒糖"},
                    {"gift_id": 34759, "price": 19900, "gift_name": "告白花束"},
                    {"gift_id": 31217, "gift_name": "星愿水晶球", "price": 100000},
                    {"gift_id": 34977, "gift_name": "告白花束", "price": 19900},
                    {"gift_id": 34978, "gift_name": "花式夸夸", "price": 33000},
                    {"gift_id": 34019, "gift_name": "星轨列车", "price": 666600},
                    {"gift_id": 34020, "gift_name": "次元之城", "price": 1245000},
                    {"gift_id": 34021, "gift_name": "小电视飞船", "price": 2999000},
                    {"gift_id": 34758, "gift_name": "棒棒糖", "price": 200},
                    {"gift_id": 34979, "gift_name": "你真好看", "price": 1000},
                    {"gift_id": 34961, "gift_name": "爱心小熊", "price": 52000},
                    {"gift_id": 34980, "gift_name": "告白气球", "price": 200000}
                ];
                await BAPI.bag_list().then(async function(bagResult) {
                    if (bagResult.data.list) {
                        for (const item of bagResult.data.list) {
                            const gift = gift_data.find(g => g.gift_id === item.gift_id);
                            if (gift && item.expire_at !== 0 && item.gift_type !== 5) {
                                dianchi += gift.price * item.gift_num/100;
                                dianchi_gift_num += item.gift_num;
                            }
                        }
                    }
                });

                document.getElementById("gift_count").textContent = `包裹电池数:${dianchi}`;
                document.getElementById("gift_num").textContent = `包裹礼物数:${dianchi_gift_num}`;
                await sleep(30*1000);
                return MY_API.check_bag_gift();
            },
            bili_ws: async (room_id, time = 0) => {
                const connect = async () => {
                    let timeLeft = time;
                    let token;
                    try {
                        const get_token = await BAPI.getConf(room_id);
                        if (get_token.code !== 0) return;
                        token = get_token.data.token;
                    } catch (error) {
                        return;
                    }
                    const ws = new WebSocket("wss://broadcastlv.chat.bilibili.com/sub");
                    let timer, timeout, timeLeftTimer;
                    const json = {
                        uid: Live_info.uid,
                        roomid: room_id,
                        protover: 1,
                        platform: "web",
                        clientver: "1.4.0",
                        key: token
                    };
                    const str2bytes = (str) => {
                        const bytes = [];
                        const len = str.length;
                        for (let i = 0; i < len; i++) {
                            const c = str.charCodeAt(i);
                            if (c >= 0x010000 && c <= 0x10FFFF) {
                                bytes.push(((c >> 18) & 0x07) | 0xF0);
                                bytes.push(((c >> 12) & 0x3F) | 0x80);
                                bytes.push(((c >> 6) & 0x3F) | 0x80);
                                bytes.push((c & 0x3F) | 0x80);
                            } else if (c >= 0x000800 && c <= 0x00FFFF) {
                                bytes.push(((c >> 12) & 0x0F) | 0xE0);
                                bytes.push(((c >> 6) & 0x3F) | 0x80);
                                bytes.push((c & 0x3F) | 0x80);
                            } else if (c >= 0x000080 && c <= 0x0007FF) {
                                bytes.push(((c >> 6) & 0x1F) | 0xC0);
                                bytes.push((c & 0x3F) | 0x80);
                            } else {
                                bytes.push(c & 0xFF);
                            }
                        }
                        return bytes;
                    };
                    const getCertification = (json) => {
                        const bytes = str2bytes(JSON.stringify(json));
                        const buffer = new ArrayBuffer(bytes.length + 16);
                        const view = new DataView(buffer);
                        view.setUint32(0, bytes.length + 16);
                        view.setUint16(4, 16);
                        view.setUint16(6, 1);
                        view.setUint32(8, 7);
                        view.setUint32(12, 1);
                        for (let i = 0; i < bytes.length; i++) {
                            view.setUint8(16 + i, bytes[i]);
                        }
                        return view;
                    };
                    const createHeartbeatPacket = () => {
                        const buffer = new ArrayBuffer(16);
                        const view = new DataView(buffer);
                        view.setUint32(0, 0);
                        view.setUint16(4, 16);
                        view.setUint16(6, 1);
                        view.setUint32(8, 2);
                        view.setUint32(12, 1);
                        return buffer;
                    };
                    ws.onopen = () => {
                        ws.send(getCertification(json).buffer);
                        timer = setInterval(() => {
                            ws.send(createHeartbeatPacket());
                        }, 30000);
                        timeLeftTimer = setInterval(() => {
                            timeLeft -= 5000;
                        }, 5000);
                        if (time > 0) {
                            timeout = setTimeout(() => {
                                cleanup();
                            }, time);
                        }
                    };
                    ws.onclose = () => {
                        cleanup();
                        if (timeLeft > 20000) {
                            setTimeout(() => {
                                MY_API.bili_ws(room_id, timeLeft - 10000);
                            }, 10000);
                        } else if (time === 0) {
                            setTimeout(() => {
                                MY_API.bili_ws(room_id);
                            }, 30000);
                        }
                    };
                    ws.onmessage = (evt) => {
                    };
                    ws.onerror = (error) => {
                        cleanup();
                    };
                    const cleanup = () => {
                        if (timer) clearInterval(timer);
                        if (timeout) clearTimeout(timeout);
                        if (timeLeftTimer) clearInterval(timeLeftTimer);
                        if (ws.readyState === WebSocket.OPEN) {
                            ws.close();
                        }
                    };
                };
                await connect();
            },
        };
        MY_API.init().then(function () {
            try {
                const promiseInit = $.Deferred();
                const uniqueCheck = () => {
                    const t = Date.now();
                    if(t - MY_API.CONFIG.js_running_mark >= 0 && t - MY_API.CONFIG.js_running_mark <= 15e3){
                        // 其他脚本正在运行
                        setTimeout(() => {
                            window.toast('检测到脚本已经运行!');
                        }, 5e3);
                        return promiseInit.reject();
                    }
                    // 没有其他脚本正在运行
                    return promiseInit.resolve();
                };
                uniqueCheck().then(() => {
                    let timer_unique;
                    const uniqueMark = () => {
                        timer_unique = setTimeout(uniqueMark, 10e3);
                        MY_API.CONFIG.js_running_mark = Date.now();
                        try {
                            localStorage.setItem(`${NAME}_TXHBCONFIG`, JSON.stringify(MY_API.CONFIG));
                            return true
                        } catch (e){
                            prettyLog('API保存出错', e);
                            return false
                        };
                    };
                    window.addEventListener('unload', () => {
                        if(timer_unique){
                            clearTimeout(timer_unique);
                            MY_API.CONFIG.js_running_mark = 0;
                            try {
                                localStorage.setItem(`${NAME}_TXHBCONFIG`, JSON.stringify(MY_API.CONFIG));
                                return true
                            } catch (e){
                                prettyLog('API保存出错', e);
                                return false
                            };
                        }
                    });
                    uniqueMark();
                    StartPlunder(MY_API);
                })
            } catch (e){
                console.error('重复运行检测错误', e);
            }
        });
    }

    async function StartPlunder(API){
        let icon_pic = "https://i0.hdslb.com/bfs/live/72bdc71f91ebec5681bb1975bc5f97862e996a34.webp"
        let ui_button = $(`<img width=100 height=100 style="position: fixed; top: 275px; right: 60px;z-index:999;" src=${icon_pic} />`)
        $('html').append(ui_button);
        ui_button.click(function () {
            $('.bagsendbox').toggle()
        });

        API.creatSetBox();
        API.check_bag_gift()

        let get_ruid_by_room = async function(room_id){
            let ruid = 0
            await BAPI.get_anchor_in_room(room_id).then(async(data) => {
                if(data.code==0 && data.data.info && data.data.info.uid){
                    ruid = data.data.info.uid;
                }
            })
            return ruid
        }

        let get_onlineNum = async function(ruid, room_id){
            let onlineNum = -1
            await BAPI.getOnlineGoldRank(ruid,room_id).then(async(data) => {
                if(data.code==0 && data.data.onlineNum){
                    onlineNum = data.data.onlineNum
                }
            })
            return onlineNum
        }

        var popularity_red_pocket_join_num_max = false
        var popularity_red_pocket_join_err = false
        var gift_anchor_join_err = false


        let check_newday = async() => { //判断是否第二天重置数据
            if(checkNewDay(API.CONFIG.check_newday_ts)){
                API.CONFIG.check_newday_ts = dateNow();
                popularity_red_pocket_join_num_max = false
                popularity_red_pocket_join_err = false
            }
        };

        check_newday()
        setInterval(check_newday, 20e3);

        let popularity_red_pocket_join = async function(ruid, room_id, lot_id, runninr_ts){
            await API.bili_ws(room_id,(runninr_ts + 20) * 1000, true)
            let roomHeart = new RoomHeart(room_id, Math.ceil((runninr_ts + 20)/60),ruid)
            await roomHeart.start()
            await sleep(10000)
            if(API.CONFIG.auto_like)BAPI.likeReportV3(room_id, Live_info.uid, ruid, Live_info.csrf_token)
            var formData = new FormData();
            formData.set("visit_id", "");
            formData.set("session_id", "");
            formData.set("room_id", room_id);
            formData.set("ruid", ruid);
            formData.set("spm_id", "444.8.red_envelope.extract");
            formData.set("jump_from", "26000");
            formData.set("build", "6790300");
            formData.set("c_locale", "en_US");
            formData.set("channel", "360");
            formData.set("device", "android");
            formData.set("mobi_app", "android");
            formData.set("platform", "android");
            formData.set("version", "6.79.0");
            formData.set("statistics", "%7B%22appId%22%3A1%2C%22platform%22%3A3%2C%22version%22%3A%226.79.0%22%2C%22abtest%22%3A%22%22%7D");
            function drawRedPacket() {
                formData.set("csrf", Live_info.csrf_token);
                formData.set("csrf_token", Live_info.csrf_token);
                formData.set("lot_id", lot_id);
                GM_xmlhttpRequest({
                    url: `https://api.live.bilibili.com/xlive/lottery-interface/v1/popularityRedPocket/RedPocketDraw`,
                    method: "post",
                    headers: {
                        "User-Agent": "Mozilla/5.0 BiliDroid/6.79.0 ([email protected]) os/android model/Redmi K30 Pro mobi_app/android build/6790300 channel/360 innerVer/6790310 osVer/11 network/2"
                    },
                    data: formData,
                    onload: async function (res) {
                        let data = JSON.parse(res.response);
                        if(data.code ==0){
                            window.toast(`【人气红包】房间号:${room_id},直播间人气红包参与成功。开奖时间:${new Date((ts_s() + runninr_ts)*1000).toLocaleString()}`, 'success');
                        }else if(data.code == 1009109){
                            popularity_red_pocket_join_num_max = true
                            window.toast(`【人气红包】${room_id}直播间人气红包参与反馈:${data.message}`, 'error')
                        }else if(data.code == 1009114){
                            popularity_red_pocket_join_err = true
                            window.toast(`【人气红包】${room_id}直播间人气红包参与反馈:${data.message}`, 'error')
                        }else{
                            popularity_red_pocket_join_err = true
                            window.toast(`【人气红包】${room_id}直播间人气红包参与反馈:${data.message}`, 'error')
                        }
                    }
                })
            }
            drawRedPacket()
        }


        let gift_anchor_join = async function(ruid, room_id, lot_id, runninr_ts){
            await API.bili_ws(room_id,(runninr_ts + 20) * 1000, true)
            let roomHeart = new RoomHeart(room_id, Math.ceil((runninr_ts + 20)/60),ruid)
            await roomHeart.start()
            await sleep(10000)
            if(API.CONFIG.auto_like)BAPI.likeReportV3(room_id, Live_info.uid, ruid, Live_info.csrf_token)
            await BAPI.anchor_draw(lot_id, room_id, Live_info.csrf_token).then(async(data) => {
                if(data.code == 0){
                    window.toast(`【礼物天选】${room_id}参与成功。开奖时间:${new Date((ts_s() + runninr_ts)*1000).toLocaleString()}`, 'success');
                }else{
                    window.toast(`【礼物天选】${room_id}参与反馈:${data.message}`, 'error')
                    if(data.code == 400){
                        document.cookie = 'x-bili-gaia-vtoken=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;'
                        document.cookie = 'GIFT_BLOCK_COOKIE=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;'
                    }else if(data.code == -352 && API.CONFIG.join_code_check){
                        gift_anchor_join_err = true
                        const r = confirm(`请打开抽奖直播间${room_id}或其他抽奖直播间手动过验证码\n完成验证后点击确认继续抽奖`);
                        if (r) {
                            gift_anchor_join_err = false
                            document.cookie = 'x-bili-gaia-vtoken=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;'
                            document.cookie = 'GIFT_BLOCK_COOKIE=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;'
                        }
                    }
                }
            });
        }

        async function popularity_red_pocket_loop() {
            if(API.CONFIG.popularity_red_pocket_join_switch && !popularity_red_pocket_join_num_max){
                let data = await get_popularity_red_pocket_data()
                if(data && data.length){
                    for(let i=0;i<data.length;i++){
                        if(data[i].end_time - ts_s() < 30)continue
                        if(data[i].total_price < API.CONFIG.gift_total_price_min)continue
                        const match = data[i].online_num.match(/\d+/)
                        if(match && API.CONFIG.popularity_red_pocket_onlineNum_switch && match[0] > API.CONFIG.popularity_red_pocket_onlineNum)continue
                        let lot_id = data[i].lid
                        let room_id = data[i].room_id
                        let runninr_ts = data[i].end_time - ts_s()
                        if(API.CONFIG.popularity_red_pocket_done_id_list.indexOf(lot_id) > -1)continue
                        let ruid = await get_ruid_by_room(room_id)
                        if(!ruid)break
                        if(API.CONFIG.popularity_red_pocket_onlineNum_switch){
                            let onlineNum = await get_onlineNum(ruid, room_id)
                            if(onlineNum > API.CONFIG.popularity_red_pocket_onlineNum){
                                API.CONFIG.popularity_red_pocket_done_id_list.push(lot_id)
                                continue
                            }
                        }else{
                            API.CONFIG.popularity_red_pocket_done_id_list.push(lot_id)
                        }
                        if(API.CONFIG.popularity_red_pocket_done_id_list.length > 100)API.CONFIG.popularity_red_pocket_done_id_list.splice(0,50)
                        API.saveConfig()
                        popularity_red_pocket_join(ruid, room_id, lot_id, runninr_ts)
                        await sleep(runninr_ts*1000)
                        await sleep(API.CONFIG.popularity_red_pocket_flash*1000)
                        break
                    }
                }
            }
            await sleep(30*1000)
            setTimeout(popularity_red_pocket_loop, 1000)
        }

        async function gift_anchor_loop() {
            if(API.CONFIG.auto_gift_anchor && !gift_anchor_join_err){
                let data = await get_gift_anchor_data()
                if(data && data.length){
                    for(let i=0;i<data.length;i++){
                        if(gift_anchor_join_err)break
                        if(API.CONFIG.gift_anchor_pass_giftname.indexOf(data[i].award_name) > -1)continue
                        if(data[i].end_time - ts_s() < 30)continue
                        if(data[i].require_text != "无" && data[i].require_text != "关注")continue
                        let lot_id = data[i].lid
                        let room_id = data[i].room_id
                        let runninr_ts = data[i].end_time - ts_s()
                        if(API.CONFIG.done_id_list.indexOf(lot_id) > -1)continue
                        let ruid = await get_ruid_by_room(room_id)
                        if(!ruid)break
                        API.CONFIG.done_id_list.push(lot_id)
                        if(API.CONFIG.done_id_list.length > 200)API.CONFIG.done_id_list.splice(0,50)
                        API.saveConfig()
                        gift_anchor_join(ruid, room_id, lot_id, runninr_ts)
                        await sleep(API.CONFIG.gift_anchor_flash*1000)
                    }
                }
            }
            await sleep(30*1000)
            setTimeout(gift_anchor_loop, 1000)
        }
        popularity_red_pocket_loop()
        gift_anchor_loop()

    }

    const checkNewDay = (ts) => {
        if(ts === 0) return true;
        let t = new Date(ts);
        let d = new Date();
        let td = t.getDate();
        let dd = d.getDate();
        return (dd !== td);
    }

    function get_gift_anchor_data(){
        return new Promise((resolve) => {
            GM_xmlhttpRequest({
                method: 'get',
                headers:{"Content-Type": "application/json"},
                url: `http://150.158.35.26:9527/gift_anchor/kasfdhjakw2da1qwsd15w3ad4q5aqfhhjc`,
                onload: function(response){
                    const res = JSON.parse(response.response);
                    resolve(res);
                },
                onerror: function(err){
                    resolve(undefined);
                }
            })
        })
    }
    function get_popularity_red_pocket_data(){
        return new Promise((resolve) => {
            GM_xmlhttpRequest({
                method: 'get',
                headers:{"Content-Type": "application/json"},
                url: `http://150.158.35.26:9527/popularity_red_pocket/kasfdhjakw2da1qwsd15w3ad4q5aqfhhjc`,
                onload: function(response){
                    const res = JSON.parse(response.response);
                    resolve(res);
                },
                onerror: function(err){
                    resolve(undefined);
                }
            })
        })
    }

})();

var BilibiliAPI = {
    getPictureHashKey(i) {
        const V = [46, 47, 18, 2, 53, 8, 23, 32, 15, 50, 10, 31, 58, 3, 45, 35, 27, 43, 5, 49, 33, 9, 42, 19, 29, 28, 14, 39, 12, 38, 41, 13, 37, 48, 7, 16, 24, 55, 40, 61, 26, 17, 0, 1, 60, 51, 30, 4, 22, 25, 54, 21, 56, 59, 6, 63, 57, 62, 11, 36, 20, 34, 44, 52], N = [];
        return V.forEach(U=>{i.charAt(U) && N.push(i.charAt(U))}),N.join("").slice(0, 32)
    },
    get_w_rid(params_str) {
        const N = "7cd084941338484aae1ad9425b84077c"
        const U = "4932caff0ff746eab6f01bf08b70ac45"
        const R = BilibiliAPI.getPictureHashKey(N + U)
        const Q = `${params_str}${R}`
        var w_rid = CryptoJS.MD5(Q).toString()
        return w_rid
    },
    runUntilSucceed: (callback, delay = 0, period = 50) => {
        setTimeout(() => {
            if(!callback())
                BilibiliAPI.runUntilSucceed(callback, period, period);
        }, delay);
    },
    processing: 0,
    ajax: (settings) => {
        if(settings.xhrFields === undefined)
            settings.xhrFields = {};
        settings.xhrFields.withCredentials = true;
        jQuery.extend(settings, {
            url: settings.url,
            method: settings.method || 'GET',
            crossDomain: true,
            dataType: settings.dataType || 'json'
        });
        const p = jQuery.Deferred();
        BilibiliAPI.runUntilSucceed(() => {
            if(BilibiliAPI.processing > 8)
                return false;
            ++BilibiliAPI.processing;
            return jQuery.ajax(settings).then((arg1, arg2, arg3) => {
                --BilibiliAPI.processing;
                p.resolve(arg1, arg2, arg3);
                return true;
            }, (arg1, arg2, arg3) => {
                --BilibiliAPI.processing;
                p.reject(arg1, arg2, arg3);
                return true;
            });
        });
        return p;
    },
    likeReportV3: (room_id, uid, ruid, csrf_token, click_time=300) => {
        const wts = Math.round(Date.now()/1000)
        const params_str = `click_time=${click_time}&room_id=${room_id}&uid=${uid}&anchor_id=${ruid}&web_location=444.8&csrf=${csrf_token}`
        const w_rid = BilibiliAPI.get_w_rid(params_str+`&wts=${wts}`)
        return BilibiliAPI.ajax({
            method: "POST",
            url: "https://api.live.bilibili.com/xlive/app-ucenter/v1/like_info_v3/like/likeReportV3?" + params_str + `&w_rid=${w_rid}&wts=${wts}`,
        })
    },
    anchor_draw: (id, room_id, csrf_token) => {
        const wts = Math.round(Date.now()/1000)
        let w_rid = BilibiliAPI.get_w_rid(`wts=${wts}`)
        let data = {
            id: id,
            platform: "pc",
            room_id:room_id,
            jump_from_str:'',
            session_id:'',
            spm_id: '444.8.interaction.anchor_draw_auto',
            live_statistics: JSON.stringify({"pc_client":"unknownUA","jumpfrom":"-99998","room_category":"0","lottery_id":id,"lottery_type":1}),
            statistics:JSON.stringify({"platform":0}),
            csrf_token: csrf_token,
            csrf: csrf_token,
            visit_id: '',
        };
        return BilibiliAPI.ajax({
            method: "POST",
            url: "https://api.live.bilibili.com/xlive/lottery-interface/v1/Anchor/Join" + `?w_rid=${w_rid}&wts=${wts}`,
            data: data
        })
    },
    getOnlineGoldRank: (ruid, room_id) => {
        return BilibiliAPI.ajax({
            url: `https://api.live.bilibili.com/xlive/general-interface/v1/rank/getOnlineGoldRank?ruid=${ruid}&roomId=${room_id}&page=1&pageSize=50`,
            method: "GET",
        })
    },
    getConf: (room_id) => {
        return BilibiliAPI.ajax({
            url: `https://api.live.bilibili.com/room/v1/Danmu/getConf?room_id=${room_id}&platform=pc&player=web`,
            method: "GET",
        })
    },
    verify_room_pwd: (room_id) => {
        return BilibiliAPI.ajax({
            url: "https://api.live.bilibili.com/room/v1/Room/verify_room_pwd",
            method: "GET",
            data:{
                room_id:room_id,
            }
        })
    },
    get_anchor_in_room: (room_id) => {//获取直播间主播信息
        return BilibiliAPI.ajax({
            url: "https://api.live.bilibili.com/live_user/v1/UserInfo/get_anchor_in_room?roomid=" + room_id,
            method: "GET",
        })
    },
    bag_list: () => {
        return BilibiliAPI.ajax({
            url: `https://api.live.bilibili.com/xlive/web-room/v1/gift/bag_list?t=${Date.now()}&room_id=5440&mobi_app=web&receive_users=[]&web_location=444.8`,
        });
    },
    bag_send: (uid, gift_id, ruid, gift_num, bag_id, room_id, csrf_token) => {
        return BilibiliAPI.ajax({
            method: 'POST',
            url: 'https://api.live.bilibili.com/xlive/revenue/v2/gift/sendBag',
            data: {
                uid: uid,
                gift_id: gift_id,
                ruid: ruid,
                gift_num: gift_num,
                bag_id: bag_id,
                platform: 'pc',
                biz_code: 'Live',
                biz_id: room_id,
                rnd: Date.now(),
                storm_beat_id: 0,
                metadata: '',
                price: 0,
                send_ruid:0,
                csrf_token: csrf_token,
                csrf: csrf_token,
                visit_id: '',
            }
        });
    },
}

//https://github.com/lzghzr/TampermonkeyJS

class RoomHeart {
    constructor(t,g,r) {
        this.roomID = t,
            this.Dotime = g,
            this.ruid = r
    }
    areaID;
    parentID;
    seq = 0;
    roomID;
    Dotime;
    get id() {
        return [this.parentID, this.areaID, this.seq, this.roomID]
    }
    buvid = this.getItem("LIVE_BUVID");
    uuid = "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g, (t => {
        const e = 16 * Math.random() | 0;
        return ("x" === t ? e : 3 & e | 8).toString(16)
    }));
    device = [this.buvid, this.uuid];
    get ts() {
        return Date.now()
    }
    _patchData = {};
    get patchData() {
        const t = [];
        for (const[e, i]of Object.entries(this._patchData))
            t.push(i);
        return t
    }
    get isPatch() {
        return 0 === this.patchData.length ? 0 : 1
    }
    W = "undefined" == typeof unsafeWindow ? window : unsafeWindow;
    ua = this.W && this.W.navigator ? this.W.navigator.userAgent : "";
    csrf = this.getItem("bili_jct") || "";
    nextInterval = Math.floor(5) + Math.floor(55 * Math.random());
    heartBeatInterval;
    secretKey;
    secretRule;
    timestamp;
    lastHeartbeatTimestamp = Date.now();
    get watchTimeFromLastReport() {
        const t = Math.ceil(((new Date).getTime() - this.lastHeartbeatTimestamp) / 1e3);
        return t < 0 ? 0 : t > this.heartBeatInterval ? this.heartBeatInterval : t
    }
    start() {
        return this.getInfoByRoom()
    }
    doneFunc = function () {};
    async getInfoByRoom() {
        if (0 === this.roomID)
            return !1;
        const t = await fetch(`//api.live.bilibili.com/room/v1/Room/get_info?room_id=${this.roomID}&from=room`, {
            mode: "cors",
            credentials: "include"
        }).then((t => t.json()));
        return 0 === t.code && (({
            area_id: this.areaID,
            parent_area_id: this.parentID,
            room_id: this.roomID
        } = t.data), 0 !== this.areaID && 0 !== this.parentID && (this.e(), !0))
    }
    async webHeartBeat() {
        if (this.seq > this.Dotime)
            return;
        const t = `${this.nextInterval}|${this.roomID}|1|0`,
              e = CryptoJS.enc.Utf8.parse(t),
              i = CryptoJS.enc.Base64.stringify(e),
              s = await fetch(`//live-trace.bilibili.com/xlive/rdata-interface/v1/heartbeat/webHeartBeat?hb=${encodeURIComponent(i)}&pf=web`, {
                  mode: "cors",
                  credentials: "include"
              }).then((t => t.json()));
        0 === s.code && (this.nextInterval = s.data.next_interval, setTimeout((() => this.webHeartBeat()), 1e3 * this.nextInterval))
    }
    async savePatchData() {
        if (this.seq > this.Dotime)
            return;
        const t = {
            id: JSON.stringify(this.id),
            device: JSON.stringify(this.device),
            ruid:this.ruid,
            ets: this.timestamp,
            benchmark: this.secretKey,
            time: this.watchTimeFromLastReport > this.heartBeatInterval ? this.heartBeatInterval : this.watchTimeFromLastReport,
            ts: this.ts,
            ua: this.ua
        },
              e = this.sypder(JSON.stringify(t), this.secretRule),
              i = Object.assign({
                  s: e
              }, t);
        this._patchData[this.roomID] = i,
            setTimeout((() => this.savePatchData()), 15e3)
    }
    async e() {
        const t = {
            id: JSON.stringify(this.id),
            device: JSON.stringify(this.device),
            ruid:this.ruid,
            ts: this.ts,
            is_patch: 0,
            heart_beat: "[]",
            ua: this.ua
        },
              e = await fetch("//live-trace.bilibili.com/xlive/data-interface/v1/x25Kn/E", {
                  headers: {
                      "content-type": "application/x-www-form-urlencoded"
                  },
                  method: "POST",
                  body: `${this.json2str(t)}&csrf_token=${this.csrf}&csrf=${this.csrf}&visit_id=`,
                  mode: "cors",
                  credentials: "include"
              }).then((t => t.json()));
        0 === e.code && (this.seq += 1, ({
            heartbeat_interval: this.heartBeatInterval,
            secret_key: this.secretKey,
            secret_rule: this.secretRule,
            timestamp: this.timestamp
        } = e.data), setTimeout((() => this.x()), 1e3 * this.heartBeatInterval))
    }
    async x() {
        if (this.seq > this.Dotime)
            return this.doneFunc();
        const t = {
            id: JSON.stringify(this.id),
            device: JSON.stringify(this.device),
            ruid:this.ruid,
            ets: this.timestamp,
            benchmark: this.secretKey,
            time: this.heartBeatInterval,
            ts: this.ts,
            ua: this.ua
        },
              e = this.sypder(JSON.stringify(t), this.secretRule),
              i = Object.assign({
                  s: e
              }, t);
        this._patchData[this.roomID] = i,
            this.lastHeartbeatTimestamp = Date.now();
        const s = await fetch("//live-trace.bilibili.com/xlive/data-interface/v1/x25Kn/X", {
            headers: {
                "content-type": "application/x-www-form-urlencoded"
            },
            method: "POST",
            body: `${this.json2str(i)}&csrf_token=${this.csrf}&csrf=${this.csrf}&visit_id=`,
            mode: "cors",
            credentials: "include"
        }).then((t => t.json()));
        0 === s.code && (this.seq += 1, ({
            heartbeat_interval: this.heartBeatInterval,
            secret_key: this.secretKey,
            secret_rule: this.secretRule,
            timestamp: this.timestamp
        } = s.data), setTimeout((() => this.x()), 1e3 * this.heartBeatInterval))
    }
    sypder(t, e) {
        const i = JSON.parse(t),
              [s, a, r, n] = JSON.parse(i.id),
              [o, c] = JSON.parse(i.device),
              h = i.benchmark,
              m = {
                  platform: "web",
                  parent_id: s,
                  area_id: a,
                  seq_id: r,
                  room_id: n,
                  buvid: o,
                  uuid: c,
                  ets: i.ets,
                  time: i.time,
                  ts: i.ts
              };
        let d = JSON.stringify(m);
        for (const t of e)
            switch (t) {
                case 0:
                    d = CryptoJS.HmacMD5(d, h).toString(CryptoJS.enc.Hex);
                    break;
                case 1:
                    d = CryptoJS.HmacSHA1(d, h).toString(CryptoJS.enc.Hex);
                    break;
                case 2:
                    d = CryptoJS.HmacSHA256(d, h).toString(CryptoJS.enc.Hex);
                    break;
                case 3:
                    d = CryptoJS.HmacSHA224(d, h).toString(CryptoJS.enc.Hex);
                    break;
                case 4:
                    d = CryptoJS.HmacSHA512(d, h).toString(CryptoJS.enc.Hex);
                    break;
                case 5:
                    d = CryptoJS.HmacSHA384(d, h).toString(CryptoJS.enc.Hex);
                    break;
                default:
                    break
            }
        return d
    }
    getItem(t) {
        return decodeURIComponent(document.cookie.replace(new RegExp("(?:(?:^|.*;)\\s*" + encodeURIComponent(t).replace(/[\-\.\+\*]/g, "\\$&") + "\\s*\\=\\s*([^;]*).*$)|^.*$"), "$1")) || ""
    }
    json2str(t) {
        let e = "";
        for (const i in t)
            e += `${i}=${encodeURIComponent(t[i])}&`;
        return e.slice(0, -1)
    }
}