批量拉黑微博点赞用户 解除拉黑

用于批量拉黑微博点赞用户 解除拉黑

// ==UserScript==
// @name         批量拉黑微博点赞用户 解除拉黑
// @namespace    https://github.com/comdotwww/batchBlockWeiboLikeUser
// @version      1.1
// @description  用于批量拉黑微博点赞用户 解除拉黑
// @author       comdotwww
// @match        http*://service.account.weibo.com/reportspam*
// @grant        GM_xmlhttpRequest
// @run-at       document-end
// @license           GPL-3.0
// @icon         
// @homepageURL       https://github.com/comdotwww/batchBlockWeiboLikeUser
// @supportURL        https://github.com/comdotwww/batchBlockWeiboLikeUser/issues
// ==/UserScript==

(function() {
    'use strict';

    // 等待页面加载完成
    window.addEventListener('load', async function() {
        // 获取report元素
        const reportElement = document.querySelector('a.mod-btn') || document.querySelector('dl dd p.module-infobox').parentNode;
        if (!reportElement) {
            console.log('未找到report元素');
            return;
        }

        /**
         * 休眠指定毫秒
         *
         * @param {Number} time
         * @returns
         */
        function sleep(time) {
            return new Promise((resolve) => setTimeout(resolve, time));
        }


        /**
         * 获取点赞用户 uid 分页列表
         *
         * @param {string} actionData
         * @param {string} page
         * @returns
         */
        async function getLikePage(actionData, page = 1) {
            if (!actionData) {
                return {}
            }
            let uidArray = [0, []]
            return new Promise(function(resolve, reject) {
                let rnd = Math.floor(Math.random() * 1e4 + 1.5788995e12);
                GM_xmlhttpRequest({
                    method: 'GET',
                    url: `https://weibo.com/aj/like/object/big?ajwvr=6&object_type=comment&page=${page}&object_id=${actionData}&__rnd=${rnd}`,
                    headers: {
                        'Content-Type': 'application/json;charset=UTF-8',
                        'referer': 'https://weibo.com/',
                    },
                    onload(res) {
                        if (res.status === 200) {
                            const json = JSON.parse(res.responseText);
                            uidArray = [json.data.page.totalpage, Array.from(json.data.html.matchAll(/uid=['"](.+?)['"]/ig))
                                .map(x => x[1]), json.data.like_counts
                            ]
                        } else {
                            alert(res.statusText);
                        }
                        resolve(uidArray);
                    },
                    onerror: function(error) {
                        console.error('请求失败:', error);
                        alert('请求失败,请检查控制台');
                    },
                });
            });

        }

        /**
         * 循环获取所有点赞用户 uid 集合
         *
         * @param {string} actionData
         * @returns
         */
        async function getLikeData(actionData) {
            const uids = []
            const uuu = await getLikePage(actionData)
            const page = uuu[0] || 0
            const uid = uuu[1] || []
            uids.push(...uid);
            if (page > 1) {
                // 休眠
                await sleep(Math.floor(Math.floor((Math.random() * 1000) + 1000)));
                const pages = new Array(page - 1).fill(0).map((x, i) => i + 2)
                const datas = await Promise.all(pages.map(x => getLikePage(actionData, x)))
                datas.forEach(x => {
                    uids.push(...x[1])
                })
            }
            return uids
        }

        /**
         * 根据用户的 uid 拉黑指定用户
         *
         * @param {string} uid
         * @returns
         */
        async function blockUser(uid) {
            let rnd = Math.floor(Math.random() * 1e4 + 1.5788995e12);
            return new Promise(function(resolve, reject) {
                GM_xmlhttpRequest({
                    method: 'POST',
                    url: 'https://weibo.com/ajax/statuses/filterUser',
                    headers: {
                        'Content-Type': 'application/json; charset=utf-8',
                        'referer': `https://weibo.com/u/${uid}`,
                        'origin': 'https://weibo.com',
                    },
                    data: JSON.stringify({ "uid": uid, "status": 1, "interact": 1, "follow": 1, "__rnd": rnd }),
                    onload(res) {
                        resolve(JSON.parse(res.responseText));
                    }
                });
            });
        }


        /**
         * 根据用户的 uid 解除拉黑指定用户
         *
         * @param {string} uid
         * @returns
         */
        async function unblockUser(uid) {
            let rnd = Math.floor(Math.random() * 1e4 + 1.5788995e12);
            return new Promise(function(resolve, reject) {
                GM_xmlhttpRequest({
                    method: 'POST',
                    url: 'https://weibo.com/ajax/statuses/deleteFilters',
                    headers: {
                        'Content-Type': 'application/json; charset=utf-8',
                        'referer': `https://weibo.com/u/${uid}`,
                        'origin': 'https://weibo.com',
                    },
                    data: JSON.stringify({ "uid": uid, "__rnd": rnd }),
                    onload(res) {
                        resolve(JSON.parse(res.responseText));
                    }
                });
            });
        }

        /**
         * 获取用户黑名单信息
         *
         * @returns
         */
        async function getUserBlockListInfo() {
            return new Promise(function(resolve, reject) {
                GM_xmlhttpRequest({
                    method: 'GET',
                    url: 'https://weibo.com/ajax/setting/getFilteredUsers?page=1&__rnd=' + Math.floor(Math.random() * 1e4 + 1.5788995e12),
                    headers: {
                        'Content-Type': 'application/json; charset=utf-8',
                        'referer': 'https://weibo.com/set/shield?type=user',
                        'origin': 'https://weibo.com',
                    },
                    onload(res) {
                        resolve(JSON.parse(res.responseText));
                    }
                });
            });
        }

        /**
         * 获取用户VIP信息
         *
         * @returns
         */
        async function getUserInfo() {
            return new Promise(function(resolve, reject) {
                GM_xmlhttpRequest({
                    method: 'GET',
                    url: 'https://new.vip.weibo.cn/aj/vipcenter/home?F=vipcenter_jump&from=&lang=zh_CN',
                    headers: {
                        'Content-Type': 'application/json; charset=utf-8',
                        'referer': 'https://new.vip.weibo.cn/vipcenter?portrait_only=1&topnavstyle=1&immersiveScroll=100&F=vipcenter_jump',
                        'origin': 'https://new.vip.weibo.cn',
                    },
                    onload(res) {
                        resolve(JSON.parse(res.responseText));
                    }
                });
            });
        }

        /**
         * 拉黑所有点赞用户 uid 集合
         *
         * @param {Array} actionData
         */
        async function blockAll(actionData) {
            if (confirm('确认拉黑该微博/该评论所有点赞用户吗?') == false || !actionData) {
                return;
            }

            const link = this
            link.onclick = undefined

            this.innerText = `正在获取点赞用户...`
            const uids = await getLikeData(actionData)
            let failed = [...uids]



            const blockUsers = async function(evt) {
                const refailed = []
                link.innerText = `已拉黑 0/${failed.length}`

                for (var i = 0; i < failed.length; i++) {
                    // 休眠
                    if (i % 5 == 0) {
                        await sleep(Math.floor(Math.floor((Math.random() * 1000) + 1000)));
                    }
                    const data = await blockUser(failed[i])
                    if (data.ok != 1) refailed.push(failed[i])

                    link.innerText = `已拉黑 ${i + 1}/${failed.length}: ${data.card.desc1} ${data.card.desc2} ${data.card.title_sub}`

                    if (data.card.pic) {
                        link.style.display = 'flex';
                        link.style.alignItems = 'center'; // 使文字和图片垂直居中
                        // 创建图片元素
                        let image = document.createElement('img');
                        image.style.width = '16px'; // 设置图片宽度
                        image.style.height = '16px'; // 设置图片高度
                        image.style.marginLeft = '5px'; // 设置图片与文字的间距
                        image.src = data.card.pic;
                        // 将图片插入到按钮中
                        link.appendChild(image);
                    }
                }

                if (refailed.length > 0) {
                    link.innerText = `再次尝试拉黑剩余${refailed.length}人`
                    link.onclick = blockUsers

                } else {
                    link.onclick = undefined
                    link.innerText = uids.length + '人已全部拉黑'
                    link.style.color = '#CCC'
                }

                failed = refailed
                evt && evt.stopPropagation()
            }
            await blockUsers()

        }

        /**
         * 取消拉黑所有点赞用户 uid 集合
         *
         * @param {Array} actionData
         */
        async function unblockAll(actionData) {
            if (confirm('确认取消拉黑该微博/该评论所有点赞用户吗?') == false || !actionData) {
                return;
            }

            const link = this
            link.onclick = undefined

            this.innerText = `正在获取点赞用户...`
            const uids = await getLikeData(actionData)
            let failed = [...uids]

            const blockUsers = async function(evt) {
                const refailed = []
                link.innerText = `已取消拉黑 0/${failed.length}`

                for (var i = 0; i < failed.length; i++) {
                    // 休眠
                    if (i % 5 == 0) {
                        await sleep(Math.floor(Math.floor((Math.random() * 1000) + 1000)));
                    }
                    const data = await unblockUser(failed[i])
                    if (data.ok != 1) refailed.push(failed[i])

                    link.innerText = `已取消拉黑 ${i + 1}/${failed.length}: ${data.card.title_sub}`

                    if (data.card.pic) {
                        link.style.display = 'flex';
                        link.style.alignItems = 'center'; // 使文字和图片垂直居中
                        // 创建图片元素
                        let image = document.createElement('img');
                        image.style.width = '16px'; // 设置图片宽度
                        image.style.height = '16px'; // 设置图片高度
                        image.style.marginLeft = '5px'; // 设置图片与文字的间距
                        image.src = data.card.pic;
                        // 将图片插入到按钮中
                        link.appendChild(image);
                    }
                }

                if (refailed.length > 0) {
                    link.innerText = `再次尝试取消拉黑剩余${refailed.length}人`
                    link.onclick = blockUsers

                } else {
                    link.onclick = undefined
                    link.innerText = uids.length + '人已全部取消拉黑'
                    link.style.color = '#CCC'
                }

                failed = refailed
                evt && evt.stopPropagation()
            }
            await blockUsers()

        }

        // 获取hid元素
        const hidElement = document.getElementById('extra_data');
        if (!hidElement) {
            console.log('未找到extra_data元素');
            return;
        }
        // 解析hid元素中的value参数
        const params = new URLSearchParams(hidElement.value);

        // 当前用户黑名单信息
        const userBlockListInfo = document.createElement('div');
        userBlockListInfo.classList.add('mod-txt');
        userBlockListInfo.textContent = '';

        // 创建“拉黑评论用户”按钮
        const blockCommentUserButton = document.createElement('button');
        blockCommentUserButton.textContent = '拉黑该用户';
        blockCommentUserButton.style.marginLeft = '10px';
        blockCommentUserButton.style.padding = '5px 10px';
        blockCommentUserButton.style.backgroundColor = '#ff4d4f';
        blockCommentUserButton.style.color = '#fff';
        blockCommentUserButton.style.border = 'none';
        blockCommentUserButton.style.borderRadius = '4px';
        blockCommentUserButton.style.cursor = 'pointer';
        blockCommentUserButton.style.marginTop = '10px'; // 设置容器与上方元素的间距

        // 创建“取消拉黑评论用户”按钮
        const unblockCommentUserButton = document.createElement('button');
        unblockCommentUserButton.textContent = '取消拉黑该用户';
        unblockCommentUserButton.style.marginLeft = '10px';
        unblockCommentUserButton.style.padding = '5px 10px';
        unblockCommentUserButton.style.backgroundColor = '#228B22';
        unblockCommentUserButton.style.color = '#fff';
        unblockCommentUserButton.style.border = 'none';
        unblockCommentUserButton.style.borderRadius = '4px';
        unblockCommentUserButton.style.cursor = 'pointer';
        unblockCommentUserButton.style.marginTop = '10px'; // 设置容器与上方元素的间距

        // 创建“拉黑点赞用户”按钮
        const blockLikeUserButton = document.createElement('button');
        const firstLikePage = await getLikePage(params.get('rid'), 1);
        blockLikeUserButton.textContent = `拉黑点赞用户:共 ${firstLikePage[2]||0} 位`;
        blockLikeUserButton.style.marginLeft = '10px';
        blockLikeUserButton.style.padding = '5px 10px';
        blockLikeUserButton.style.backgroundColor = '#ff4d4f';
        blockLikeUserButton.style.color = '#fff';
        blockLikeUserButton.style.border = 'none';
        blockLikeUserButton.style.borderRadius = '4px';
        blockLikeUserButton.style.cursor = 'pointer';
        blockLikeUserButton.style.marginTop = '10px'; // 设置容器与上方元素的间距

        // 创建“解除拉黑点赞用户”按钮
        const unblockLikeUserButton = document.createElement('button');
        unblockLikeUserButton.textContent = `解除拉黑点赞用户:共 ${firstLikePage[2]||0} 位`;
        unblockLikeUserButton.style.marginLeft = '10px';
        unblockLikeUserButton.style.padding = '5px 10px';
        unblockLikeUserButton.style.backgroundColor = '#228B22';
        unblockLikeUserButton.style.color = '#fff';
        unblockLikeUserButton.style.border = 'none';
        unblockLikeUserButton.style.borderRadius = '4px';
        unblockLikeUserButton.style.cursor = 'pointer';
        unblockLikeUserButton.style.marginTop = '10px'; // 设置容器与上方元素的间距

        // 拉黑提提示信息
        const blockInfo = document.createElement('div')
        blockInfo.innerHTML = '<div class="mod-txt" style="margin-top: 10px;">屏蔽拉黑用户,目前非会员用户可以屏蔽或拉黑5000个微博用户,会员用户屏蔽上限如下(更新时间:2024年3月5日),信息来源:<a href="https://kefu.weibo.com/faqdetail?id=18937" target="_blank">微博客服</a><br><img src="https://cs.s.weibo.com/knowledge/atts/21a423806fbb28f99f818acd6f0da0bf.jpg"></div>'

        reportElement.parentNode.style.flexDirection = 'column'; // 垂直排列

        // 将按钮插入到report元素后面
        reportElement.parentNode.insertBefore(userBlockListInfo, reportElement.nextSibling);
        // 换行
        const br0 = document.createElement('div')
        br0.innerHTML = ''
        reportElement.parentNode.insertBefore(br0, userBlockListInfo.nextSibling);
        reportElement.parentNode.insertBefore(blockCommentUserButton, br0.nextSibling);

        const br1 = document.createElement('div')
        br1.innerHTML = ''
        reportElement.parentNode.insertBefore(br1, blockCommentUserButton.nextSibling);
        reportElement.parentNode.insertBefore(unblockCommentUserButton, br1.nextSibling);

        const br2 = document.createElement('div')
        br2.innerHTML = ''
        reportElement.parentNode.insertBefore(br2, unblockCommentUserButton.nextSibling);
        reportElement.parentNode.insertBefore(blockLikeUserButton, br2.nextSibling);
        const br3 = document.createElement('div')
        br3.innerHTML = ''
        reportElement.parentNode.insertBefore(br3, blockLikeUserButton.nextSibling);
        reportElement.parentNode.insertBefore(unblockLikeUserButton, br3.nextSibling);
        const br4 = document.createElement('div')
        br4.innerHTML = ''
        reportElement.parentNode.insertBefore(br4, unblockLikeUserButton.nextSibling);
        reportElement.parentNode.insertBefore(blockInfo, br4.nextSibling);

        const userBlockData = await getUserBlockListInfo();
        if (userBlockData && userBlockData.total) {
            userBlockListInfo.textContent = `当前用户黑名单人数: ${userBlockData.total} 人, 用户VIP信息: `;
            const userVipData = await getUserInfo();
            if (userVipData && userVipData.data.baseInfo.user_info.gif_icon) {
                // 创建图片元素
                const image = document.createElement('img');
                image.src = userVipData.data.baseInfo.user_info.gif_icon
                image.style.width = '50px'; // 设置图片宽度
                image.style.height = '25px'; // 设置图片高度

                // 将图片插入到按钮中
                userBlockListInfo.appendChild(image);
            } else {
                userBlockListInfo.textContent = `当前用户黑名单人数: ${userBlockData.total} 人, 用户VIP信息: 无`;
            }
        }

        // 为“拉黑评论用户”按钮添加点击事件
        blockCommentUserButton.onclick = async function() {
            const rUid = params.get('r_uid');
            if (rUid) {
                // 当前用户昵称
                const n = document.querySelector('.mod-user .m-box .mod-text-box p a') || document.querySelector('dl dd p.module-infobox a')
                const nickname = n ? n.innerText : ''
                if (confirm(`确认拉黑该微博/该评论用户:${nickname} 吗?`) == true) {
                    // 在这里添加拉黑评论用户的逻辑
                    const data = await blockUser(rUid)
                    if (data.ok != 1) {
                        alert('拉黑失败,请重试。');
                    } else {
                        this.onclick = undefined
                        this.innerText = '已拉黑 ' + data.card.title_sub
                        this.style.color = '#CCC'
                        if (data.card.pic) {
                            this.style.display = 'flex';
                            this.style.alignItems = 'center'; // 使文字和图片垂直居中
                            // 创建图片元素
                            const image = document.createElement('img');
                            image.src = data.card.pic;
                            image.style.width = '16px'; // 设置图片宽度
                            image.style.height = '16px'; // 设置图片高度
                            image.style.marginLeft = '5px'; // 设置图片与文字的间距

                            // 将图片插入到按钮中
                            this.appendChild(image);
                        }
                    }
                }
            } else {
                alert('未找到用户id参数');
            }
        };

        // 为“取消拉黑评论用户”按钮添加点击事件
        unblockCommentUserButton.onclick = async function() {
            const rUid = params.get('r_uid');
            if (rUid) {
                // 当前用户昵称
                const n = document.querySelector('.mod-user .m-box .mod-text-box p a') || document.querySelector('dl dd p.module-infobox a')
                const nickname = n ? n.innerText : ''
                if (confirm(`确认取消拉黑该微博/该评论用户:${nickname} 吗?`) == true) {
                    // 在这里添加拉黑评论用户的逻辑
                    const data = await unblockUser(rUid)
                    if (data.ok != 1) {
                        alert('取消拉黑失败,请重试。');
                    } else {
                        this.onclick = undefined
                        this.innerText = '已取消拉黑 ' + data.card.title_sub
                        this.style.color = '#CCC'
                        if (data.card.pic) {
                            this.style.display = 'flex';
                            this.style.alignItems = 'center'; // 使文字和图片垂直居中
                            // 创建图片元素
                            const image = document.createElement('img');
                            image.src = data.card.pic;
                            image.style.width = '16px'; // 设置图片宽度
                            image.style.height = '16px'; // 设置图片高度
                            image.style.marginLeft = '5px'; // 设置图片与文字的间距

                            // 将图片插入到按钮中
                            this.appendChild(image);
                        }
                    }
                }
            } else {
                alert('未找到用户id参数');
            }
        };

        if ((firstLikePage[2] || 0) == 0) {
            blockLikeUserButton.onclick = undefined
            unblockLikeUserButton.onclick = undefined
        } else {
            // 为“拉黑点赞用户”按钮添加点击事件
            blockLikeUserButton.onclick = blockAll.bind(blockLikeUserButton, params.get('rid'));
            // 为“取消拉黑点赞用户”按钮添加点击事件
            unblockLikeUserButton.onclick = unblockAll.bind(unblockLikeUserButton, params.get('rid'));
        }

    });
})();