海角天涯-解锁海角社区资源

无限制解锁收费资源,包括钻石贴、金币贴,下载视频,复制视频链接,自动展开帖子

// ==UserScript==
// @name         海角天涯-解锁海角社区资源
// @namespace    tianya365.top
// @homepage     https://vip.tianya365.top
// @version      1.4.0
// @description  无限制解锁收费资源,包括钻石贴、金币贴,下载视频,复制视频链接,自动展开帖子
// @icon         
// @author       tianya365.top
// @exclude      https://*.tianya365.top/*
// @include      https://hj*.*/*
// @include      https://h*j*.*/*
// @include      https://*.hj*.*/*
// @include      https://hai*.top/*
// @match        https://haijiao.com/*
// @match        https://www.haijiao.com/*
// @match        *://*/post/details*
// @connect      *
// @run-at       document-start
// @grant        none
// @antifeature  payment
// @charset		 UTF-8
// ==/UserScript==

(function () {
    'use strict';

    let version = '1.4.0';
    let newVersion = '';
    let m3u8_url = '';
    let play_url = '';
    let hasPic = false;
    let video = null;
    let topic = null;
    let shortVideoUrl = '';
    let server = window.localStorage.getItem('hjty_server');
    let freeToken = window.localStorage.getItem('hjty_free_token');

    if (server == null) {
        server = 'https://vip.tianya365.top';
        window.localStorage.setItem('hjty_server', server);
    }

    if (freeToken == null) {
        freeToken = '8b9591fcdb4e4340bc815cb183aa02f8'
        window.localStorage.setItem('hjty_free_token', freeToken);
    }

    function importJs(src, onload) {
        let script = document.createElement("script");
        script.setAttribute("type", "text/javascript");
        script.src = src;
        script.onload = onload;
        document.getElementsByTagName('head')[0].appendChild(script);
    }

    function loadStyle() {
        $('head').append(`
<style>
.hjty-dialog-mask{position: fixed;top: 0;left: 0;width: 100%;height: 100%;background-color: rgba(0, 0, 0, 0.5);z-index: 3000;}
.hjty-dialog{position: fixed;top: 50%;left: 50%;transform: translate(-50%, -50%);background-color: #ffffff;box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);padding: 20px;font-size: 14px;
    border: 1px solid #f1f1f1;
    border-radius: 5px;
    width: 250px;}
.hjty-dialog-title{font-size: 16px;font-weight: bold;margin-bottom: 10px;text-align: center;}
.hjty-dialog-content p{color: #333333;margin-bottom: 10px;}
.hjty-dialog-content ul{list-style: none;padding: 0;margin: 0;}
.hjty-dialog-content ul li{
    margin-bottom: 5px;
    border: 2px solid #d5d5d5;
    padding: 3px 10px;
    border-radius: 4px;
    background: #f7f7f7;
    cursor: pointer;
}
.hjty-dialog-content ul li.active{border-color: #409eff;background: #ecf5ff;}
.hjty-dialog-content ul li span.status{display: inline-block;width: 10px;height: 10px;border-radius: 50%;margin-right: 5px;background-color: #a9a9a9;}
.hjty-dialog-content ul li span.status-success{background-color: #67c23a;}
.hjty-dialog-content ul li span.status-fail{background-color: #f56c6c;}
.hjty-dialog-content ul li span.timeout{color: #999999;font-size: 12px;float: right;margin-top: 1px;}
.hjty-dialog-footer{margin-top: 20px;text-align: center;}
.hjty-dialog-btn{padding: 5px 10px;background-color: #409eff;color: #ffffff;border: none;border-radius: 4px;cursor: pointer;}
#play-box {height: 100%;display: flex;align-items: center;background-color: #000;}
.van-popup.van-popup--center.play-box {width: 100%;height: 100%;max-width: 100%;}
#bbs_float_menu{z-index:1!important;}
#body.el-popup-parent--hidden{overflow: auto;}
.van-list .van-list__loading,
.sell-btn:before,
.el-message-box,
.v-modal,
.containeradvertising,
.ishide,
.el-message-box__wrapper,
#tidio-chat{display: none !important;}
</style
        `);
    }

    function importCss(href) {
        let link = document.createElement("link");
        link.setAttribute("rel", "stylesheet");
        link.href = href;
        document.getElementsByTagName('head')[0].appendChild(link);
    }

    function checkUrl(url) {
        return new Promise((resolve, reject) => {
            const xhr = new XMLHttpRequest();
            const startTime = Date.now();
            const timeoutId = setTimeout(() => {
                resolve(null);
            }, 5000);

            xhr.open('HEAD', url, true);
            xhr.onload = function() {
                clearTimeout(timeoutId);
                if (xhr.status >= 200 && xhr.status < 400) {
                    const responseTime = Date.now() - startTime;
                    resolve({ url: url, time: responseTime });
                } else {
                    resolve(null);
                }
            };

            xhr.onerror = function() {
                clearTimeout(timeoutId);
                resolve(null);
            };

            xhr.send();
        });
    }

    function switchServer() {
        $('body').append(`
<div class="hjty-dialog-mask">
    <div class="hjty-dialog">
        <div class="hjty-dialog-title">海角天涯提示</div>
        <div class="hjty-dialog-content">
            <p>请选择服务器节点</p>
            <ul>
                <li data-url="https://vip.tianya365.top"><span class="status"></span> 服务器1 <span class="timeout">测速中</span></li>
                <li data-url="https://haijiao.tianya365.top"><span class="status"></span> 服务器2 <span class="timeout">测速中</span></li>
                <li data-url="https://new.tianya365.top"><span class="status"></span> 服务器3 <span class="timeout">测速中</span></li>
                <li data-url="https://hj.tianya365.top"><span class="status"></span> 服务器4 <span class="timeout">测速中</span></li>
            </ul>
        </div>
        <div class="hjty-dialog-footer">
            <button class="hjty-dialog-btn">确定</button>
        </div>
    </div>
</div>
        `);

        $('.hjty-dialog-content ul li').each(function() {
            const url = $(this).attr('data-url');
            if (server === url) {
                $(this).addClass('active');
            } else {
                $(this).removeClass('active');
            }
            checkUrl(url + '/login').then(result => {
                if (result) {
                    $(this).find('.status').addClass('status-success');
                    $(this).find('.timeout').html(`${result.time}ms`);
                } else {
                    $(this).find('.status').addClass('status-fail');
                    $(this).find('.timeout').html('超时');
                }
            })
        });

        $('.hjty-dialog-content ul li').click(function() {
            $('.hjty-dialog-content ul li').not($(this)).removeClass('active');
            $(this).addClass('active');
            server = $(this).attr('data-url');
            window.localStorage.setItem('hjty_server', server);
        });
        $('.hjty-dialog-btn').click(function() {
            location.reload()
        });
    }

    window.onload = function () {
        loadStyle();
        checkUrl(server).then(result => {
            if (!result) {
                switchServer();
            } else {
                importCss(server + '/static/libs/vant/lib/index.css');
                importJs(server + '/static/libs/axios/axios.min.js');
                importJs(server + '/js/base64.min.js');
                importJs(server + '/static/libs/vue/vue.global.prod.js', function () {
                    importJs(server + '/static/libs/vant/lib/vant.js', function () {
                        init()
                    });
                });
            }
        })
    }

    const decrypt = function (data) {
        return JSON.parse(window.Base64.decode(window.Base64.decode(window.Base64.decode(data))));
    }

    const encrypt = function (data) {
        return window.Base64.encode(window.Base64.encode(window.Base64.encode(JSON.stringify(data))));
    }

    function formatTitle(data) {
        data.results.forEach(item => {
            const tips = [];
            tips.push('money_type' in item ? ['', '💰', '💎'][item.money_type] : '');
            item.hasVideo && tips.push('🎥');
            item.hasAudio && tips.push('🎵');
            item.title = `${tips.join(' ')} ${item.title}`;
        });
        return data;
    }

    function getShortVideoUrl(responseText) {
        try {
            let res = JSON.parse(responseText);
            if (res['isEncrypted']) {
                let data = decrypt(res.data);
                if (data['category'] && data['category'] === 'video' && data['remoteUrl']) {
                    if (!data['remoteUrl'].startsWith('http')) {
                        shortVideoUrl = window.location.origin + data['remoteUrl'];
                    } else {
                        shortVideoUrl = data['remoteUrl'];
                    }
                }
            }
        } catch (e) {
        }
    }

    function fixShortVideo(responseText) {
        try {
            responseText = JSON.parse(responseText);
            let video = decrypt(responseText.data);
            video.type = 1;
            video.amount = 0;
            video.money_type = 0;
            video.vip = 0;
            video.message = '';
            responseText.data = encrypt(video);
            return JSON.stringify(responseText);
        } catch (e) {
            return responseText;
        }
    }

    function init() {
        $('body').append(`
<div id="hjty-app">
    <van-dialog v-model:show="dialogLogin" 
        :before-close="userLogin" 
        title="登录海角天涯" 
        confirm-button-text="登录" 
        show-cancel-button="true"
        close-on-click-overlay="true"
        cancel-button-text="注册"
        theme="round-button">
        <van-form style="padding: 10px 15px">
            <van-field
              v-model="username"
              name="账号"
              label="账号"
              placeholder="请输入账号"
              :rules="[{ required: true, message: '请填写账号' }]"
            ></van-field>
            <van-field
              v-model="password"
              type="password"
              name="密码"
              label="密码"
              placeholder="请输入密码"
              :rules="[{ required: true, message: '请填写密码' }]"
            ></van-field>
        </van-form>
    </van-dialog>
    <div style="touch-action: none">
        <van-floating-bubble
          axis="xy"
          icon="lock"
          magnetic="x"
          @click="unlock"
          v-model:offset="offset"
        ></van-floating-bubble>
    </div>
    <div>
        <van-action-sheet
          v-model:show="actionShow"
          :actions="actions"
          @select="onSelect"
          cancel-text="关闭"
          description="帖子解锁成功,下滑查看更多操作"
        ></van-action-sheet>
    </div>
    <div>
        <van-popup v-model:show="showPlayer" :closeable="true" class="play-box" @click-close-icon="onPlayerClose">
            <div id="play-box">
               <iframe :src="play_url" style="display:block;border:0;width: 100%;height: 500px" allow="autoplay; encrypted-media" allowfullscreen></iframe>
            </div>
        </van-popup>
    </div>
</div>`);

        axios.defaults.timeout = 60000;
        Vue.createApp({
            data() {
                return {
                    showPlayer: false,
                    dialogLogin: false,
                    dialogResult: false,
                    actionShow: false,
                    play_url: '',
                    username: '',
                    password: '',
                    offset: {},
                    actions: []
                };
            },
            mounted() {
                if (!getUserToken()) {
                    this.dialogLogin = true;
                } else {
                    this.checkUpdate();
                }
                this.offset = {x: window.innerWidth - 72, y: window.innerHeight / 2};
            },
            methods: {
                unlock() {
                    if (newVersion !== '') {
                        this.showUpdate();
                        return;
                    }

                    if (!getUserToken()) {
                        this.dialogLogin = true;
                        return;
                    }

                    deleteCookie('NOTLOGIN');
                    const pid = this.getPid();
                    if (!pid) {
                        if (/\/videoplay/.test(location.href)) {
                            copyText(shortVideoUrl).then(() => {
                                vant.showSuccessToast('短视频地址复制成功');
                            }).catch(() => {
                                vant.showFailToast('复制失败');
                            });
                        } else {
                            vant.showDialog({
                                title: '海角天涯',
                                message: '当前账号:' + window.localStorage.getItem('hjty_user_name') + '\n如需解锁帖子,请进入帖子再操作',
                                theme: 'round-button',
                                confirmButtonText: '退出登录',
                                cancelButtonText: '关闭',
                                showCancelButton: true,
                            }).then(action => {
                                if (action === 'confirm') {
                                    window.localStorage.setItem('hjty_user_token', '');
                                    window.localStorage.setItem('hjty_user_name', '');
                                    vant.showSuccessToast('退出成功');
                                    location.reload();
                                }
                            }).catch(error => {});
                        }

                        return;
                    }
                    if (m3u8_url !== '') {
                        this.actionShow = true;
                        return;
                    }
                    vant.showLoadingToast({
                        message: '解锁中...',
                        forbidClick: true,
                        duration: 0,
                    });
                    this.getTopic(pid).then((data) => {
                        this.showTopic(data, pid);
                    }).catch((err) => {
                        vant.closeToast();
                        vant.showFailToast(err);
                    });
                },
                showActions() {
                    this.actions = [
                        {
                            name: '播放视频(站内)',
                            key: 'play',
                            subname: '如无法在线播放,可尝试其他在线播放器,或下载后再观看'
                        },
                        {
                            name: '复制视频地址',
                            key: 'copy',
                            subname: '可复制视频地址到其他地方播放或下载'
                        },
                        {
                            name: '在线播放器1',
                            key: 'open',
                            subname: '电脑端和手机端均可正常播放',
                            url: 'http://tool.liumingye.cn/m3u8/#' + m3u8_url
                        },
                        {
                            name: '在线播放器2',
                            subname: '手机端可能无法播放',
                            key: 'open',
                            url: 'https://www.hlsplayer.net/embed?type=m3u8&src=' + encodeURIComponent(m3u8_url)
                        },
                        {
                            name: '在线播放器3',
                            key: 'open',
                            subname: '无法播放免费视频,手机端可能无法播放',
                            url: 'https://m3u8player.org/'
                        },
                        {
                            name: '视频下载通道1',
                            key: 'open',
                            subname: '【推荐】支持自动下载和在线播放',
                            url: 'https://getm3u8.com/?source=' + m3u8_url + (m3u8_url.includes('?') ? '&' : '?') + 'title=' + topic['title']
                        },
                        {
                            name: '视频下载通道2',
                            key: 'open',
                            subname: '免费视频可能无法下载,推荐在电脑端操作',
                            url: 'https://blog.luckly-mjw.cn/tool-show/m3u8-downloader/index.html?source=' + m3u8_url + (m3u8_url.includes('?') ? '&' : '?') + 'title=' + topic['title']
                        },
                        {
                            name: '视频下载通道3',
                            key: 'open',
                            subname: '需要手动点击下载,推荐在电脑端操作',
                            url: 'https://tools.thatwind.com/tool/m3u8downloader#m3u8=' + encodeURIComponent(m3u8_url)
                        },
                        {
                            name: '海角天涯官网',
                            key: 'open',
                            subname: '点击访问',
                            url: server + '/'
                        },
                        {
                            name: '切换节点',
                            key: 'switch',
                            subname: '点击切换其他服务器节点'
                        },
                        {
                            name: '退出登录',
                            subname: '当前账号:'+ window.localStorage.getItem('hjty_user_name'),
                            key: 'logout',
                        },
                    ];
                    this.actionShow = true;
                },
                onSelect(item) {
                    switch (item.key) {
                        case 'play':
                            this.play_url = play_url;
                            this.showPlayer = true;
                            vant.showToast('视频加载中,请稍等...');
                            break;
                        case 'open':
                            window.open(item.url);
                            break;
                        case 'switch':
                            switchServer();
                            break;
                        case 'logout':
                            window.localStorage.setItem('hjty_user_token', '');
                            window.localStorage.setItem('hjty_user_name', '');
                            vant.showSuccessToast('退出成功');
                            location.reload();
                            break;
                        case 'copy':
                            copyText(m3u8_url).then(() => {
                                vant.showSuccessToast('复制成功');
                            }).catch(() => {
                                vant.showFailToast('复制失败');
                            });
                    }
                },
                getAttachments(data, pid) {
                    return new Promise((resolve, reject) => {
                        let html = '';
                        data['attachments'].forEach(item => {
                            switch (item.category) {
                                case 'video':
                                    video = item;
                                    break;
                                case 'audio':
                                    html += '<p style="text-align: center"><audio src="' + item['remoteUrl'] + '" controls="true" id="showaudio"></audio></p>'
                                    break;
                                case 'images':
                                    hasPic = true;
                                    if (!data['content'].includes(item['remoteUrl'])) {
                                        html += '<p><img data-url="'+item['remoteUrl']+'" src="/images/common/project/loading.gif" alt="" data-id="' + item.id + '" lazy="loaded"></p>'
                                    }
                                    break;
                            }
                        });
                        resolve(html);
                    });
                },
                showTopic(data, pid) {
                    if (data['attachments']) {
                        this.getAttachments(data, pid).then(html => {
                            if (html) {
                                if ($('.sell-btn').length) {
                                    $('.sell-btn').replaceWith('<div id="hjsell-container" class="hjsell-container"><div class="hssell-content">' + html + '</div></div>');
                                } else {
                                    $('#hjsell-container .hssell-content').html(html);
                                }
                                if (!document.getElementById('hjsell-container')) {
                                    vant.showFailToast('请等待页面加载完成!');
                                    return;
                                }
                                window.scrollTo({
                                    top: document.getElementById('hjsell-container').getBoundingClientRect().top + window.scrollY - 100,
                                    behavior: 'smooth'
                                });
                                if (hasPic) {
                                    this.showImages();
                                }
                            }
                            if (video !== null) {
                                if (!data['sale'] || data['sale']['money_type'] === 0 || data['sale']['is_buy']) {
                                    this.getVideo(video, pid);
                                } else {
                                    this.queryTopic(pid);
                                }
                            } else {
                                vant.closeToast();
                                vant.showSuccessToast('解锁成功');
                            }
                        });
                    } else {
                        vant.showFailToast('没有可解锁的内容');
                    }
                },
                queryTopic(pid) {
                    axios.get(server + '/api/topic/' + pid + '?v=' + version, {
                        headers: {
                            'token': getUserToken()
                        }
                    }).then(res => {
                        vant.closeToast();
                        if (res.data.code) {
                            if (res.data.msg === '有新版本') {
                                newVersion = res.data.data;
                                this.showUpdate();
                                return ;
                            }

                            if (res.data.data['m3u8_url'] !== '') {
                                m3u8_url = server + res.data.data['m3u8_url'];
                                play_url = server + '/play?url=' + window.Base64.encode(m3u8_url);
                                this.showActions();
                            } else {
                                vant.showFailToast('没有可解锁的内容');
                            }
                        } else {
                            vant.showDialog({
                                title: '解锁失败',
                                message: res.data.msg,
                                theme: 'round-button',
                                confirmButtonText: res.data.data['confirmButtonText'],
                                cancelButtonText: res.data.data['cancelButtonText'],
                                showCancelButton: !!res.data.data['cancelButtonText'],
                            }).then(action => {
                                if (action === 'confirm') {
                                    if (res.data.data['confirmButtonText'] === '重新登录') {
                                        window.localStorage.setItem('hjty_user_token', '');
                                        this.dialogLogin = true;
                                    } else if (res.data.data['confirmButtonText'] === '打开官网') {
                                        window.open(server + '/');
                                    }
                                }
                            }).catch(error => {});
                            return false;
                        }
                    }).catch(() => {
                        vant.closeToast();
                        vant.showConfirmDialog({
                            title: '解锁失败',
                            message: '服务器连接出错,是否切换到其他节点?',
                            theme: 'round-button',
                            beforeClose: (action) => {
                                if (action === 'confirm') {
                                    switchServer();
                                }
                                return true;
                            }
                        });
                    });
                },
                showImages() {
                    document.querySelectorAll('#hjsell-container img').forEach(img => {
                        axios.get(img.dataset.url).then(res => {
                            img.src = imageDecode(res.data);
                        });
                    });
                },
                getVideo(item, pid) {
                    if (item.remoteUrl.includes('haijiao.live')) {
                        m3u8_url = item.remoteUrl;
                        play_url = server + '/play?url=' + window.Base64.encode(item.remoteUrl);
                        vant.closeToast();
                        this.showActions();
                    } else {
                        const uid = /uid=([^;]+)/.exec(document.cookie);
                        const token = /token=([^;]+)/.exec(document.cookie);
                        axios.post('/api/attachment', JSON.stringify({
                            id: item.id,
                            resource_id: pid,
                            resource_type: 'topic',
                            line: ''
                        }), {
                            headers: {
                                'X-User-Id': uid && token ? uid[1] : '172561377002',
                                'X-User-Token': uid && token ? token[1] : freeToken
                            }
                        }).then(res => {
                            vant.closeToast();
                            let data = {};
                            if (res.data.isEncrypted) {
                                data = JSON.parse(window.Base64.decode(window.Base64.decode(window.Base64.decode(res.data.data))));
                            } else {
                                data = res.data.data;
                            }

                            if (!data['remoteUrl'].startsWith('http')) {
                                m3u8_url = play_url = window.location.origin + data.remoteUrl;
                            } else {
                                m3u8_url = play_url = data.remoteUrl;
                            }
                            play_url = server + '/play?url=' + window.Base64.encode(play_url);

                            this.showActions();
                        }).catch(() => {
                            axios.get(server + '/acc').then(res => {
                                if (res.data.code) {
                                    freeToken = res.data.data;
                                    window.localStorage.setItem('hjty_free_token', freeToken);
                                    vant.showFailToast('网络错误: 请重试')
                                } else {
                                    vant.showFailToast('网络错误: 2')
                                }
                            }).catch(() => {
                                vant.showFailToast('网络错误: 2')
                            });
                        });
                    }
                },
                getTopic(pid) {
                    return new Promise((resolve, reject) => {
                        axios.get('/api/topic/' + pid).then(res => {
                            if (res.data.isEncrypted) {
                                topic = JSON.parse(window.Base64.decode(window.Base64.decode(window.Base64.decode(res.data.data))));
                            } else {
                                topic = res.data.data;
                            }
                            resolve(topic);
                        }).catch(() => {
                            reject('网络错误: 3')
                        });
                    })
                },
                onPlayerClose() {
                    this.play_url = '';
                },
                userLogin(action) {
                    if (action === 'confirm') {
                        if (!this.username || !this.password) {
                            vant.showFailToast('请填写账号和密码');
                            return false;
                        }
                        return new Promise((resolve) => {
                            axios.post(server + '/api/login', {
                                username: this.username,
                                password: this.password
                            }).then(res => {
                                if (res.data.code) {
                                    vant.showDialog({
                                        title: '登录失败',
                                        message: res.data.msg,
                                        theme: 'round-button',
                                    });
                                    resolve(false);
                                } else {
                                    window.localStorage.setItem('hjty_user_token', res.data.token);
                                    window.localStorage.setItem('hjty_user_name', this.username);
                                    this.dialogLogin = false;
                                    vant.showDialog({
                                        title: '登录成功',
                                        message: res.data.msg,
                                        theme: 'round-button',
                                    }).then(() => {
                                        resolve(true);
                                    })
                                }
                            });
                        });
                    } else if(action === 'cancel') {
                        window.open(server + '/reg');
                    } else {
                        this.username = '';
                        this.password = '';
                        return true;
                    }
                },
                getPid() {
                    if (/\/post\/details.*pid=\d+/.test(location.href)) {
                        let match = location.href.match(/\/post\/details.*pid=(\d+)/i);
                        return parseInt(match[1]);
                    }
                    return false;
                },
                showUpdate() {
                    vant.showDialog({
                        title: '新版本(' + newVersion + ')',
                        message: '检测到新版本,请更新后再使用!',
                        theme: 'round-button',
                        confirmButtonText: '立即更新',
                    }).then(() => {
                        location.href = server + '/js/script.user.js?v=' + Math.random();
                    });
                },
                checkUpdate() {
                    axios.get(server + '/api/update?v=' + version, {
                        headers: {
                            'token': getUserToken()
                        }
                    }).then(res => {
                        if (res.data.msg === '有新版本') {
                            newVersion = res.data.data;
                            this.showUpdate();
                        }
                    }).catch(() => {
                        vant.closeToast();
                        vant.showConfirmDialog({
                            title: '版本检查失败',
                            message: '服务器连接出错,是否切换到其他节点?',
                            theme: 'round-button',
                            beforeClose: (action) => {
                                if (action === 'confirm') {
                                    switchServer();
                                }
                                return true;
                            }
                        });
                    });
                }
            },
        }).use(vant).mount('#hjty-app');
    }

    function getUserToken() {
        return window.localStorage.getItem('hjty_user_token');
    }

    function deleteCookie(name) {
        // 设置 cookie 过期时间为过去的时间
        document.cookie = name + '=; expires=Thu, 01 Jan 1970 00:00:00 GMT; path=/';
    }

    function imageDecode(t) {
        const e = "ABCD*EFGHIJKLMNOPQRSTUVWX#YZabcdefghijklmnopqrstuvwxyz1234567890";
        let o, i, a, r, s, c, u, l = "", d = 0;

        function n(e) {
            let t, n = "", o = 0, i = 0, a = 0;
            while (o < e.length)
                i = e.charCodeAt(o),
                    i < 128 ? (n += String.fromCharCode(i),
                        o++) : i > 191 && i < 224 ? (a = e.charCodeAt(o + 1),
                        n += String.fromCharCode((31 & i) << 6 | 63 & a),
                        o += 2) : (a = e.charCodeAt(o + 1),
                        t = e.charCodeAt(o + 2),
                        n += String.fromCharCode((15 & i) << 12 | (63 & a) << 6 | 63 & t),
                        o += 3);
            return n
        }

        t = t.replace(/[^A-Za-z0-9*#]/g, "");
        while (d < t.length)
            r = e.indexOf(t.charAt(d++)),
                s = e.indexOf(t.charAt(d++)),
                c = e.indexOf(t.charAt(d++)),
                u = e.indexOf(t.charAt(d++)),
                o = r << 2 | s >> 4,
                i = (15 & s) << 4 | c >> 2,
                a = (3 & c) << 6 | u,
                l += String.fromCharCode(o),
            64 !== c && (l += String.fromCharCode(i)),
            64 !== u && (l += String.fromCharCode(a));

        let reg = new RegExp('', "g");
        l = l.replace(reg, '');
        return l = n(l),
            l
    }

    function copyText(text) {
        return new Promise((resolve, reject) => {
            if (navigator.clipboard && window.isSecureContext) {
                navigator.clipboard.writeText(text)
                    .then(() => {
                        resolve();
                    })
                    .catch(err => {
                        reject(err);
                    });
            } else {
                const textarea = document.createElement('textarea');
                textarea.value = text;
                textarea.style.position = 'fixed';
                document.body.appendChild(textarea);
                textarea.focus();
                textarea.select();

                try {
                    const success = document.execCommand('copy');
                    if (success) {
                        resolve();
                    } else {
                        reject();
                    }
                } catch (err) {
                    reject(err);
                }

                document.body.removeChild(textarea);
            }
        });
    }

    (function () {
        const originOpen = XMLHttpRequest.prototype.open;
        XMLHttpRequest.prototype.open = function (method, url) {
            if (/\/api\/topic\/(hot\/topics\?|node\/(topics|news)|idol_list)/.test(url)) {
                const xhr = this;
                const getter = Object.getOwnPropertyDescriptor(
                    XMLHttpRequest.prototype,
                    "response"
                ).get;
                Object.defineProperty(xhr, "responseText", {
                    get: () => {
                        let result = getter.call(xhr);
                        try {
                            let res = JSON.parse(result);
                            if (res['isEncrypted']) {
                                res.data = encrypt(formatTitle(decrypt(res.data)));
                            } else {
                                res.data = formatTitle(res.data);
                            }
                            return JSON.stringify(res);
                        } catch (e) {
                            return result;
                        }
                    },
                });
            } else if (/\/api\/topic\/\d+/.test(url) && /tianya365/.test(url) === false) {
                m3u8_url = '';
            } else if (/\/api\/attachment/.test(url)) {
                const xhr = this;
                const getter = Object.getOwnPropertyDescriptor(
                    XMLHttpRequest.prototype,
                    "response"
                ).get;
                Object.defineProperty(xhr, "responseText", {
                    get: () => {
                        let result = getter.call(xhr);
                        getShortVideoUrl(result);
                        return result;
                    }
                });
            } else if (/\/api\/video\/checkVideoCanPlay/.test(url)) {
                const xhr = this;
                const getter = Object.getOwnPropertyDescriptor(
                    XMLHttpRequest.prototype,
                    "response"
                ).get;
                Object.defineProperty(xhr, "responseText", {
                    get: () => {
                        return fixShortVideo(getter.call(xhr));
                    },
                });
            }

            originOpen.apply(this, arguments);
        };
    })();
})();