新浪微博一键清空助手

一键清空微博、清空关注、清空粉丝、清空收藏、清空点赞记录

/* eslint-disable no-unused-vars */
/* eslint-disable no-extra-semi */
/* eslint-disable no-console */
// ==UserScript==
// @name         新浪微博一键清空助手
// @namespace    https://gf.qytechs.cn/zh-CN/users/1452603-moreanx
// @version      0.0.1
// @description  一键清空微博、清空关注、清空粉丝、清空收藏、清空点赞记录
// @author       MoreanX
// @match        https://weibo.com/*
// @icon         https://www.google.com/s2/favicons?domain=weibo.com
// @license      MIT
// @require      https://lf9-cdn-tos.bytecdntp.com/cdn/expire-1-M/jquery/1.11.3/jquery.js
// @require      https://lf3-cdn-tos.bytecdntp.com/cdn/expire-1-M/jquery-cookie/1.4.1/jquery.cookie.js
// @grant        none
// ==/UserScript==

; (function () {
	'use strict'

	const jq = window.jQuery

	const HELPER_NAME = '微博一键清空助手'
	const TOKEN = jq.cookie('XSRF-TOKEN')
	const WB_CONFIG = window.$CONFIG
	const UID = WB_CONFIG.uid
	const USER = WB_CONFIG.user

	const showNewWeoboTip = () => {
		const newWeiboEntry = jq('a[action-type="changeversion"]')

		if (!newWeiboEntry[0]) {
			return setTimeout(showNewWeoboTip, 500)
		}

		const tip = jq('<div />')

		tip
			.css({
				position: 'fixed',
				top: 70,
				left: 10,
				width: 200,
				height: 30,
				color: '#f00',
				background: '#fff',
				border: '1px solid #f00',
				lineHeight: '30px',
				textAlign: 'center',
				cursor: 'pointer',
			})
			.text('当前是旧版,是否切换到新版?')
			.click(() => {
				if (newWeiboEntry[0]) {
					newWeiboEntry[0].click()
				}
			})

		jq('#plc_frame').append(tip)
	}

	if (!USER) {
		return showNewWeoboTip()
	}

	const STATUSES_COUNT = USER.statuses_count
	const FRIENDS_COUNT = USER.friends_count
	const FOLLOWERS_COUNT = USER.followers_count
	const URL_PREFIX = 'https://weibo.com/u'
	const c_app = jq('#app')
	const c_menu = jq('<div />')
	const c_notice = jq('<div />')
	const c_btn = jq('<div />')

	if (!UID) return

	// 当前删除页码
	let deletePage
	// 已删除数
	let deletedCount
	// 停止清空
	let stop
	// 折叠菜单
	let fold

	const utils = {
		// alert fail
		alertFail: (jqXHR, textStatus, errorThrown) => {
			var error = '状态码:' + jqXHR.status + ',异常:' + errorThrown
			alert('读取数据失败,请稍后重试\n' + error)
		},

		// 检查是否在当前页
		checkURL: (url, title) => {
			const isCurrent = window.location.href.indexOf(url) !== -1
			if (!isCurrent) {
				utils.showConfirm(
					`当前操作需要前往 ${title} 页面,是否跳转?`,
					() => { window.location.href = url },
					() => { }
				)
			}
			return isCurrent
		},

		// 输出提示信息
		showNotice: html => {
			c_notice.show().html(`<div style="padding: 5px;"> ${html} </div>`)
		},

		// 显示删除进度
		showDeleteNotice: (count, no) => {
			if (count === null) {
				utils.showNotice(` <div> <div>正在删除第 ${deletePage} 页,第 ${no} 条</div> </div> `)
			} else {
				// 剩余数
				const remain = count - deletedCount

				utils.showNotice(`
          <div>
            <div>总共 ${count} 条</div>
            <div style="border-bottom 1px solid #000;">剩余 ${remain} 条</div>
            	<div>正在删除第 ${deletePage} 页,第 ${no} 条</div>
			</div>`)
			}
		},

		// log
		log: (...args) => {
			console.log(`${HELPER_NAME}:`, ...args)
		},

		// 串行Promise
		serialPromise: (promises, callback) => {
			let i = 0

			const next = () => {
				if (i < promises.length) {
					promises[i++]().then(next)
				} else {
					callback()
				}
			}

			next()
		},
	}

	utils.log('微博 token = ', TOKEN)
	utils.log('window.$CONFIG =', WB_CONFIG)
	utils.log('uid = ' + UID)

	// 重置
	const reset = () => {
		deletePage = 0
		deletedCount = 0
		stop = false
		fold = false
	}

	// 结束
	const end = () => {
		utils.log('删除完成')
		utils.showNotice(`<div style="color:#52c41a;">✓ 删除完成</div > `)
		c_btn.hide()
		setTimeout(() => {
			utils.showConfirm(
				'操作已完成,是否刷新页面?',
				() => { location.reload() },
				() => { }
			)
		}, 100)
	}

	// 显示确认对话框
	utils.showConfirm = (message, onConfirm, onCancel) => {
		utils.showNotice(`
        <div style="margin-bottom:12px;color:#333;font-size:14px;line-height:1.5;">
            ${message}
		</div>
        <div style="display:flex;justify-content:space-between;gap:10px;">
            <button id="confirmBtn" style="${BTN_STYLE} background:#ff4d4f;flex:1;">确定</button>
            <button id="cancelBtn" style="${BTN_STYLE} background:#f0f0f0;flex:1;">取消</button>
		</div>
    `)
		jq('#confirmBtn').click(() => {
			c_notice.hide()
			onConfirm()
		})
		jq('#cancelBtn').click(() => {
			c_notice.hide()
			onCancel()
		})
	}

	/** ===== 清空微博 ===== */

	// 清空微博
	const cleanWeibo = () => {
		if (!utils.checkURL(URL_PREFIX + '/' + UID, '我的主页')) return
		utils.showConfirm(
			'确定要清空所有微博吗?此操作不可撤销!',
			() => {
				reset()
				c_btn.show()
				utils.showNotice(`<div style="color:#666;"> 正在准备删除微博... </div > `)
				getWeiboList()
			},
			() => { }
		)
	}

	// 获取微博列表
	const getWeiboList = (page = 1) => {
		if (stop) return

		jq.ajax({
			url: '/ajax/statuses/mymblog?uid=' + UID + '&page=' + page + '&feature=0',
			type: 'GET',
			dataType: 'json',
		})
			.done(function (res) {
				utils.log('获取微博分页', res)
				if (res && res.data && res.data.list) {
					if (res.data.list.length === 0) {
						// 如果第2页也没有,则结束
						if (page === 2) {
							end()
						} else {
							// 第1页没有微博,有可能是微博bug,去第2页看看
							getWeiboList(2)
						}

						return
					}

					deletePage++
					utils.log('第 ', deletePage, ' 页')

					// 循环promise
					const promisesTask = res.data.list.map((item, index) => {
						return () =>
							new Promise(resolve => {
								const oriMid = item.ori_mid
								const id = item.id
								const no = index + 1

								if (stop) return

								utils.log('待删除微博', no, id)
								utils.showDeleteNotice(STATUSES_COUNT, no)

								if (oriMid) {
									// 删除快转
									deleteWeibo(oriMid).done(resolve)
								} else {
									// 正常删除
									deleteWeibo(id).done(resolve)
								}
							})
					})

					utils.serialPromise(promisesTask, () => {
						setTimeout(() => {
							getWeiboList()
						}, 2000)
					})
				}
			})
			.fail(utils.alertFail)
	}

	// 删除微博
	const deleteWeibo = id => {
		const postData = { id: id }

		return jq
			.ajax({
				url: '/ajax/statuses/destroy',
				contentType: 'application/json;charset=UTF-8',
				type: 'POST',
				dataType: 'json',
				headers: {
					'x-xsrf-token': TOKEN,
				},
				data: JSON.stringify(postData),
			})
			.done(function (res) {
				deletedCount++
				utils.log('已删除微博', id, res)
			})
			.fail(utils.alertFail)
	}

	/** ===== 清空关注列表 ===== */

	// 清空关注列表
	const cleanFollow = () => {
		if (!utils.checkURL(URL_PREFIX + '/page/follow/' + UID, '我的关注')) return
		utils.showConfirm(
			'确定要清空所有关注的人吗?此操作不可撤销!',
			() => {
				reset()
				c_btn.show()
				utils.showNotice(`<div style="color:#666;"> 在准备删除关注用户... </div > `)
				getFollowList()
			},
			() => { }
		)
	}

	// 获取微博关注列表
	const getFollowList = () => {
		if (stop) return

		jq.ajax({
			url: '/ajax/friendships/friends?uid=' + UID + '&page=1',
			type: 'GET',
			dataType: 'json',
		})
			.done(function (res) {
				utils.log('获取微博关注分页', res)
				if (res && res.users) {
					if (res.users.length === 0) {
						return end()
					}

					deletePage++

					utils.log('第 ', deletePage, ' 页')

					// 循环promise
					const promisesTask = res.users.map((item, index) => {
						return () =>
							new Promise(resolve => {
								setTimeout(() => {
									const id = item.id
									const no = index + 1

									if (stop) return

									utils.log('待删除关注用户', no, id)
									utils.showDeleteNotice(FRIENDS_COUNT, no)
									deleteFollow(id).done(resolve)
								}, Math.random() * 500 + 500)
							})
					})

					utils.serialPromise(promisesTask, () => {
						setTimeout(() => {
							getFollowList()
						}, 1000)
					})
				}
			})
			.fail(utils.alertFail)
	}

	// 取消关注
	const deleteFollow = id => {
		const postData = { uid: id }

		return jq
			.ajax({
				// 注:微博接口单词拼写错误,应该是 destroy
				url: '/ajax/friendships/destory',
				contentType: 'application/json;charset=UTF-8',
				type: 'POST',
				dataType: 'json',
				headers: {
					'x-xsrf-token': TOKEN,
				},
				data: JSON.stringify(postData),
			})
			.done(function (res) {
				deletedCount++
				utils.log('已取消关注', id, res)
			})
			.fail(utils.alertFail)
	}

	/** ===== 清空粉丝列表 ===== */

	// 清空粉丝列表
	const cleanFans = () => {
		const url = URL_PREFIX + '/page/follow/' + UID + '?relate=fans'
		if (!utils.checkURL(url, '我的粉丝')) return
		utils.showConfirm(
			'确定要清空所有粉丝吗?此操作不可撤销!',
			() => {
				reset()
				c_btn.show()
				utils.showNotice(`<div style="color:#666;"> 正在准备移除粉丝... </div > `)
				getFansList()
			},
			() => { }
		)
	}

	// 获取微博粉丝列表
	const getFansList = () => {
		if (stop) return

		jq.ajax({
			url: '/ajax/friendships/friends?uid=' + UID + '&relate=fans&page=1',
			type: 'GET',
			dataType: 'json',
		})
			.done(function (res) {
				utils.log('获取微博粉丝分页', res)
				if (res && res.users) {
					if (res.users.length === 0) {
						return end()
					}

					deletePage++

					utils.log('第 ', deletePage, ' 页')

					// 循环promise
					const promisesTask = res.users.map((item, index) => {
						return () =>
							new Promise(resolve => {
								setTimeout(() => {
									const id = item.id
									const no = index + 1

									if (stop) return

									utils.log('待删除粉丝', no, id)
									utils.showDeleteNotice(FOLLOWERS_COUNT, no)
									deleteFans(id).done(resolve)
								}, Math.random() * 500 + 500)
							})
					})

					utils.serialPromise(promisesTask, () => {
						setTimeout(() => {
							getFansList()
						}, 1000)
					})
				}
			})
			.fail(utils.alertFail)
	}

	// 移除粉丝
	const deleteFans = id => {
		const postData = { uid: id }

		return jq
			.ajax({
				url: '/ajax/profile/destroyFollowers',
				contentType: 'application/json;charset=UTF-8',
				type: 'POST',
				dataType: 'json',
				headers: {
					'x-xsrf-token': TOKEN,
				},
				data: JSON.stringify(postData),
			})
			.done(function (res) {
				deletedCount++
				utils.log('已删除粉丝', id, res)
			})
			.fail(utils.alertFail)
	}

	/** ===== 清空赞列表 ===== */

	// 清空赞列表
	const cleanLike = () => {
		const url = URL_PREFIX + '/page/like/' + UID
		if (!utils.checkURL(url, '我的赞')) return
		utils.showConfirm(
			'确定要清空所有的赞吗?此操作不可撤销!',
			() => {
				reset()
				c_btn.show()
				utils.showNotice(`<div style="color:#666;">正在准备移除赞...</div > `)
				getLikeList()
			},
			() => { }
		)
	}

	// 获取微博赞列表
	const getLikeList = () => {
		if (stop) return

		// 微博好像有bug,第1页的赞被删除后,后面的列表就无法显示,所以暂时不删除第1页数据
		if (deletePage === 0) {
			deletePage = 1
		}

		jq.ajax({
			url: '/ajax/statuses/likelist?uid=' + UID + '&relate=fans&page=1',
			type: 'GET',
			dataType: 'json',
		})
			.done(function (res) {
				utils.log('获取微博赞分页', res)
				if (res && res.data && res.data.list) {
					if (res.data.list.length === 0) {
						return end()
					}

					deletePage++

					utils.log('第 ', deletePage, ' 页')

					// 循环promise
					const promisesTask = res.data.list.map((item, index) => {
						return () =>
							new Promise(resolve => {
								setTimeout(() => {
									const id = item.id
									const no = index + 1

									if (stop) return

									utils.log('待删除赞', no, id)
									utils.showDeleteNotice(null, no)
									deleteLike(id).done(resolve)
								}, Math.random() * 500 + 500)
							})
					})

					utils.serialPromise(promisesTask, () => {
						setTimeout(() => {
							getLikeList()
						}, 1000)
					})
				}
			})
			.fail(utils.alertFail)
	}

	// 移除赞
	const deleteLike = id => {
		const postData = { id: String(id) }

		return jq
			.ajax({
				url: '/ajax/statuses/cancelLike',
				contentType: 'application/json;charset=UTF-8',
				type: 'POST',
				dataType: 'json',
				headers: {
					'x-xsrf-token': TOKEN,
				},
				data: JSON.stringify(postData),
			})
			.done(function (res) {
				deletedCount++
				utils.log('已删除赞', id, res)
			})
			.fail(utils.alertFail)
	}

	/** ===== 清空收藏列表 ===== */

	// 清空收藏列表
	const cleanFav = () => {
		const url = URL_PREFIX + '/page/fav/' + UID
		if (!utils.checkURL(url, '我的收藏')) return
		utils.showConfirm(
			'确定要清空所有的收藏吗?此操作不可撤销!',
			() => {
				reset()
				c_btn.show()
				utils.showNotice(`<div style="color:#666;">正在准备移除收藏...</div > `)
				getFavList()
			},
			() => { }
		)
	}

	// 获取微博收藏列表
	const getFavList = () => {
		if (stop) return

		jq.ajax({
			url: '/ajax/favorites/all_fav?uid=' + UID + '&page=1',
			type: 'GET',
			dataType: 'json',
		})
			.done(function (res) {
				utils.log('获取微博收藏分页', res)
				if (res && res.data) {
					if (res.data.length === 0) {
						return end()
					}

					deletePage++

					utils.log('第 ', deletePage, ' 页')

					// 循环promise
					const promisesTask = res.data.map((item, index) => {
						return () =>
							new Promise(resolve => {
								setTimeout(() => {
									const id = item.id
									const no = index + 1

									if (stop) return

									utils.log('待删除收藏', no, id)
									utils.showDeleteNotice(null, no)
									deleteFav(id).done(resolve)
								}, Math.random() * 500 + 500)
							})
					})

					utils.serialPromise(promisesTask, () => {
						setTimeout(() => {
							getFavList()
						}, 1000)
					})
				}
			})
			.fail(utils.alertFail)
	}

	// 移除收藏
	const deleteFav = id => {
		const postData = { id: String(id) }

		return jq
			.ajax({
				url: '/ajax/statuses/destoryFavorites',
				contentType: 'application/json;charset=UTF-8',
				type: 'POST',
				dataType: 'json',
				headers: {
					'x-xsrf-token': TOKEN,
				},
				data: JSON.stringify(postData),
			})
			.done(function (res) {
				deletedCount++
				utils.log('已删除收藏', id, res)
			})
			.fail(utils.alertFail)
	}


	// 统一按钮样式
	const BTN_STYLE = `
    padding: 8px 16px;
    border: none;
    border-radius: 4px;
    background: #f0f0f0;
    color: #333;
    font-size: 14px;
    cursor: pointer;
    transition: all 0.2s;
    min-width: 80px;
    margin: 0 5px;`


	/** ===== 初始化 ===== */

	// 修改菜单按钮样式
	const initMenu = () => {
		const menuList = [
			{ text: '清空微博', onClick: cleanWeibo },
			{ text: '清空关注', onClick: cleanFollow },
			{ text: '清空粉丝', onClick: cleanFans },
			{ text: '清空收藏', onClick: cleanFav },
			{ text: '清空赞', onClick: cleanLike }
		]

		c_menu.css({
			position: 'fixed',
			top: '80px',
			left: '10px',
			zIndex: '9999',
			fontFamily: 'system-ui, -apple-system, sans-serif'
		})

		const hideBtn = jq('<button>')
			.css({
				width: '80px',
				height: '36px',
				background: '#f0f0f0',
				border: '1px solid #d9d9d9',
				borderRadius: '4px',
				cursor: 'pointer',
				fontSize: '14px',
				color: '#666',
				marginBottom: '8px'
			})
			.text('收起菜单')
			.hover(
				() => jq(this).css({ background: '#e6e6e6' }),
				() => jq(this).css({ background: '#f0f0f0' })
			)
			.click(() => {
				fold = !fold
				jq(this).text(fold ? '展开菜单' : '收起菜单')
				container.toggle()
			})

		const container = jq('<div>').css({
			width: '140px',
			background: '#fff',
			borderRadius: '4px',
			border: '1px solid #e8e8e8',
			boxShadow: '0 2px 10px rgba(0,0,0,0.1)'
		})

		menuList.forEach((item, index) => {
			jq('<button>')
				.css({
					display: 'block',
					width: '100%',
					padding: '10px 16px',
					border: 'none',
					borderBottom: index === menuList.length - 1 ? 'none' : '1px solid #f0f0f0',
					background: 'none',
					color: '#333',
					textAlign: 'left',
					fontSize: '14px',
					cursor: 'pointer',
					transition: 'all 0.2s'
				})
				.hover(
					() => jq(this).css({ background: '#f5f5f5', color: '#1890ff' }),
					() => jq(this).css({ background: 'none', color: '#333' })
				)
				.text(item.text)
				.click(item.onClick)
				.appendTo(container)
		})

		c_menu.append(hideBtn, container)
		c_app.append(c_menu)
	}

	// 修改停止按钮样式
	const initBtn = () => {
		c_btn.css({
			display: 'none',
			position: 'fixed',
			top: '70px',
			right: '10px',
			padding: '10px 20px',
			background: '#ff4d4f',
			color: 'white',
			border: 'none',
			borderRadius: '4px',
			fontSize: '14px',
			cursor: 'pointer',
			boxShadow: '0 2px 4px rgba(0,0,0,0.1)',
			zIndex: '9999'
		})
			.text('停止操作')
			.hover(
				() => jq(this).css({ background: '#ff7875' }),
				() => jq(this).css({ background: '#ff4d4f' })
			)
			.click(() => {
				stop = true
				c_btn.hide()
				c_notice.hide()
				utils.showNotice('已停止当前操作')
			})

		c_app.append(c_btn)
	}

	// 修改提示框样式
	const initNotice = () => {
		c_notice.css({
			display: 'none',
			position: 'fixed',
			top: '110px',
			right: '10px',
			width: '240px',
			padding: '16px',
			background: '#fff',
			borderRadius: '6px',
			border: '1px solid #e8e8e8',
			boxShadow: '0 4px 12px rgba(0,0,0,0.15)',
			zIndex: '9999',
			fontSize: '14px',
			lineHeight: '1.5'
		})

		c_app.append(c_notice)
	}

	// 初始化
	const init = () => {
		reset()
		initMenu()
		initBtn()
		initNotice()
	}

	init()
})()

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址