/* 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()
})()