你果然关注了这些人(微博特征用户关注检测)

新浪微博,查看用户关注「典型」用户的数量。一个常见的应用场景就是看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
}

})();