您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Greasy Fork is available in English.
新浪微博,查看用户关注「典型」用户的数量。一个常见的应用场景就是看ta关注了哪些你会拉黑的人。
// ==UserScript== // @name 你果然关注了这些人(微博特征用户关注检测) // @namespace http://tampermonkey.net/ // @version 0.5 // @description 新浪微博,查看用户关注「典型」用户的数量。一个常见的应用场景就是看ta关注了哪些你会拉黑的人。 // @author evalcony // @homepageURL https://github.com/evalcony/weibo_feature_user_maching // @match https://weibo.com/* // @match https://m.weibo.cn/* // @icon  // ==/UserScript== (function() { 'use strict'; // 这里填写要匹配的用户名,以英文逗号分隔 // 例如 'aaa','bbb','ccc' const target_user_list = [ '' ] // ---------------------------------------------------------------------------------------- const MY_COOKIE = document.cookie // 用户关注列表是否被禁止查看 let focus_list_forbid = false // 已经搜索过的用户缓存数据 let searched_user_cach = [] // 用户关注的 target_user_list 命中数 let searched_user_target_match_num_cache = [] // 用户关注的具体 target_user let searched_user_target_match_user_cache = new Map(); // 最大查询分页数 <= 11 (11是微博设置的查看上限) let max_page = 5 // 命中数 const buf = new ArrayBuffer(4); const total_match_num = new Int32Array(buf); // 命中的 user_list let match_user_list = [] let showFlag = 0 new MutationObserver((mutations) => { let mutation = mutations[0] showFlag += 1 if (showFlag % 2 == 0) { // 移除 document.querySelector('#my_div').remove() document.querySelector('#my_focus_user_div').remove() return } let popCard = mutation.target let nameCard = popCard.childNodes[0].childNodes[0].childNodes[2].childNodes[1] let hrefCard = popCard.childNodes[0].childNodes[1].childNodes[0].childNodes[0].childNodes[1] let username = nameCard.innerText let href = hrefCard.href // 获得uid let uid = getUserId(href) // console.log(href) console.log(username) search(uid, username) // 渲染 flushEle(popCard, username) }).observe(document.querySelector('.popcard'), { //childList: true, attributes: true, //subtree: true, }); // console.log(Atomics.load(total_match_num, 0)) function flushEle(popCard, name) { let div = makeFocusDiv(name) popCard.appendChild(div) let div_user_list = makeFocusUserListDiv(name) popCard.appendChild(div_user_list) } function getUserId(href) { // https://weibo.com/u/page/follow/123123123?relate=fans let l = href.indexOf('follow/') let r = href.indexOf('?') let text = href.substring(l+'follow/'.length, r) return text } function makeFocusDiv(name) { // 获取缓存的命中user数量 let cachedPos = cached(name) console.log(searched_user_cach) console.log(searched_user_target_match_num_cache) let num = 0 if (cachedPos != -1) { num = searched_user_target_match_num_cache[cachedPos] } else { num = '正在请求数据...请稍后' } let div = document.createElement('div'); div.id = 'my_div' if (focus_list_forbid) { div.textContent = '只有粉丝才能查看关注列表' } else { div.textContent = num } return div } function makeFocusUserListDiv(name) { let div = document.createElement('div'); div.id = 'my_focus_user_div' let user_list = '' let cachedPos = cached(name) if (cachedPos != -1) { let u_list = searched_user_target_match_user_cache.get(name) if (u_list.length > 0) { u_list.forEach(item => { const p = document.createElement('p'); p.textContent = item; div.appendChild(p); }); } } else { } // div.textContent = user_list return div } // 查询用户关注列表 async function search(uid, name) { if (cached(name) != -1) { // do cached job return } focus_list_forbid = false // 初始化 match_user_list = [] // 遍历关注列表 Atomics.store(total_match_num, 0, 0); // 原子写入 for (var i = 1; i <= max_page; i++) { await async_fetch(uid, i) } console.log("查询结果:" + name + "_" + Atomics.load(total_match_num, 0)) if (focus_list_forbid) { searched_user_cach.push(name) searched_user_target_match_num_cache.push('只有粉丝才能查看关注列表') searched_user_target_match_user_cache.set(name, []) } else { searched_user_cach.push(name) searched_user_target_match_num_cache.push('命中特征用户数量:' + Atomics.load(total_match_num, 0)) searched_user_target_match_user_cache.set(name, match_user_list) } } // 查找缓存数组的下标 function cached(name) { for (var i in searched_user_cach) { if (name == searched_user_cach[i]) { return i } } return -1 } // 同步请求数据 async function async_fetch(uid, page_num) { if (focus_list_forbid) { return } var myHeaders = new Headers(); myHeaders.append('Cookie', MY_COOKIE); // 设置Cookie myHeaders.append("Referer", "https://m.weibo.cn/"); myHeaders.append("Sec-Fetch-Mode", "cors"); myHeaders.append("Accept", "application/json, text/plain, */*"); myHeaders.append("Origin", "https://weibo.com"); myHeaders.append("User-Agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Safari/537.36"); myHeaders.append("Client-Version", "v2.40.83"); myHeaders.append("Content-Type", "application/json"); var requestOptions = { method: 'GET', headers: myHeaders, }; var url = 'https://weibo.com/ajax/friendships/friends?page=${page_num}&uid=${uid}' url = url.replace('${page_num}', page_num).replace('${uid}', uid); await fetch(url, requestOptions) .then(response => response.text()) .then(result => fetchPromise(result)) .catch(error => console.log('error', error)); } async function fetchPromise(result) { var r = result.indexOf('博主设置仅针对粉丝展示全部关注') if (r != -1) { console.log('博主设置仅针对粉丝展示全部关注') focus_list_forbid = true return } friendsDataSolver(result) } // 数据解析 function friendsDataSolver(result) { console.log('网络请求') const parsedData = JSON.parse(result); // 设置最大页数 let total_number = parsedData.total_number var total_page_size = parseInt(total_number/20) if (total_page_size < 11) { max_page = total_page_size } var users = parsedData.users for (var i in users) { const u = users[i] let f = isInTargetUserList(u.screen_name) //console.log(f + ' ' + u.screen_name) if (f) { console.log(f + ' ' + u.screen_name) Atomics.add(total_match_num, 0, 1) match_user_list.push(u.screen_name) } } } // 是否命中 function isInTargetUserList(name) { for (var i in target_user_list) { if (name == target_user_list[i]) { return true } } return false } })();