[Bilibili] 关注管理器

快速排序和筛选你的关注列表,一键取关不再关注的UP等

目前为 2022-04-05 提交的版本。查看 最新版本

  1. // ==UserScript==
  2. // @name [Bilibili] 关注管理器
  3. // @namespace ckylin-bilibili-foman
  4. // @version 0.2.17
  5. // @description 快速排序和筛选你的关注列表,一键取关不再关注的UP等
  6. // @author CKylinMC
  7. // @supportURL https://github.com/CKylinMC/UserJS
  8. // @require https://gf.qytechs.cn/scripts/429720-cktools/code/CKTools.js?version=1034581
  9. // @include http://space.bilibili.com/*
  10. // @include https://space.bilibili.com/*
  11. // @connect api.bilibili.com
  12. // @grant GM_registerMenuCommand
  13. // @grant GM_getResourceText
  14. // @grant GM_setValue
  15. // @grant GM_getValue
  16. // @grant GM_removeValue
  17. // @grant unsafeWindow
  18. // @license GPL-3.0-only
  19. // @compatible chrome 80+
  20. // @compatible firefox 74+
  21. // ==/UserScript==
  22. (function () {
  23. 'use strict';
  24. const s = {
  25. get(key, def) {
  26. const val = GM_getValue('autoExtendInfo');
  27. if (typeof (val) == 'undefined' || val === null) return def;
  28. return val;
  29. },
  30. set(key, val) {
  31. GM_setValue('autoExtendInfo', val);
  32. },
  33. del(key) {
  34. if (typeof (GM_removeValue) == 'function') GM_removeValue(key);
  35. else GM_setValue(key, undefined);
  36. }
  37. };
  38. const datas = {
  39. status: 0,
  40. total: 0,
  41. fetched: 0,
  42. pages: 0,
  43. followings: [],
  44. mappings: {},
  45. dommappings: {},
  46. checked: [],
  47. tags: {},
  48. self: 0,
  49. isSelf: false,
  50. currUid: 0,
  51. fetchstat: "OK",
  52. currInfo: {
  53. black: -1,
  54. follower: -1,
  55. following: -1,
  56. mid: -1,
  57. whisper: -1,
  58. },
  59. preventUserCard: false,
  60. settings: {
  61. get autoExtendInfo() {
  62. return s.get('autoExtendInfo', true);
  63. },
  64. set autoExtendInfo(val) {
  65. s.set('autoExtendInfo', val);
  66. },
  67. get lazyRenderForList() {
  68. return s.get('lazyRenderForList', true);
  69. },
  70. set lazyRenderForList(val) {
  71. s.set('lazyRenderForList', val);
  72. },
  73. get batchOperationDelay() {
  74. return s.get('batchOperationDelay', .5);
  75. },
  76. set batchOperationDelay(val) {
  77. s.set('batchOperationDelay', val);
  78. },
  79. }
  80. };
  81. const cfg = {
  82. debug: false,
  83. retrial: 3,
  84. enableNewModules: false,
  85. VERSION: "0.2.17 Beta",
  86. infobarTemplate: ()=>`共读取 ${datas.fetched} 条关注`,
  87. titleTemplate: () => `<h1>关注管理器 FoMan <small>v${cfg.VERSION} ${cfg.debug ? "debug" : ""}</small></h1>`,
  88.  
  89. // Turn this on will abort all alerts.
  90. I_KNOW_WHAT_IM_DOING: false
  91. }
  92. const get = q => document.querySelector(q);
  93. const getAll = q => document.querySelectorAll(q);
  94. const wait = t => new Promise(r => setTimeout(r, t));
  95. const batchDelay = async () => await wait(datas.settings.batchOperationDelay*1000);
  96. const log = (...m) => cfg.debug && console.log('[FoMan]', ...m);
  97. const mdi = (name, asHTML=true, px = '10', extras = []) => {
  98. const i = CKTools.domHelper('i', {
  99. classnames: ['mdi',`mdi-${name}`, `mdi-${px}px`, ...extras],
  100. text:' '
  101. });
  102. return asHTML ? i.outerHTML : i;
  103. };
  104. const getSelfId = async () => {
  105. let stat = unsafeWindow.UserStatus;
  106. let retrial = 20;
  107. while (stat === null || stat === undefined) {
  108. if (--retrial < 0) return 0;
  109. log("Waiting for userstatus...")
  110. await wait(200);
  111. }
  112. if (!stat.userInfo.isLogin) {
  113. log("NOT LOGIN");
  114. return -1
  115. }
  116. log("User:", stat.userInfo.mid, stat.userInfo);
  117. return stat.userInfo.mid;
  118. };
  119. async function copy(txt = '') {
  120. try {
  121. await navigator.clipboard.writeText(txt);
  122. return true;
  123. } catch (e) {
  124. return false;
  125. }
  126. }
  127. function download(filename, text) {
  128. var element = document.createElement('a');
  129. element.setAttribute('href', 'data:text/plain;charset=utf-8,' + encodeURIComponent(text));
  130. element.setAttribute('download', filename);
  131. element.style.display = 'none';
  132. document.body.appendChild(element);
  133. element.click();
  134. document.body.removeChild(element);
  135. }
  136. const makeDom = async (domname, func = () => {
  137. }) => {
  138. if(CKTools.domHelper) return CKTools.domHelper(domname, func);
  139. const d = document.createElement(domname);
  140. if(typeof(func)=='function') func.constructor.name=='AsyncFunction' ? await func(d) : func(d);
  141. return d;
  142. };
  143. const isHardCoreMember = d => d.is_senior_member ===1;
  144. const isFans = d => d.attribute === 6;
  145. const isWhisper = d => d.attribute === 1;
  146. const isNearly = d => {
  147. const nearly = (new Date).getTime() - (60 * 60 * 24 * 7 * 4 * 3 * 1000);
  148. return parseInt(d + "000") > nearly;
  149. }
  150. const isLongAgo = (d) => {
  151. const loneAgo = (new Date).getTime() - (60 * 60 * 24 * 7 * 4 * 12 * 2 * 1000);
  152. return parseInt(d + "000") < loneAgo;
  153. }
  154. /* StackOverflow 10730362 */
  155. const getCookie = (name) => {
  156. const value = `; ${document.cookie}`;
  157. const parts = value.split(`; ${name}=`);
  158. if (parts.length === 2) return parts.pop().split(';').shift();
  159. }
  160. const getCSRFToken = () => getCookie("bili_jct");
  161. const getBgColor = () => {/*兼容blbl进化的夜间模式*/
  162. try {
  163. let color = getComputedStyle(document.body).backgroundColor;
  164. if (color === "rgba(0, 0, 0, 0)") return "white";
  165. else return color;
  166. } catch (e) {
  167. return "white"
  168. }
  169. }
  170.  
  171. const getCurrentUid = async () => {
  172. setInfoBar("正在查询当前用户UID");
  173. let paths = location.pathname.split('/');
  174. if (paths.length > 1) {
  175. return paths[1];
  176. } else throw "Failed to get current ID";
  177. };
  178. const getHeaders = () => {
  179. return {
  180. "user-agent": unsafeWindow.navigator.userAgent,
  181. "cookie": unsafeWindow.document.cookie,
  182. "origin": "space.bilibili.com",
  183. "referer": "https://www.bilibili.com/"
  184. }
  185. };
  186. const getUInfoURL = uid => `https://api.bilibili.com/x/space/acc/info?mid=${uid}`;
  187. const getGroupURL = () => `https://api.bilibili.com/x/relation/tags`;
  188. const getWhispersURL = (pn,ps=50) => `https://api.bilibili.com/x/relation/whispers?pn=${pn}&ps=${ps}&order=desc&order_type=attention`;
  189. const getFetchURL = (uid, pn) => `https://api.bilibili.com/x/relation/followings?vmid=${uid}&pn=${pn}&ps=50&order=desc&order_type=attention`;
  190. const getUnfolURL = () => `https://api.bilibili.com/x/relation/modify`;
  191. const getFollowURL = () => `https://api.bilibili.com/x/relation/batch/modify`;
  192. const getLatestVidURL = uid => `https://api.bilibili.com/x/space/arc/search?mid=${uid}&ps=1&pn=1`
  193. const getSubInfoURL = uid => `https://api.bilibili.com/x/relation/stat?vmid=${uid}`;
  194. const getCreateGroupURL = ()=> `https://api.bilibili.com/x/relation/tag/create`;
  195. const getRenameGroupURL = ()=> `https://api.bilibili.com/x/relation/tag/update`;
  196. const getRemoveGroupURL = ()=> `https://api.bilibili.com/x/relation/tag/del`;
  197. const getMoveToGroupURL = ()=> `https://api.bilibili.com/x/relation/tags/addUsers`;
  198. const getCopyToGroupURL = ()=> `https://api.bilibili.com/x/relation/tags/copyUsers`;
  199. const getDynamicURL = (selfid,hostid)=>`https://api.vc.bilibili.com/dynamic_svr/v1/dynamic_svr/space_history?visitor_uid=${selfid}&host_uid=${hostid}&offset_dynamic_id=0&need_top=1&platform=web`;
  200. const getRequest = path => new Request(path, {
  201. method: 'GET',
  202. headers: getHeaders(),
  203. credentials: "include"
  204. });
  205. const getPostRequest = (path, body = null) => new Request(path, {
  206. method: 'POST',
  207. headers: getHeaders(),
  208. credentials: "include",
  209. body
  210. });
  211. const cacheGroupList = async () => {
  212. setInfoBar("正在获取分组信息...");
  213. try {
  214. const jsonData = await (await fetch(getRequest(getGroupURL()))).json();
  215. if (jsonData && jsonData.code === 0) {
  216. datas.tags = [];
  217. for (let tag of jsonData.data) {
  218. datas.tags[tag.tagid] = tag;
  219. }
  220. return true;
  221. } else {
  222. log(jsonData);
  223. return false;
  224. }
  225. } catch (err) {
  226. log(err);
  227. return false;
  228. }
  229. };
  230. const createGroup = async (tagname) => {
  231. setInfoBar(`正在创建新的分组"${tagname}"...`);
  232. try {
  233. const jsonData = await (await fetch(
  234. getPostRequest(getCreateGroupURL(),
  235. new URLSearchParams({
  236. tag: tagname,
  237. csrf: getCSRFToken()
  238. }))));
  239. if (jsonData.code === 0) return true;
  240. else throw new Error(jsonData.message);
  241. }catch(err){
  242. log(err);
  243. return false;
  244. } finally {
  245. await cacheGroupList();
  246. CacheManager.save();
  247. }
  248. }
  249. const renameGroup = async (tagid, tagname) => {
  250. setInfoBar(`正在修改分组为"${tagname}"...`);
  251. try {
  252. const jsonData = await (await fetch(
  253. getPostRequest(getRenameGroupURL(),
  254. new URLSearchParams({
  255. tagid,
  256. name: tagname,
  257. csrf: getCSRFToken()
  258. }))));
  259. if (jsonData.code === 0) return true;
  260. else throw new Error(jsonData.message);
  261. }catch(err){
  262. log(err);
  263. return false;
  264. } finally {
  265. await cacheGroupList();
  266. CacheManager.save();
  267. await renderListTo(get("#CKFOMAN-MAINLIST"),datas.followings,true);
  268. resetInfoBar();
  269. }
  270. }
  271. const removeGroup = async (tagid) => {
  272. setInfoBar(`正在移除分组"${tagid}"...`);
  273. try {
  274. const jsonData = await (await fetch(
  275. getPostRequest(getRemoveGroupURL(),
  276. new URLSearchParams({
  277. tagid,
  278. csrf: getCSRFToken()
  279. }))));
  280. if (jsonData.code === 0) return true;
  281. else throw new Error(jsonData.message);
  282. }catch(err){
  283. log(err);
  284. return false;
  285. } finally {
  286. await cacheGroupList();
  287. CacheManager.save();
  288. await renderListTo(get("#CKFOMAN-MAINLIST"),datas.followings,true);
  289. resetInfoBar();
  290. }
  291. }
  292. const moveUserToDefaultGroup = uids => moveUserToGroup(uids, [0]);//unused
  293. const moveUserToGroup = async (uids, tagids) => {
  294. setInfoBar(`正在移动用户分组...`);
  295. try {
  296. const jsonData = await (await fetch(
  297. getPostRequest(getMoveToGroupURL(),
  298. new URLSearchParams({
  299. fids: uids.join(','),
  300. tagids: tagids.join(','),
  301. csrf: getCSRFToken()
  302. }))).json());
  303. if (jsonData.code === 0) {
  304. for (let uid of uids) {
  305. const u = parseInt(uid);
  306. let targetUser;
  307. if (datas.mappings.includes(u)) {
  308. targetUser = datas.mappings[u];
  309. } else if (datas.mappings.includes(uid)) {
  310. targetUser = datas.mappings[uid];
  311. } else {
  312. //TODO: need reload
  313. }
  314. targetUser.tag = tagids.map(i=>parseInt(i))
  315. }
  316. return true;
  317. }
  318. else throw new Error(jsonData.message);
  319. }catch(err){
  320. log(err);
  321. return false;
  322. }
  323. }
  324. const copyUserToGroup = async (uids, tagids) => {
  325. setInfoBar(`正在添加用户分组...`);
  326. try {
  327. const jsonData = await (await fetch(
  328. getPostRequest(getCopyToGroupURL(),
  329. new URLSearchParams({
  330. fids: uids.join(','),
  331. tagids: tagids.join(','),
  332. csrf: getCSRFToken()
  333. }))).json());
  334. log(jsonData,jsonData.code,jsonData.code===0);//TODO:BUG
  335. if (jsonData.code == 0) {
  336. for (let uid of uids) {
  337. const u = parseInt(uid);
  338. let targetUser;
  339. if (datas.mappings.includes(u)) {
  340. targetUser = datas.mappings[u];
  341. } else if (datas.mappings.includes(uid)) {
  342. targetUser = datas.mappings[uid];
  343. } else {
  344. //TODO: need reload
  345. }
  346. targetUser.tag = (function () {
  347. const tag = [];
  348. for (const tid of [...targetUser.tag, ...tagids]) {
  349. const ntid = parseInt(tid);
  350. if(!tag.includes(ntid)) tag.push(ntid)
  351. }
  352. return tag;
  353. })()
  354. }
  355. return true;
  356. }
  357. else throw new Error(jsonData.message);
  358. }catch(err){
  359. log(err);
  360. return false;
  361. }
  362. }
  363. const getCurrSubStat = async uid => {
  364. try {
  365. const jsonData = await (await fetch(getRequest(getSubInfoURL(uid)))).json();
  366. if (jsonData && jsonData.code === 0) {
  367. return jsonData.data;
  368. } else {
  369. log("Failed fetch self info: unexpected response", jsonData);
  370. return null;
  371. }
  372. } catch (e) {
  373. log("Failed fetch self info: error found", e);
  374. return null;
  375. }
  376. }
  377. const getLatestVideoPublishDate = async uid => {
  378. try {
  379. const jsonData = await (await fetch(getRequest(getLatestVidURL(uid)))).json();
  380. if (jsonData && jsonData.code === 0) {
  381. if (
  382. jsonData.data
  383. && jsonData.data.list
  384. ) {
  385. let mostCates = "";
  386. if (jsonData.data.list.tlist.length !== 0) {
  387. let max = 0, name="";
  388. for(let itemname of Object.keys(jsonData.data.list.tlist)){
  389. const item = jsonData.data.list.tlist[itemname];
  390. if(item.count>max){
  391. max = item.count;
  392. name= item.name;
  393. }else if(item.count===max){
  394. name+= "、"+item.name;
  395. }
  396. }
  397. mostCates = name;
  398. }
  399. if (jsonData.data.list.vlist.length === 0) {
  400. return {ok: false,mostCates:mostCates}
  401. }
  402. const vid = jsonData.data.list.vlist[0];
  403. return {ok: true, value: vid.created,vinfo: {aid:vid.aid,title:vid.title,pic:vid.pic,play:vid.play},mostCates:mostCates}
  404. } else {
  405. return {ok: false}
  406. }
  407. } else {
  408. return {ok: false}
  409. }
  410. } catch (e) {
  411. log(uid,e)
  412. return {ok: false}
  413. }
  414. };
  415. const getTypeNameFromDynamicTypeID = (id,fallback='?') => {
  416. switch (+id) {
  417. case 1:
  418. return mdi('share')+"转发动态";
  419. case 2:
  420. return mdi('image-multiple')+"相册图片";
  421. case 4:
  422. return mdi('text');//文字动态
  423. case 8:
  424. return mdi('youtube')+"视频投稿";
  425. case 16:
  426. return mdi('video-box')+"小视频";
  427. case 64:
  428. return mdi('newspaper-variant-outline')+"专栏文章";
  429. case 128:
  430. return fallback;
  431. case 256:
  432. return mdi('playlist-music')+"音频投稿";
  433. case 512:
  434. return mdi('filmstrip-box-multiple')+"番剧更新";
  435. case 1024:
  436. return fallback;
  437. case 2048:
  438. return mdi('playlist-play')+"歌单分享";
  439. case 4300:
  440. return mdi('playlist-star')+"收藏夹";
  441. default: return fallback;
  442. }
  443. }
  444. const getContentFromDynamic = (card) => {
  445. if (!card) return '无法解析内容(空内容)';
  446. if (card.item?.content) return card.item.content;
  447. if (card.aid) return 'av'+card.aid+' | <b>'+card.title + "</b><br>简介: " + card.desc;
  448. if (card.item?.pictures) return card.item.pictures_count + '张图片';
  449. if (card.origin) return `转发自${card?.user?.uname}: ${card.item?.content}`;
  450. if (card.item?.description) return card.item.description;
  451. if(card.summary) return `cv${card.id} | <b>${card.title}</b><br>简介: ${card.summary}`
  452. return '无法解析内容(未知特征)';
  453. }
  454. const parseDynamic = (d)=>{
  455. const dynamic = {
  456. id: d.desc.dynamic_id_str,
  457. sender:d.desc.user_profile,
  458. like: d.desc.like,
  459. comment: d.desc.comment,
  460. repost: d.desc.repost,
  461. status: d.desc.status,
  462. timestamp: d.desc.timestamp,
  463. type: d.desc.type,
  464. content: getContentFromDynamic(d.card),
  465. origin: (d.desc.orig_dy_id && d.desc.orig_dy_id !== 0) ? (
  466. d.card.origin = JSON.parse(d.card.origin),
  467. getContentFromDynamic(d.card.origin)
  468. ):null,
  469. istop: d.extra.is_space_top===1,
  470. isrepost: d.desc.orig_dy_id&&d.desc.orig_dy_id!==0,
  471. publisher: d.desc.orig_dy_id ? (d.desc.orig_dy_id === 0 ? d.card.user : d.card.origin_user.info) : d.card.user,
  472. prefix: getTypeNameFromDynamicTypeID(d.desc.type),
  473. origprefix: getTypeNameFromDynamicTypeID(d.desc.orig_type)
  474. };
  475. return dynamic;
  476. }
  477. const getDynamic = async uid => {
  478. try {
  479. const jsonData = await (await fetch(getRequest(getDynamicURL(datas.self,uid)))).json();
  480. if (jsonData && jsonData.code === 0) {
  481. const data = jsonData.data.cards;
  482. const dynamics = {
  483. top:null,
  484. next:null,
  485. }
  486. if(!data || data.length === 0) {
  487. return dynamics;
  488. }
  489. let d = data.shift();
  490. d.card = JSON.parse(d.card);
  491. let obj = parseDynamic(d);
  492. if(obj.istop){
  493. dynamics.top = obj;
  494. let nd = data.shift();
  495. nd.card = JSON.parse(nd.card);
  496. let nobj = parseDynamic(nd);
  497. dynamics.next = nobj;
  498. }else{
  499. dynamics.next = obj;
  500. }
  501. return dynamics;
  502. } else {
  503. log("Failed fetch self info: unexpected response", jsonData);
  504. return null;
  505. }
  506. } catch (e) {
  507. log("Failed fetch self info: error found", e);
  508. return null;
  509. }
  510. }
  511. const getUserStats = async (uid, withraw=false) => {
  512. try {
  513. const jsonData = await (await fetch(getRequest(getUInfoURL(uid)))).json();
  514. if (jsonData && jsonData.code === 0) {
  515. const udata = jsonData.data;
  516. const parsedData = {
  517. ok: true,
  518. level: udata.level,
  519. banned: udata.silence === 1,
  520. RIP: udata.sys_notice === 20,
  521. disputed: udata.sys_notice === 8,
  522. notice: udata.sys_notice,
  523. sign: udata.sign,
  524. cates: udata.tags,
  525. lives: udata.live_room,
  526. official_verify: udata.official_verify??udata.official,
  527. };
  528. if(withraw){
  529. return Object.assign({},udata,parsedData);
  530. }
  531. return parsedData
  532. }
  533. } catch (e) {
  534.  
  535. }
  536. return {ok: false}
  537. }
  538. const fillUserStatus = async (uid, refresh=false) => {
  539. setInfoBar(`正在为${uid}填充用户信息`)
  540. uid = parseInt(uid);
  541. if(datas.mappings[uid]&&datas.mappings[uid].filled){
  542. log(uid,"already filled")
  543. resetInfoBar();
  544. return datas.mappings[uid];
  545. }
  546. const userinfo = await getUserStats(uid,refresh);
  547. if (userinfo.ok) {
  548. if(refresh) datas.mappings[uid] = userinfo;
  549. datas.mappings[uid].level = userinfo.level;
  550. datas.mappings[uid].banned = userinfo.banned;
  551. datas.mappings[uid].RIP = userinfo.RIP;
  552. datas.mappings[uid].disputed = userinfo.disputed;
  553. datas.mappings[uid].notice = userinfo.notice;
  554. datas.mappings[uid].sign = userinfo.sign;
  555. datas.mappings[uid].cates = userinfo.cates;
  556. datas.mappings[uid].lives = userinfo.lives;
  557. datas.mappings[uid].filled = true;
  558. if (!userinfo.banned && !userinfo.RIP) {
  559. const lastUpdate = await getLatestVideoPublishDate(uid);
  560. log(uid,lastUpdate)
  561. if (lastUpdate.ok) {
  562. datas.mappings[uid].lastUpdate = lastUpdate.value;
  563. datas.mappings[uid].lastUpdateInfo = lastUpdate.vinfo;
  564. }
  565. if(lastUpdate.mostCates) datas.mappings[uid].mostCates = lastUpdate.mostCates;
  566. }
  567. log(uid, datas.mappings[uid]);
  568. } else {
  569. log(uid, "fetch space info failed");
  570. }
  571. resetInfoBar();
  572. return datas.mappings[uid];
  573. }
  574. const RELE_ACTION = {
  575. FOLLOW:1,
  576. UNFOLLOW:2,
  577. WHISPER:3,
  578. UNWHISPER:4,
  579. BLOCK:5,
  580. UNBLOCK:6,
  581. KICKFANS:7
  582. }
  583. const batchOperateUser = async (uids = [], actCode) => {
  584. if (uids.length === 0) return {ok: false, res: "UIDS is empty"};
  585. if(!Object.values(RELE_ACTION).includes(actCode)){
  586. if(Object.keys(RELE_ACTION).includes(actCode)){
  587. actCode = RELE_ACTION[actCode];
  588. }else{
  589. return {ok: false, res: "Unknown action code"};
  590. }
  591. }
  592. const act = actCode;
  593. log("Batch Operating with Action Code",act);
  594. const operate = async(_uids,_act)=>{
  595. try {
  596. const jsonData = await (await fetch(getPostRequest(getFollowURL(), new URLSearchParams(`fids=${_uids.join(',')}&act=${_act}&re_src=11&jsonp=jsonp&csrf=${getCSRFToken()}`)))).json()
  597. if (jsonData && jsonData.code === 0) return {ok: true, uids, res: ""};
  598. return {ok: false, uids, res: jsonData.message, data: jsonData.data};
  599. } catch (e) {
  600. return {ok: false, uids, res: e.message};
  601. }
  602. }
  603. const list = [...uids];
  604. const results = {ok:true,uids,res:"",data:{failed_fids:[],failed_results:[]}};//failed_fids
  605. if(list.length>50) log("WARNING: Operating with more than 50 items, it may cause some issues.");
  606. while(list.length){
  607. const currents = list.splice(0,50);
  608. const result = await operate(currents,act);
  609. if(!result.ok){
  610. results.ok = false;
  611. results.res="部分请求出现错误";
  612. results.data.failed_fids.concat(result.data.failed_fids);
  613. results.data.failed_results.push(result);
  614. }
  615. }
  616. log("Results:",results);
  617. return results;
  618. }
  619. const convertToWhisper = async (uids)=>{
  620. log("Unfollowing",uids);
  621. let unfo = uids.length===1?await operateUser(uids[0],RELE_ACTION.UNFOLLOW):await batchOperateUser(uids,RELE_ACTION.UNFOLLOW);
  622. log("Unfollowed:",unfo);
  623. if(!unfo.ok) return unfo;
  624. log("Whispering",uids);
  625. let whis = uids.length===1?await operateUser(uids[0],RELE_ACTION.WHISPER):await batchOperateUser(uids,RELE_ACTION.WHISPER);
  626. log("Whispered:",whis);
  627. return whis;
  628. }
  629. const convertToFollow = async (uids)=>{
  630. log("Unwhispering",uids);
  631. let unwh = uids.length===1?await operateUser(uids[0],RELE_ACTION.UNWHISPER):await batchOperateUser(uids,RELE_ACTION.UNWHISPER);
  632. log("Unwhispered:",unwh);
  633. if(!unwh.ok) return unwh;
  634. log("Following",uids);
  635. let foll = uids.length===1?await operateUser(uids[0],RELE_ACTION.FOLLOW):await batchOperateUser(uids,RELE_ACTION.FOLLOW);
  636. log("Followed:",foll);
  637. return foll;
  638. }
  639. // CSDN https://blog.csdn.net/namechenfl/article/details/91968396
  640. function numberFormat(value) {
  641. let param = {};
  642. let k = 10000,
  643. sizes = ['', '万', '亿', '万亿'],
  644. i;
  645. if (value < k) {
  646. param.value = value
  647. param.unit = ''
  648. } else {
  649. i = Math.floor(Math.log(value) / Math.log(k));
  650. param.value = ((value / Math.pow(k, i))).toFixed(2);
  651. param.unit = sizes[i];
  652. }
  653. return param;
  654. }
  655. const operateUser = async (uid, actCode) => {
  656. if(!Object.values(RELE_ACTION).includes(actCode)){
  657. if(Object.keys(RELE_ACTION).includes(actCode)){
  658. actCode = RELE_ACTION[actCode];
  659. }else{
  660. return {ok: false, res: "Unknown action code"};
  661. }
  662. }
  663. const act = actCode;
  664. log("Operating with Action Code",act);
  665. try {
  666. const jsonData = await (await fetch(getPostRequest(getUnfolURL(), new URLSearchParams(`fid=${uid}&act=${act}&re_src=11&jsonp=jsonp&csrf=${getCSRFToken()}`)))).json()
  667. if (jsonData && jsonData.code === 0) return {ok: true, uid, res: ""};
  668. return {ok: false, uid, res: jsonData.message};
  669. } catch (e) {
  670. return {ok: false, uid, res: e.message};
  671. }
  672. }
  673. const unfollowUser = async (uid,iswhisper=false) => {
  674. try {
  675. if(datas.isSelf){
  676. iswhisper = datas.mappings[uid].attribute===1 || datas.mappings[uid].isWhisper;
  677. }
  678. return operateUser(uid,iswhisper?RELE_ACTION.UNWHISPER:RELE_ACTION.UNFOLLOW);
  679. } catch (e) {
  680. return {ok: false, uid, res: e.message};
  681. }
  682. }
  683. const unfollowUsers = async uids => {
  684. let okgroup = [];
  685. let errgroup = [];
  686. for (let uid of uids) {
  687. setInfoBar(`正在取关 ${uid} ...`)
  688. let result = await unfollowUser(uid);
  689. log(result);
  690. if (result.ok) {
  691. okgroup.push(uid);
  692. } else {
  693. errgroup.push(uid);
  694. }
  695. await batchDelay();
  696. }
  697. setInfoBar(`取关完成`)
  698. return {
  699. ok: errgroup.length === 0,
  700. okgroup, errgroup
  701. }
  702. }
  703. const fetchFollowings = async (uid, page = 1) => {
  704. let retry = cfg.retrial;
  705. while (retry-- > 0) {
  706. try {
  707. const jsonData = await (await fetch(getRequest(getFetchURL(uid, page)))).json();
  708. if (jsonData) {
  709. if (jsonData.code === 0) return jsonData;
  710. if (jsonData.code === 22007) {
  711. retry = -1;
  712. datas.fetchstat = "GUEST-LIMIT";
  713. throw "Not the owner of uid " + uid;
  714. }
  715. if(jsonData.code === 22115) {
  716. retry = -1;
  717. datas.fetchstat = "PERMS-DENIED";
  718. throw "Permission denied.";
  719. }
  720. }
  721. log("Unexcept fetch result", "retry:", retry, "uid:", uid, "p:", page, "data", jsonData)
  722. } catch (e) {
  723. if(datas.fetchstat==="OK")datas.fetchstat = "ERRORED";
  724. log("Errored while fetching followings", "retry:", retry, "uid:", uid, "p:", page, "e:", e);
  725. }
  726. }
  727. return null;
  728. }
  729. const fetchWhisperFollowings = async (uid, page = 1) => {
  730. if(!datas.isSelf) return null;
  731. let retry = cfg.retrial;
  732. while (retry-- > 0) {
  733. try {
  734. const jsonData = await (await fetch(getRequest(getWhispersURL(page)))).json();
  735. if (jsonData) {
  736. if (jsonData.code === 0) {
  737. for(let item of jsonData.data.list){
  738. item.isWhisper = true;
  739. }
  740. return jsonData;
  741. }
  742. if (jsonData.code === 22007) {
  743. retry = -1;
  744. datas.fetchstat = "GUEST-LIMIT";
  745. throw "Not the owner of uid " + uid;
  746. }
  747. if(jsonData.code === 22115) {
  748. retry = -1;
  749. datas.fetchstat = "PERMS-DENIED";
  750. throw "Permission denied.";
  751. }
  752. }
  753. log("Unexcept fetch result", "retry:", retry, "uid:", uid, "p:", page, "data", jsonData)
  754. } catch (e) {
  755. if(datas.fetchstat==="OK")datas.fetchstat = "ERRORED";
  756. log("Errored while fetching followings", "retry:", retry, "uid:", uid, "p:", page, "e:", e);
  757. }
  758. }
  759. return null;
  760. }
  761. const getFollowings = async (force = false) => {
  762. if (datas.status === 1) {
  763. log("Task canceled due to busy");
  764. return;
  765. }
  766. log("Fetching followings with param force =",force?"true":"false");
  767. cfg.infobarTemplate = ()=>`共读取 ${datas.fetched} 条关注`;
  768. datas.status = 1;
  769. datas.checked = [];
  770. let currentPageNum = 1;
  771. const uid = await getCurrentUid();
  772. const self = await getSelfId();
  773. datas.currUid = uid;
  774. datas.self = self;
  775. if (self === -1) {
  776. if(!cfg.I_KNOW_WHAT_IM_DOING)alertModal("没有登录(不可用)", "你没有登录(不可用),部分功能可能无法正常工作。", "确定");
  777. log("Not login");
  778. } else if (self === 0) {
  779. if(!cfg.I_KNOW_WHAT_IM_DOING)alertModal("获取当前用户信息失败", "无法得知当前页面是否为你的个人空间,因此部分功能可能无法正常工作。", "确定");
  780. log("Failed fetch current user");
  781. } else if (self + "" !== uid) {
  782. if(!cfg.I_KNOW_WHAT_IM_DOING)alertModal("他人的关注列表", "这不是你的个人空间,因此获取的关注列表也不是你的列表。<br>非本人关注列表最多显示前250个关注。<br>你仍然可以对其进行筛选,但是不能进行操作。", "确定");
  783. log("Other's space.");
  784. } else if (self + "" === uid) {
  785. datas.isSelf = true;
  786. }
  787. unsafeWindow.FoMan_CurrentUser = ()=>createUserInfoCardFromOthers(datas.currUid);
  788. cfg.titleTemplate = ()=>`<h1>关注管理器 <small>v${cfg.VERSION} ${cfg.debug?"debug":""} <span style="color:grey;font-size:x-small;margin-right:12px;float:right">当前展示: UID:${datas.currUid} ${datas.isSelf?"(你)":`(${document.title.replace("的个人空间_哔哩哔哩_bilibili","").replace("的个人空间_哔哩哔哩_Bilibili","")})`} <a href='javascript:void(0)' onclick='FoMan_CurrentUser()'>👁️‍🗨️</a></span></small></h1>`
  789. setTitle();
  790. let needreload = force || !CacheManager.load();
  791. const currInfo = await getCurrSubStat(uid);
  792. if (datas.currInfo.following !== -1 && currInfo !== null) {
  793. if (force === false && datas.currInfo.following === currInfo.following && datas.currInfo.whisper === currInfo.whisper) {
  794. if (datas.fetched > 0)
  795. needreload = false;
  796. } else if(!needreload && (datas.currInfo.following !== currInfo.following || datas.currInfo.whisper !== currInfo.whisper)){
  797. alertModal("自动重新加载","检测到数据变化,已经自动重新加载。","确定");
  798. needreload = true;
  799. }
  800. }
  801. datas.currInfo = currInfo;
  802. if (needreload) {
  803. datas.followings = [];
  804. datas.mappings = {};
  805. datas.fetched = 0;
  806. const firstPageData = await fetchFollowings(uid, currentPageNum);
  807. if (!firstPageData) throw "Failed to fetch followings";
  808. datas.total = firstPageData.data.total;
  809. datas.pages = Math.floor(datas.total / 50) + (datas.total % 50 ? 1 : 0);
  810. datas.followings = datas.followings.concat(firstPageData.data.list);
  811. datas.fetched += firstPageData.data.list.length;
  812. firstPageData.data.list.forEach(it => {
  813. datas.mappings[parseInt(it.mid)] = it;
  814. })
  815. currentPageNum += 1;
  816. for (; currentPageNum <= datas.pages; currentPageNum++) {
  817. const currentData = await fetchFollowings(uid, currentPageNum);
  818. if (!currentData) break;
  819. datas.followings = datas.followings.concat(currentData.data.list);
  820. datas.fetched += currentData.data.list.length;
  821. currentData.data.list.forEach(it => {
  822. datas.mappings[parseInt(it.mid)] = it;
  823. });
  824. setInfoBar(`正在查询关注数据:已获取 ${datas.fetched} 条数据`);
  825. }
  826. log("isSelf? ",datas.isSelf);
  827. if(datas.isSelf){
  828. setInfoBar(`正在查询悄悄关注数据`);
  829. let whisperPageNum =1;
  830. let fetched = 0;
  831. const whisperPages = Math.floor(datas.currInfo.whisper / 50) + (datas.currInfo.whisper % 50 ? 1 : 0);
  832. for(; whisperPageNum<=whisperPages;whisperPageNum++){
  833. const currentData = await fetchWhisperFollowings(whisperPageNum);
  834. log(currentData);
  835. if (!currentData) break;
  836. datas.followings = datas.followings.concat(currentData.data.list);
  837. fetched += currentData.data.list.length;
  838. currentData.data.list.forEach(it => {
  839. datas.mappings[parseInt(it.mid)] = it;
  840. });
  841. setInfoBar(`正在查询悄悄关注数据:已获取 ${fetched} 条数据`);
  842. }
  843. }
  844. CacheManager.save();
  845. }else{
  846. log("Using last result.");
  847. cfg.infobarTemplate = ()=>`共读取 ${datas.fetched} 条关注(缓存,<a href="javascript:void(0)" onclick="openFollowManager(true)">点此重新加载</a>)`
  848. setInfoBar("使用上次数据");
  849. }
  850. datas.status = 2;
  851. log("fetch completed.");
  852. autoCacheCleaner();
  853. }
  854. const autoCacheCleaner = (force=false) => {
  855. let size = CacheManager.getSize();
  856. if (force || size >= 2) {
  857. setInfoBar("正在整理缓存空间...");
  858. alertModal('请稍等', '由于缓存空间到达警戒值,正在自动整理缓存,请稍等...');
  859. CacheManager.prune();
  860. let aftersize = CacheManager.getSize();
  861. if (aftersize >= 2) {
  862. alertModal('请稍等', '缓存空间仍然处于警戒值以上,整理缓存无效,正在自动清理缓存,请稍等...');
  863. CacheManager.clean();
  864. }
  865. aftersize = CacheManager.getSize();
  866. alertModal('清理完成', '本次自动清理释放了' + (size - aftersize) + ' MB缓存空间。', "确定");
  867. resetInfoBar();
  868. }
  869. }
  870. unsafeWindow.FoManCleaner = (force=false)=>autoCacheCleaner(force);
  871. const CacheProvider = {
  872. storage: window.localStorage,
  873. prefix: "Unfollow_",
  874. expire: 1000*60*60*2,
  875. getKey:(key)=>CacheProvider.prefix+key,
  876. valueWrapper: (value='',no=false)=>{
  877. log(JSON.stringify({
  878. et: no?(new Date('2999/1/1')).getTime():(new Date()).getTime()+CacheProvider.expire,
  879. vl: value
  880. }));
  881. return JSON.stringify({
  882. et: no?(new Date('2999/1/1')).getTime():(new Date()).getTime()+CacheProvider.expire,
  883. vl: value
  884. });
  885. },
  886. getValue: (value="{}",key=null,noprefix=false)=>{
  887. try{
  888. const itemArc = JSON.parse(value);
  889. if(itemArc.hasOwnProperty('et')&&itemArc.et>=(new Date()).getTime()){
  890. return itemArc.vl;
  891. }
  892. if(key)CacheProvider.del(key,noprefix);
  893. return null;
  894. }catch(e){
  895. if(key)CacheProvider.del(key,noprefix);
  896. return null;
  897. }
  898. },
  899. list: ()=>Object.keys(CacheProvider.storage).filter(el=>el.startsWith(CacheProvider.prefix)),
  900. has: (key,noprefix=false)=>{
  901. if(!noprefix){
  902. key = CacheProvider.getKey(key);
  903. }
  904. return CacheProvider.storage.getItem(key)===null;
  905. },
  906. valid: (key,noprefix=false)=>{
  907. if(!noprefix){
  908. key = CacheProvider.getKey(key);
  909. }
  910. if(CacheProvider.has(key,true)){
  911. const value = CacheProvider.storage.getItem(key);
  912. return CacheProvider.getValue(value,key,true)!==null;
  913. }else return false;
  914. },
  915. set: (key,val,noexpire=false,noprefix = false)=>{
  916. if(!noprefix){
  917. key = CacheProvider.getKey(key);
  918. }
  919. CacheProvider.storage.setItem(key,CacheProvider.valueWrapper(val,noexpire));
  920. },
  921. get: (key,fallback=null,noprefix=false)=>{
  922. if(!noprefix){
  923. key = CacheProvider.getKey(key);
  924. }
  925. const result = CacheProvider.storage.getItem(key);
  926. log('Cache-get-with-key',key,result);
  927. if(result===null) return fallback;
  928. log('Cache-get-parsed-value',key,CacheProvider.getValue(result,key,true));
  929. return CacheProvider.getValue(result,key,true);
  930. },
  931. del: (key,noprefix=false)=>{
  932. if(!noprefix){
  933. key = CacheProvider.getKey(key);
  934. }
  935. delete CacheProvider.storage[key];
  936. },
  937. prune: ()=>{
  938. const count = {
  939. valid:0,expired:0
  940. };
  941. CacheProvider.list().forEach(it=>{
  942. if(!it) return;
  943. if(CacheProvider.valid(it,true)){
  944. count.valid++;
  945. }else{
  946. count.expired++;
  947. }
  948. })
  949. return;
  950. },
  951. getSize: (filter = (key) => key.startsWith(CacheProvider.prefix)) => {
  952. const sum = (...args)=>args.reduce((a,b)=>a+b,0);
  953. return sum(...Object.keys(CacheProvider.storage).filter(filter).map(it=>CacheProvider.storage.getItem(it).length));
  954. }
  955. }
  956. const CacheManager = {
  957. version: 1,
  958. save:(uid=datas.currUid)=>{
  959. const {total,fetched,pages,followings,tags,currInfo} = datas;
  960. const tagclone = {};
  961. for(let tn of Object.keys(tags)){
  962. tagclone[tn+''] = tags[tn];
  963. }
  964. /*log({
  965. total,fetched,pages,followings,mappings,tagclone,currInfo
  966. });*/
  967. CacheProvider.set(`cache_${uid}`,{
  968. total,fetched,pages,followings,tagclone,currInfo,cacheVersion:CacheManager.version
  969. });
  970. },
  971. load:(uid=datas.currUid)=>{
  972. if(!datas.isSelf) return false;
  973. const cached = CacheProvider.get(`cache_${uid}`);
  974. if(cached===null) return false;
  975. else{
  976. const { total, fetched, pages, followings, tagclone, currInfo } = cached;
  977. if (!cached.cacheVersion || cached.cacheVersion < CacheManager.version) {
  978. CacheProvider.del(`cache_${uid}`);
  979. return false;
  980. }
  981. const tags = {};
  982. for(let tn of Object.keys(tagclone)){
  983. tags[parseInt(tn)] = tagclone[tn];
  984. }
  985. const mappings = {};
  986. for (const follow of followings) {
  987. mappings[+follow.mid] = follow;
  988. }
  989. const cdata = {total,fetched,pages,followings,mappings,tags,currInfo};
  990. for(let n of Object.keys(cdata)){
  991. datas[n] = cdata[n];
  992. }
  993. return true;
  994. }
  995. },
  996. prune: () => {
  997. CacheProvider.prune();
  998. try{
  999. CacheProvider.list().forEach(el => {
  1000. const value = CacheProvider.get(el, null, true);
  1001. if (!value.cacheVersion||value.cacheVersion < CacheManager.version) CacheProvider.del(el, true);
  1002. });
  1003. return true;
  1004. }catch(e){
  1005. log(e);
  1006. return false;
  1007. }
  1008. },
  1009. clean:()=>{
  1010. try{
  1011. CacheProvider.list().forEach(el=>CacheProvider.del(el,true));
  1012. return true;
  1013. }catch(e){
  1014. log(e);
  1015. return false;
  1016. }
  1017. },
  1018. getSize: () => {
  1019. return (CacheProvider.getSize()/1024/1024).toFixed(2);
  1020. }
  1021. }
  1022. const clearStyles = (className = "CKFOMAN") => {
  1023. let dom = document.querySelectorAll("style." + className);
  1024. if (dom) [...dom].forEach(e => e.remove());
  1025. }
  1026. const addStyle = (s, className = "CKFOMAN", mode = "append") => {
  1027. switch (mode) {
  1028. default:
  1029. case "append":
  1030. break;
  1031. case "unique":
  1032. if (document.querySelector("style." + className)) return;
  1033. break;
  1034. case "update":
  1035. clearStyles(className);
  1036. break;
  1037. }
  1038. let style = document.createElement("style");
  1039. style.classList.add(className);
  1040. style.innerHTML = s;
  1041. document.body.appendChild(style);
  1042. }
  1043. const setTitle = (val = null)=>{
  1044. const title = get("#CKFOMAN-titledom");
  1045. if(val!=null) title.innerHTML = val;
  1046. else title.innerHTML = cfg.titleTemplate();
  1047. }
  1048. const getFloatWindow = () => {
  1049. addMdiBtnStyle();
  1050. addStyle(`
  1051. #CKFOMAN{
  1052. position: fixed;
  1053. z-index: 99000;
  1054. top: 50%;
  1055. left: 50%;
  1056. width: 800px;
  1057. width: 60vw;
  1058. height: 800px;
  1059. height: 80vh;
  1060. background: white;
  1061. border-radius: 8px;
  1062. padding: 12px;
  1063. transform: translate(-50%,-50%);
  1064. transition: all .3s;
  1065. box-shadow: 0 2px 8px grey;
  1066. }
  1067. #CKFOMAN.hide{
  1068. opacity: 0;
  1069. pointer-events: none;
  1070. transform: translate(-50%,-50%) scale(0.95);
  1071. }
  1072. #CKFOMAN.show{
  1073. transform: translate(-50%,-50%) scale(1);
  1074. }
  1075. #CKFOMAN-container{
  1076. width: 100%;
  1077. /*overflow-y: auto;
  1078. overflow-x: hidden;
  1079. max-height: calc(80vh - 70px);*/
  1080. position: relative;
  1081. display: flex;
  1082. flex-direction: column;
  1083. flex-wrap: nowrap;
  1084. justify-content: flex-start;
  1085. align-items: stretch;
  1086. }
  1087. .CKFOMAN-scroll-list{
  1088. margin: 6px auto;
  1089. overflow-y: auto;
  1090. overflow-x: hidden;
  1091. display: flex;
  1092. flex-wrap: nowrap;
  1093. flex-direction: column;
  1094. max-height: calc(80vh - 80px);
  1095. }
  1096. .CKFOMAN-data-inforow{
  1097. border-radius: 6px;
  1098. flex: 1;
  1099. width: 100%;
  1100. display: flex;
  1101. padding: 6px;
  1102. color: #aaa;
  1103. transition: background .3s;
  1104. }
  1105. .CKFOMAN-data-inforow:hover{
  1106. background: #2196f361;
  1107. transition: background .1s;
  1108. }
  1109. .CKFOMAN-data-inforow-toggle{
  1110. margin: 3px 8px;
  1111. }
  1112. .CKFOMAN-toolbar-btns{
  1113. flex: 1;
  1114. border: none;
  1115. background: #2196f3;
  1116. border-radius: 3px;
  1117. margin: 0 6px;
  1118. padding: 3px;
  1119. color: white;
  1120. box-shadow: 0 2px 3px grey;
  1121. /*box-sizing: border-box;*/
  1122. /*border: 2px solid #00000000;*/
  1123. transition: all .5s;
  1124. }
  1125. .CKFOMAN-toolbar-btns:hover{
  1126. /*filter: brightness(0.85);*/
  1127. background: #00467e!important;
  1128. transition: all .15s;
  1129. /*border-bottom: solid 2px white;*/
  1130. }
  1131. .CKFOMAN-toolbar-btns.red{
  1132. background: #e91e63!important;
  1133. }
  1134. .CKFOMAN-toolbar-btns:hover.red{
  1135. background: #8c002f!important;
  1136. }
  1137. .CKFOMAN-toolbar-btns.green{
  1138. background: #4caf50!important;
  1139. }
  1140. .CKFOMAN-toolbar-btns:hover.green{
  1141. background: #1b5e20!important;
  1142. }
  1143. .CKFOMAN-toolbar-btns.orange{
  1144. background: #e64a19!important;
  1145. }
  1146. .CKFOMAN-toolbar-btns:hover.orange{
  1147. background: #bf360c!important;
  1148. }
  1149. .CKFOMAN-toolbar-btns.grey{
  1150. background: #949494!important;
  1151. color: grey!important;
  1152. }
  1153. .CKFOMAN-toolbar-btns:hover.grey{
  1154. background: #878787!important;
  1155. color: grey!important;
  1156. }
  1157. #CKFOMAN-sortbtns-container>button{
  1158. flex: 1 0 40% !important;
  1159. margin: 4px 4px;
  1160. }
  1161. #CKFOMAN .mdi-close:hover{
  1162. color: #ff5722;
  1163. }
  1164. `, "CKFOMAN-mainWindowcss", "unique");
  1165. const id = "CKFOMAN";
  1166. let win = document.querySelector("#" + id);
  1167. if (win) return win;
  1168. win = document.createElement("div");
  1169. win.id = id;
  1170.  
  1171. const closebtn = document.createElement("div");
  1172. closebtn.innerHTML = `<i class="mdi mdi-18px mdi-close"></i>`
  1173. closebtn.style.float = "right";
  1174. closebtn.style.color = (getBgColor()==="white")?"black":"white";
  1175. closebtn.onclick = hidePanel;
  1176. win.appendChild(closebtn);
  1177.  
  1178. const titleText = document.createElement("div");
  1179. titleText.id="CKFOMAN-titledom";
  1180. titleText.innerHTML = cfg.titleTemplate();
  1181. win.appendChild(titleText);
  1182.  
  1183. const infoBar = document.createElement("div");
  1184. infoBar.style.height = "30px";
  1185. infoBar.style.lineHeight = "30px";
  1186. infoBar.style.width = "100%";
  1187. infoBar.style.textAlign = "center";
  1188. infoBar.id = id + "-infobar";
  1189. win.appendChild(infoBar);
  1190.  
  1191. const container = document.createElement("div");
  1192. container.id = id + "-container";
  1193. win.appendChild(container);
  1194.  
  1195. win.className = "hide";
  1196. document.body.appendChild(win);
  1197. return win;
  1198. }
  1199. const getContainer = () => {
  1200. return getFloatWindow().querySelector("#CKFOMAN-container");
  1201. }
  1202. const setInfoBar = (content = '') => {
  1203. const bar = getFloatWindow().querySelector("#CKFOMAN-infobar");
  1204. if (bar) bar.innerHTML = content;
  1205. return bar;
  1206. }
  1207. const resetInfoBar = () => {
  1208. wait(50).then(() => {
  1209. let str = cfg.infobarTemplate();
  1210. if (datas.checked.length > 0) {
  1211. str += `,已选中 ${datas.checked.length} 条`;
  1212. }
  1213. setInfoBar(str);
  1214. });
  1215. }
  1216. const divider = () => {
  1217. const div = document.createElement("div");
  1218. div.style.width = "30%";
  1219. div.style.borderBottom = "solid grey 2px";
  1220. div.style.lineHeight = "3px";
  1221. div.style.margin = "0 auto";
  1222. div.style.marginBottom = "3px";
  1223. div.style.height = "6px";
  1224. div.style.overflow = "hidden";
  1225. div.style.padding = "0";
  1226. return div;
  1227. }
  1228. const showPanel = () => {
  1229. const panel = getFloatWindow();
  1230. panel.style.backgroundColor = getBgColor();
  1231. panel.className = "show";
  1232. }
  1233. const hidePanel = () => {
  1234. const panel = getFloatWindow();
  1235. panel.className = "hide";
  1236. }
  1237. const openModal = (title = '', content) => {
  1238. blockWindow();
  1239. let modal = get("#CKFOMAN-modal");
  1240. if (!modal) modal = initModal();
  1241. modal.setTitle(title);
  1242. modal.setContent(content);
  1243. modal.show();
  1244. }
  1245. const isModalShowing = () => {
  1246. let modal = get("#CKFOMAN-modal");
  1247. if (modal) return modal.classList.contains("show");
  1248. else return false;
  1249. }
  1250. const hideModal = () => {
  1251. blockWindow(false);
  1252. let modal = get("#CKFOMAN-modal");
  1253. if (modal) modal.hide();
  1254. }
  1255. const initModal = () => {
  1256. addStyle(`
  1257. #CKFOMAN-modal{
  1258. position: fixed;
  1259. z-index: 99010;
  1260. top: 50%;
  1261. left: 50%;
  1262. width: 300px;
  1263. width: 30vw;
  1264. /*height: 300px;
  1265. height: 50vh;*/
  1266. background: white;
  1267. border-radius: 8px;
  1268. padding: 12px;
  1269. transform: translate(-50%,-50%);
  1270. transition: all .3s;
  1271. box-shadow: 0 2px 8px grey;
  1272. max-height: 95vh;
  1273. overflow-y: auto;
  1274. }
  1275. #CKFOMAN-modal.show{
  1276. opacity: 1;
  1277. transform: translate(-50%,-50%) scale(1);
  1278. }
  1279. #CKFOMAN-modal.hide{
  1280. opacity: 0;
  1281. pointer-events: none;
  1282. transform: translate(-50%,-50%) scale(0.9);
  1283. }
  1284. .CKFOMAN-modal-content>div{
  1285. display: flex;
  1286. margin: 6px 10px;
  1287. flex-wrap: wrap;
  1288. flex-direction: column;
  1289. align-content: space-around;
  1290. justify-content: space-between;
  1291. align-items: stretch;
  1292. }
  1293. .CKFOMAN-modal-content button,
  1294. .CKFOMAN-modal-content input,
  1295. .CKFOMAN-modal-content keygen,
  1296. .CKFOMAN-modal-content optgroup,
  1297. .CKFOMAN-modal-content select,
  1298. .CKFOMAN-modal-content textarea
  1299. {
  1300. border-width: 2px;
  1301. border-color: transparent;
  1302. margin: 2px;
  1303. border-radius: 3px;
  1304. transition: all .3s;
  1305. }
  1306. .CKFOMAN-modal-content button:hover,
  1307. .CKFOMAN-modal-content input:hover,
  1308. .CKFOMAN-modal-content keygen:hover,
  1309. .CKFOMAN-modal-content optgroup:hover,
  1310. .CKFOMAN-modal-content select:hover,
  1311. .CKFOMAN-modal-content textarea:hover
  1312. {
  1313. border-color: grey;
  1314. }
  1315.  
  1316. .CKFOMAN-toolbar-btns>i.mdi {
  1317. float: right;
  1318. }
  1319. `, "CKFOMAN-modal-css", "unique");
  1320. const modal = document.createElement("div");
  1321. modal.id = "CKFOMAN-modal";
  1322. modal.className = "hide";
  1323.  
  1324. const header = document.createElement("h2");
  1325. header.className = "CKFOMAN-modal-title"
  1326. modal.appendChild(header);
  1327.  
  1328. modal.setTitle = (t = '') => {
  1329. header.innerHTML = t;
  1330. }
  1331.  
  1332. const contents = document.createElement("div");
  1333. contents.className = "CKFOMAN-modal-content";
  1334. modal.appendChild(contents);
  1335.  
  1336. modal.setContent = async (c) => {
  1337. let ct = c;
  1338. if (ct instanceof Function) {
  1339. ct = await ct();
  1340. }
  1341. if (ct instanceof HTMLElement) {
  1342. contents.innerHTML = '';
  1343. contents.appendChild(ct);
  1344. return;
  1345. }
  1346. if (ct instanceof String) {
  1347. contents.innerHTML = ct;
  1348. return;
  1349. }
  1350. log("unknown: ", ct);
  1351. }
  1352. modal.addContent = async (c) => {
  1353. let ct = c;
  1354. if (ct instanceof Function) {
  1355. ct = await ct();
  1356. }
  1357. if (ct instanceof HTMLElement) {
  1358. contents.appendChild(ct);
  1359. return;
  1360. }
  1361. if (ct instanceof String) {
  1362. contents.innerHTML += ct;
  1363. return;
  1364. }
  1365. log("unknown: ", ct);
  1366. }
  1367.  
  1368. modal.close = closeModal;
  1369. modal.open = openModal;
  1370. modal.show = () => {
  1371. modal.style.backgroundColor = getBgColor();
  1372. modal.className = "show";
  1373. }
  1374. modal.hide = () => {
  1375. modal.className = "hide";
  1376. }
  1377.  
  1378. document.body.appendChild(modal);
  1379. return modal;
  1380. }
  1381. const closeModal = () => {
  1382. blockWindow(false);
  1383. let modal = get("#CKFOMAN-modal");
  1384. if (modal) modal.remove();
  1385. }
  1386. const addMdiBtnStyle = () => {
  1387. if (document.querySelector("#CKFOMAN-MDICSS")) return;
  1388. document.head.innerHTML += `<link id="CKFOMAN-MDICSS" rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@mdi/font@5.9.55/css/materialdesignicons.min.css"/>`;
  1389. }
  1390. const refreshChecked = () => {
  1391. setInfoBar(`正在刷新后台数据...`);
  1392. const all = getAll("#CKFOMAN .CKFOMAN-data-inforow-toggle");
  1393. if (!all) return;
  1394. for (let it of all) {
  1395. const mid = it.getAttribute("data-targetmid");
  1396. if (it.checked) {
  1397. if (!datas.checked.includes(mid) && !datas.checked.includes(parseInt(mid))) {
  1398. datas.checked.push(mid);
  1399. }
  1400. } else {
  1401. if (datas.checked.includes(mid)) {
  1402. datas.checked.splice(datas.checked.indexOf(mid), 1);
  1403. } else if (datas.checked.includes(parseInt(mid))) {
  1404. datas.checked.splice(datas.checked.indexOf(parseInt(mid)), 1);
  1405. }
  1406. }
  1407. }
  1408. resetInfoBar();
  1409. return datas.checked;
  1410. }
  1411. const toggleSwitch = (mid, status = false, operateDom = true) => {
  1412. setToggleStatus(mid, status, operateDom);
  1413. //unsafeWindow.postMessage(`CKFOMANSTATUSCHANGES|${mid}|${status ? 1 : 0}`)
  1414. }
  1415. const upinfoline = async data => {
  1416. let invalid = isInvalid(data);
  1417. let info = datas.mappings[parseInt(data.mid)] || {};
  1418. return await makeDom("li", async item => {
  1419. item.className = "CKFOMAN-data-inforow";
  1420. if(datas.settings.lazyRenderForList)item.style.contentVisibility = "auto";
  1421. item.onclick = e => {
  1422. if (e.target.classList.contains("CKFOMAN-data-inforow-name")) {
  1423. //open("https://space.bilibili.com/" + data.mid);
  1424. createUserInfoCard(info);
  1425. } else if (e.target.tagName !== "INPUT") {
  1426. const toggle = item.querySelector("input");
  1427. toggleSwitch(data.mid, !toggle.checked);
  1428. //toggle.checked = !toggle.checked;
  1429. } else {
  1430. const toggle = item.querySelector("input");
  1431. wait(50).then(() => toggleSwitch(data.mid, toggle.checked, false))
  1432. }
  1433. //resetInfoBar();
  1434. }
  1435. item.setAttribute("data-special", data.special);
  1436. item.setAttribute("data-invalid", data.invalid ? "1" : "0");
  1437. item.setAttribute("data-fans", data.attribute === 6 ? "1" : "0");
  1438. item.setAttribute("data-vip", data.vip.vipType !== 0 ? "1" : "0");
  1439. item.setAttribute("data-official", data.official_verify.type === 1 ? "1" : "0");
  1440. let title = data.mid + "";
  1441. item.appendChild(await makeDom("input", toggle => {
  1442. toggle.className = "CKFOMAN-data-inforow-toggle";
  1443. toggle.type = "checkbox";
  1444. toggle.checked = datas.checked.includes(data.mid + "") || datas.checked.includes(parseInt(data.mid));
  1445. toggle.setAttribute("data-targetmid", data.mid);
  1446. toggle.onchange = e => {
  1447. toggleSwitch(data.mid, toggle.checked);
  1448. }
  1449. // toggle.onchange = e =>{
  1450. // if(toggle.checked){
  1451. // if(!datas.checked.includes(data.mid)){
  1452. // datas.checked.push(data.mid);
  1453. // }
  1454. // }else{
  1455. // if(datas.checked.includes(data.mid)){
  1456. // datas.checked.splice(datas.checked.indexOf(data.mid),1);
  1457. // }
  1458. // }
  1459. // }
  1460. }));
  1461. item.appendChild(await makeDom("img", avatar => {
  1462. avatar.src = data.face;
  1463. avatar.style.flex = "1";
  1464. avatar.style.height = "18px";
  1465. avatar.style.maxWidth = "18px";
  1466. avatar.style.borderRadius = "50%";
  1467. avatar.style.verticalAlign = "middle";
  1468. avatar.style.marginRight = "18px";
  1469. avatar.loading = "lazy";
  1470. }));
  1471. item.appendChild(await makeDom("span", name => {
  1472. name.className = "CKFOMAN-data-inforow-name";
  1473. name.innerText = data.uname;
  1474. name.style.flex = "1";
  1475. if (invalid) {
  1476. name.style.textDecoration = "line-through 3px red";
  1477. } else {
  1478. name.style.fontWeight = "bold";
  1479. if (data.isWhisper === true || data.attribute=== 1) {
  1480. name.innerHTML = `<i class="mdi mdi-18px mdi-eye-off" style="vertical-align: middle;color:gray!important" title="悄悄关注"></i>` + name.innerHTML;
  1481. title += " | 悄悄关注";
  1482. }
  1483. if (data.special === 1) {
  1484. name.innerHTML = `<i class="mdi mdi-18px mdi-heart" style="vertical-align: middle;color:orangered!important" title="特别关注"></i>` + name.innerHTML;
  1485. title += " | 特别关注";
  1486. }
  1487. if (data.attribute === 6) {
  1488. name.innerHTML = `<i class="mdi mdi-18px mdi-swap-horizontal" style="vertical-align: middle;color:orangered!important" title="互相关注"></i>` + name.innerHTML;
  1489. title += " | 互相关注";
  1490. }
  1491. if (data.vip.vipType !== 0) {
  1492. name.style.color = "#e91e63";
  1493. }
  1494. if (data.official_verify.type === 1) {
  1495. name.style.textDecoration = "underline";
  1496. name.style.color = "#c67927";
  1497. title += " | 认证账号";
  1498. }
  1499. if (info.banned) {
  1500. name.style.color = "grey";
  1501. name.innerHTML = `<i class="mdi mdi-18px mdi-cancel" style="vertical-align: middle;color:red!important" title="账号已封禁"></i>` + name.innerHTML;
  1502. title += " | 账号已封禁";
  1503. }
  1504. if (info.RIP) {
  1505. name.innerHTML = `<i class="mdi mdi-18px mdi-candle" style="vertical-align: middle;color:black!important" title="纪念账号"></i>` + name.innerHTML;
  1506. title += " | 纪念账号";
  1507. }
  1508. if (info.disputed) {
  1509. name.innerHTML = name.innerHTML + `<i class="mdi mdi-18px mdi-frequently-asked-questions" style="vertical-align: middle;color:orangered!important" title="账号有争议"></i>`;
  1510. title += " | 账号有争议";
  1511. }
  1512. if (info.notice && info.notice.content && !info.banned && !info.RIP && !info.disputed) {
  1513. name.innerHTML = name.innerHTML + `<i class="mdi mdi-18px mdi-information" style="vertical-align: middle;color:grey!important" title="${info.notice.toString()}"></i>`;
  1514. title += " | " + (info.notice.content ? info.notice.content : "账号状态未知");
  1515. }
  1516. }
  1517. }));
  1518. item.appendChild(await makeDom("span", subtime => {
  1519. subtime.style.flex = "1";
  1520. subtime.innerHTML = "关注于" + (new Intl.DateTimeFormat('zh-CN').format(data.mtime + "000"));
  1521. if (isNearly(data.mtime)) {
  1522. title += " | 最近关注";
  1523. } else if (isLongAgo(data.mtime)) {
  1524. title += " | 很久前关注";
  1525. }
  1526. }));
  1527. item.appendChild(await makeDom("span", tagsdom => {
  1528. tagsdom.style.flex = "1";
  1529. if (data.tag === null || data.tag.length === 0 || ["[0]", "[-10]"].includes(JSON.stringify(data.tag)))
  1530. tagsdom.innerHTML = "";
  1531. else {
  1532. let name = "";
  1533. for (let gid of data.tag) {
  1534. if (gid === 0 || gid === -10) continue;
  1535. if (name !== "") name += ",";
  1536. if (gid in datas.tags) {
  1537. name += datas.tags[gid].name;
  1538. } else {
  1539. name += "?";
  1540. }
  1541. }
  1542. tagsdom.innerHTML = name;
  1543. }
  1544. }));
  1545. item.appendChild(await makeDom("span", mark => {
  1546. mark.style.flex = "1";
  1547. if (invalid) {
  1548. title += " | 账号已注销";
  1549. } else {
  1550. if (data.special === 1) {
  1551. mark.innerHTML = "特别关注&nbsp;&nbsp;";
  1552. }
  1553. if (data.official_verify.type === 1) {
  1554. mark.innerText = data.official_verify.desc.substring(0, 15);
  1555. } else if (data.vip.vipType !== 0) {
  1556. mark.innerText = data.vip.vipType === 1 ? "大会员" : "年费大会员"
  1557. title += " | " + mark.innerText;
  1558. }
  1559. if (info.lastUpdate) {
  1560. if (isLongAgo(info.lastUpdate)) {
  1561. title += " | 很久没有发布视频";
  1562. }
  1563. if (isNearly(info.lastUpdate)) {
  1564. title += " | 最近有发布视频";
  1565. }
  1566. }
  1567. }
  1568. }));
  1569. log(info, title);
  1570. item.setAttribute("title", title);
  1571. });
  1572. }
  1573. const taginfoline = (data,clickCallback=()=>{},selected = false,showExtras = true,hideOptions = false) => {
  1574. return makeDom("li", async item => {
  1575. let couldRename = true;
  1576. item.className = "CKFOMAN-data-inforow";
  1577. item.onclick = e => {
  1578. if(e.path.filter(el=>el.tagName==="BUTTON"||el.tagName==="INPUT").length){
  1579. return;
  1580. }else{
  1581. clickCallback(e,data);
  1582. }
  1583. }
  1584. item.setAttribute("data-id", data.tagid);
  1585. item.setAttribute("data-name", data.name);
  1586. item.setAttribute("data-count", data.count);
  1587. item.setAttribute("data-tip", data.tip);
  1588. if(!hideOptions)item.appendChild(await makeDom("input", toggle => {
  1589. toggle.className = "CKFOMAN-data-inforow-toggle";
  1590. toggle.type = "checkbox";
  1591. toggle.checked = selected;
  1592. toggle.setAttribute("data-tagid", data.tagid);
  1593. }));
  1594. item.appendChild(await makeDom("span", name => {
  1595. name.className = "CKFOMAN-data-inforow-name";
  1596. switch(data.tagid){
  1597. case 0:
  1598. case '0':
  1599. couldRename = false;
  1600. name.innerHTML = `默认分类`.italics();
  1601. item.setAttribute("title", "默认的关注分类,包含全部未分组的关注项目。\n不可删除");
  1602. break;
  1603. case -10:
  1604. case '-10':
  1605. couldRename = false;
  1606. name.innerHTML = `特别关注`.italics();
  1607. item.setAttribute("title", "默认的特别关注分类,包含全部特别关注的关注项目。\n不可删除");
  1608. break;
  1609. default:
  1610. name.innerText = `${data.name}`;
  1611. item.setAttribute("title", `用户创建的分组 "${data.name}"\n删除后用户将被移动到默认分类`);
  1612. }
  1613. name.style.flex = "1";
  1614. }));
  1615. item.appendChild(await makeDom("span", subtime => {
  1616. subtime.style.flex = "1";
  1617. subtime.innerHTML = `${data.tagid}`;
  1618. }));
  1619. item.appendChild(await makeDom("span", subtime => {
  1620. subtime.style.flex = "1";
  1621. subtime.innerHTML = `包含 ${data.count} 个内容`;
  1622. }));
  1623. if(showExtras)item.appendChild(await makeDom("button", renamebtn => {
  1624. renamebtn.style.flex = ".4";
  1625. renamebtn.innerHTML = `更名`;
  1626. renamebtn.style.height = "23px";
  1627. renamebtn.style.margin = "0";
  1628. renamebtn.style.padding = "2px";
  1629. renamebtn.classList.add("CKFOMAN-toolbar-btns");
  1630. if(!couldRename){
  1631. renamebtn.setAttribute("disabled",true);
  1632. renamebtn.classList.add("grey");
  1633. }
  1634. renamebtn.onclick = async ()=>{
  1635. let newname = prompt("请输入新的分类名字",data.name).trim();
  1636. if(newname.length!==0){
  1637. if(newname!=data.name){
  1638. const result = await renameGroup(data.tagid,newname);
  1639. if(result){
  1640. await alertModal("分组重命名","分组重命名成功,重新打开窗口以显示修改后的数据。","确定");
  1641. }else{
  1642. await alertModal("分组重命名","分组重命名完成,但是不能确定结果。请刷新页面,然后查看是否生效。","确定");
  1643. }
  1644. }
  1645. }
  1646. };
  1647. }));
  1648. });
  1649. }
  1650. const doUnfollowChecked = async () => {
  1651. const checked = datas.checked;
  1652. if (!checked || checked.length === 0) return alertModal("无法操作", "实际选中数量为0,没有任何人被选中取关。", "");
  1653. await alertModal("正在取消关注...", `正在取关${checked.length}个用户,请耐心等候~`);
  1654. const result = await unfollowUsers(checked);
  1655. if (result.ok) {
  1656. await alertModal("操作结束", `已取关 ${result.okgroup.length} 个用户。`, "继续");
  1657. } else {
  1658. await alertModal("操作结束", `已取关 ${result.okgroup.length} 个用户,但有另外 ${result.errgroup.length} 个用户取关失败。`, "继续");
  1659. }
  1660. datas.checked = [];
  1661. log("取关结果", result);
  1662. createMainWindow();
  1663. }
  1664. const isInvalid = data => {
  1665. return (data.face === "http://i0.hdslb.com/bfs/face/member/noface.jpg"
  1666. || data.face === "https://i0.hdslb.com/bfs/face/member/noface.jpg")
  1667. && data.uname === "账号已注销";
  1668. }
  1669. const alertModal = async (title = "", content = "", okbtn = "hidden") => {
  1670. if (isModalShowing()) {
  1671. hideModal();
  1672. await wait(200);
  1673. }
  1674. openModal(title, await makeDom("div", async container => {
  1675. container.appendChild(await makeDom("div", tip => {
  1676. tip.innerHTML = content;
  1677. }))
  1678. if (okbtn !== "hidden")
  1679. container.appendChild(await makeDom("div", async btns => {
  1680. btns.style.display = "flex";
  1681. btns.appendChild(await makeDom("button", btn => {
  1682. btn.className = "CKFOMAN-toolbar-btns";
  1683. btn.innerHTML = okbtn;
  1684. btn.onclick = e => hideModal();
  1685. }))
  1686. }))
  1687. }))
  1688. await wait(300);
  1689. }
  1690. const showCacheQuotaModal = async () => {
  1691. hideModal();
  1692. await wait(300);
  1693. const size = CacheManager.getSize();
  1694. let content = "本地缓存空间已占用 " + size + " MB。";
  1695. if(size < 1.8){
  1696. content += "无需处理。定期整理缓存可以减少空间占用。";
  1697. }else if (size < 2.5) {
  1698. content += "<b>建议整理缓存。</b>";
  1699. } else {
  1700. content += "<b>建议整理或清理缓存以避免缓存空间超出配额。</b>";
  1701. }
  1702. content += "<br><br>FoMan使用本地存储空间保存缓存,本地存储在不同浏览器中有不同的限额,最小的为2.5MB(Opera),最大的为10MB(Chromium)。请注意此空间非FoMan独占,B站自身和其他插件也会占用此空间,因此建议经常进行整理。";
  1703. content += "<br><br><b>整理缓存</b>仅会清理过期和过时的缓存。<br><br>默认情况下,FoMan存储的缓存有效期为2小时,超过2小时的缓存和数据总数发生变化时都会触发强制放弃缓存重新加载。"
  1704. content += "<br><br><b>清空缓存</b>会清理所有由FoMan产生的缓存。若配额达到上限,或经常查看其他人关注列表,则建议使用此功能。"
  1705. alertModal("缓存使用说明",content,"确定");
  1706. }
  1707. const createUserInfoCardFromOthers = async(uid)=>{
  1708. if(!uid) return;
  1709. const i = await fillUserStatus(uid, true).catch(err => log(err));
  1710. await createUserInfoCard(i, false, true);
  1711. };
  1712. const createUserInfoCard = async (info, refilldata = true, noactions = false)=>{
  1713. if(datas.preventUserCard) return;
  1714. log(info);
  1715. if(datas.settings.autoExtendInfo){
  1716. alertModal("请稍后...");
  1717. if(refilldata) await fillUserStatus(info.mid).catch(err => log(err));
  1718. info.dynamics = await getDynamic(info.mid).catch(err => log(err));
  1719. info['stats'] = await getCurrSubStat(info.mid);
  1720. }
  1721. hideModal();
  1722. await wait(300);
  1723. openModal("", await makeDom("div", async container => {
  1724. const infocard = await makeDom("div", async card => {
  1725. card.style.display = "flex";
  1726. card.style.flexDirection = "row";
  1727. card.style.minHeight = "100px";
  1728. card.style.minWidth = "400px";
  1729. [
  1730. await makeDom("img", async img => {
  1731. img.style.flex = "1";
  1732. img.style.maxWidth = "70px";
  1733. img.setAttribute("loading","lazy");
  1734. img.src = info.face;
  1735. img.style.width = "70px";
  1736. img.style.height = "70px";
  1737. img.style.borderRadius = "50%";
  1738. img.style.margin = "0 30px";
  1739. }),
  1740. await makeDom("div", async upinfo=>{
  1741. upinfo.style.flex = "1";
  1742. upinfo.style.maxWidth = "300px";
  1743. upinfo.innerHTML = `<b style="color:${info.vip['nickname_color']};font-size: large">${info.uname??info.name??'未知昵称'}</b> <span style="display:inline-block;transform: translateY(-5px);font-size:xx-small;line-height:1.2;padding:1px 3px;border-radius:6px;background: ${info.vip.vipType>0?(info.vip.label['bg_color']||"#f06292"):"rgba(0,0,0,0)"};color: ${info.vip.label['text_color']||"white"}">${info.vip.vipType>1?info.vip.label.text:info.vip.vipType>0?"大会员":""}</span>`;
  1744. if(info.level){
  1745. upinfo.innerHTML+= `<div style="display: inline-block;border-radius:3px;line-height: 1.2;padding: 1px 3px;background:#f06292;margin-left: 12px;color:white">LV${info.level}${isHardCoreMember(info)?" ⚡ (硬核)":""}</div>`;
  1746. }
  1747. upinfo.innerHTML+= `<div style="color:gray;border-left: 2px solid gray;padding-left: 2px;font-style: italic;">${info.sign}</div>`;
  1748. if(info.official_verify.type!==-1){
  1749. let color = "gray";
  1750. switch(info.official_verify.type){
  1751. case 0:
  1752. color="goldenrod";
  1753. break;
  1754. case 1:
  1755. color="#FB7299";
  1756. break;
  1757. case 2:
  1758. color="dodgerblue";
  1759. break;
  1760. }
  1761. upinfo.innerHTML+= `<div style="color:${color}">${info.official_verify.desc}</div>`;
  1762. }
  1763. if(info.stats){
  1764. const { follower, following }=info.stats;
  1765. const [fans,subs] = [numberFormat(follower), numberFormat(following)];
  1766. upinfo.innerHTML+= `<div style="color:gray">${fans.value}${fans.unit}粉丝 / ${subs.value}${subs.unit}关注</div>`;
  1767. }
  1768. if(info.tag){
  1769. let folders = "分类:";
  1770. for(let t of info.tag){
  1771. if(t in datas.tags){
  1772. folders +=" "+datas.tags[t].name;
  1773. }
  1774. }
  1775. upinfo.innerHTML+= `<div style="color:gray;font-weight:bold">${folders}</div>`;
  1776. }
  1777. let subinfo = "";
  1778. if(info.special===1){
  1779. subinfo+= `<span style="color:deeppink;margin-right:6px;">特别关注</span>`;
  1780. }
  1781. if(info.attribute===6){
  1782. subinfo+= `<span style="color:indianred;margin-right:6px;">互相关注</span>`;
  1783. }
  1784. if(info.isWhisper === true || info.attribute=== 1){
  1785. subinfo+= `<span style="color:yellowgreen;margin-right:6px;">悄悄关注</span>`;
  1786. }
  1787. if(subinfo.length){
  1788. upinfo.innerHTML+= `<div>${subinfo}</div>`
  1789. }
  1790. if(info.notice && info.notice.id){
  1791. upinfo.innerHTML+= `<div style="border-radius:6px;padding:3px;background:${info.notice.bg_color};color:${info.notice.text_color};"><a href="${info.notice.url}">${info.notice.content}</a></div>`;
  1792. }
  1793. if(info.banned){
  1794. upinfo.innerHTML+= `<div style="border-radius:6px;padding:3px;background:black;color:white;">账号已封禁</div>`;
  1795. }
  1796. if(info.cates && info.cates.length){
  1797. upinfo.innerHTML+= `<div style="color:gray">标签: ${info.cates.join(", ")}</div>`;
  1798. }
  1799. if(info.mostCates && info.mostCates.length){
  1800. upinfo.innerHTML+= `<div style="color:gray">主要投稿分区: ${info.mostCates}</div>`;
  1801. }
  1802. if(info.mid){
  1803. upinfo.innerHTML+= `<div style="color:gray">UID: ${info.mid}</div>`;
  1804. }
  1805. if(info.mtime){
  1806. const regdate = new Date(info.mtime*1000);
  1807. upinfo.innerHTML+= `<div style="color:gray">关注于 ${regdate.getFullYear()}年${regdate.getMonth()+1}月${regdate.getDate()}日</div>`;
  1808. }
  1809. })
  1810. ].forEach(el=>card.appendChild(el));
  1811. })
  1812. container.appendChild(infocard);
  1813. if(unsafeWindow.FoManPlugins&&unsafeWindow.FoManPlugins.RememberFollows){
  1814. const followinfo = unsafeWindow.FoManPlugins.RememberFollows.get(+info.mid);
  1815. if(followinfo){
  1816. const fodate = new Date(followinfo.timestamp);
  1817. [
  1818. divider(),
  1819. await makeDom("div",async post=>{
  1820. post.innerHTML = "<h3 style='padding: 6px 0;'>关注记录</h3>";
  1821. post.appendChild(await makeDom("div",async vidcard=>{
  1822. vidcard.style.display = "flex";
  1823. vidcard.style.flexDirection = "row";
  1824. vidcard.style.minHeight = "80px";
  1825. vidcard.style.minWidth = "400px";
  1826. [
  1827. await makeDom("div",async vidinfo=>{
  1828. vidinfo.innerHTML = `<div style="font-weight:bold;font-size:larger;color:grey">${followinfo.videoName}</div>`;
  1829. vidinfo.innerHTML+= `<div style="color:grey">${fodate.getFullYear()}年${fodate.getMonth()+1}月${fodate.getDate()}日 · 当时UP名: <a href="https://space.bilibili.com/${followinfo.mid}">${followinfo.upName}</a></div>`;
  1830. })
  1831. ].forEach(el=>vidcard.appendChild(el));
  1832. vidcard.onclick = ()=>open(`https://www.bilibili.com/video/${followinfo.videoId}`)
  1833. }))
  1834. })
  1835. ].forEach(el=>container.appendChild(el));
  1836. }
  1837. }
  1838. if(info.dynamics){
  1839. if(info.dynamics.top){
  1840. let dynamic = info.dynamics.top;
  1841. let content = (()=>{
  1842. if(!dynamic.content || dynamic.content.length===0) return "无内容";
  1843. let short = dynamic.content.substring(0,300);
  1844. short = short.split("\n").slice(0,4).join("\n");
  1845. if(short!=dynamic.content) short+="...";
  1846.  
  1847. return short.replaceAll("\n","<br>");
  1848. })();
  1849. const pushdate = new Date(dynamic.timestamp*1000);
  1850. [
  1851. divider(),
  1852. await makeDom("div",async post=>{
  1853. post.innerHTML = "<h3 style='padding: 6px 0;'>置顶动态</h3>";
  1854. post.appendChild(await makeDom("div",async vidcard=>{
  1855. vidcard.style.display = "flex";
  1856. vidcard.style.flexDirection = "row";
  1857. vidcard.style.minHeight = "80px";
  1858. vidcard.style.minWidth = "400px";
  1859. [
  1860. await makeDom("div",async vidinfo=>{
  1861. vidinfo.innerHTML = `<div style="font-weight:normal;font-size:smaller;color:#858585">[${dynamic.prefix}] ${content}</div>`;
  1862. vidinfo.innerHTML+= `<div style="color:grey"><i class="mdi mdi-10px mdi-chevron-double-right"></i> ${pushdate.getFullYear()}年${pushdate.getMonth()+1}月${pushdate.getDate()}日 - ${dynamic.like??'?'}点赞 ${dynamic.repost??'?'}转发 ${dynamic.comment??'?'}评论</div>`;
  1863. if(dynamic.isrepost){
  1864. vidinfo.innerHTML+= `<div style="color:grey"><i class="mdi mdi-10px mdi-share"></i> 转发自<b onclick="open('https://space.bilibili.com/${dynamic.publisher.uid}')">${dynamic.publisher.uname}</b> 的 [${dynamic.origprefix}]:<div style='border-left: 2px solid gray;padding-left:6px'>${dynamic.origin.substr(0,100)}...</div></div>`;
  1865. }
  1866. })
  1867. ].forEach(el=>vidcard.appendChild(el));
  1868. vidcard.onclick = ()=>open(`https://t.bilibili.com/${dynamic.id}?tab=2`)
  1869. }))
  1870. })
  1871. ].forEach(el=>container.appendChild(el));
  1872. }
  1873. if(info.dynamics.next){
  1874. let dynamic = info.dynamics.next;
  1875. let content = (()=>{
  1876. if(!dynamic.content || dynamic.content.length===0) return "无内容";
  1877. let short = dynamic.content.substring(0,300);
  1878. short = short.split("\n").slice(0,4).join("\n");
  1879. if(short!=dynamic.content) short+="...";
  1880. return short.replaceAll("\n","<br>");
  1881. })();
  1882. const pushdate = new Date(dynamic.timestamp*1000);
  1883. [
  1884. divider(),
  1885. await makeDom("div",async post=>{
  1886. post.innerHTML = "<h3 style='padding: 6px 0;'>最新动态</h3>";
  1887. post.appendChild(await makeDom("div",async vidcard=>{
  1888. vidcard.style.display = "flex";
  1889. vidcard.style.flexDirection = "row";
  1890. vidcard.style.minHeight = "80px";
  1891. vidcard.style.minWidth = "400px";
  1892. [
  1893. await makeDom("div",async vidinfo=>{
  1894. vidinfo.innerHTML = `<div style="font-weight:normal;font-size:smaller;color:#858585">[${dynamic.prefix}] ${content}</div>`;
  1895. vidinfo.innerHTML+= `<div style="color:grey"><i class="mdi mdi-10px mdi-chevron-double-right"></i> ${pushdate.getFullYear()}年${pushdate.getMonth()+1}月${pushdate.getDate()}日 - ${dynamic.like??'?'}点赞 ${dynamic.repost??'?'}转发 ${dynamic.comment??'?'}评论</div>`;
  1896. if(dynamic.isrepost){
  1897. vidinfo.innerHTML+= `<div style="color:grey"><i class="mdi mdi-10px mdi-share"></i> 转发自<b onclick="open('https://space.bilibili.com/${dynamic.publisher.uid}')">${dynamic.publisher.uname}</b> 的 [${dynamic.origprefix}]:<div style='border-left: 2px solid gray;padding-left:6px'>${dynamic.origin.substr(0,100)}...</div></div>`;
  1898. }
  1899. })
  1900. ].forEach(el=>vidcard.appendChild(el));
  1901. vidcard.onclick = ()=>open(`https://t.bilibili.com/${dynamic.id}?tab=2`)
  1902. }))
  1903. })
  1904. ].forEach(el=>container.appendChild(el));
  1905. }
  1906. }
  1907. if(info.lastUpdate && info.lastUpdateInfo){
  1908. const pushdate = new Date(info.lastUpdate*1000);
  1909. [
  1910. divider(),
  1911. await makeDom("div",async post=>{
  1912. post.innerHTML = "<h3 style='padding: 6px 0;'>最新投稿</h3>";
  1913. post.appendChild(await makeDom("div",async vidcard=>{
  1914. vidcard.style.display = "flex";
  1915. vidcard.style.flexDirection = "row";
  1916. vidcard.style.minHeight = "80px";
  1917. vidcard.style.minWidth = "400px";
  1918. [
  1919. await makeDom("img", img=>{
  1920. img.style.flex = "1";
  1921. img.style.maxWidth = "80px";
  1922. img.style.height = "50px";
  1923. img.setAttribute("loading","lazy");
  1924. img.src = info.lastUpdateInfo.pic;
  1925. img.style.borderRadius = "6px";
  1926. img.style.margin = "0px 12px 0px 10px";
  1927. }),
  1928. await makeDom("div",async vidinfo=>{
  1929. vidinfo.innerHTML = `<div style="font-weight:bold;font-size:larger;color:grey">${info.lastUpdateInfo.title}</div>`;
  1930. vidinfo.innerHTML+= `<div style="color:grey">${pushdate.getFullYear()}年${pushdate.getMonth()+1}月${pushdate.getDate()}日</div>`;
  1931. })
  1932. ].forEach(el=>vidcard.appendChild(el));
  1933. vidcard.onclick = ()=>open(`https://www.bilibili.com/av${info.lastUpdateInfo.aid}`)
  1934. }))
  1935. })
  1936. ].forEach(el=>container.appendChild(el));
  1937. }
  1938. if(info.lives && info.lives.liveStatus!==0){
  1939. [
  1940. divider(),
  1941. await makeDom("div",async post=>{
  1942. post.innerHTML = "<h3 style='padding: 6px 0;'>直播间</h3>";
  1943. post.appendChild(await makeDom("div",async vidcard=>{
  1944. vidcard.style.display = "flex";
  1945. vidcard.style.flexDirection = "row";
  1946. vidcard.style.minHeight = "80px";
  1947. vidcard.style.minWidth = "400px";
  1948. [
  1949. await makeDom("img", img=>{
  1950. img.style.flex = "1";
  1951. img.style.maxWidth = "80px";
  1952. img.style.height = "50px";
  1953. img.setAttribute("loading","lazy");
  1954. img.src = info.lives.cover;
  1955. img.style.borderRadius = "6px";
  1956. img.style.margin = "0px 12px 0px 10px";
  1957. }),
  1958. await makeDom("div",async vidinfo=>{
  1959. vidinfo.innerHTML = `<div style="font-weight:bold;font-size:larger;color:grey">${info.lives.title}</div>`;
  1960. vidinfo.innerHTML+= `<div style="color:grey">正在${info.lives.liveStatus===2?'轮':'直'}播 - 房间号: ${info.lives.roomid}</div>`;
  1961. })
  1962. ].forEach(el=>vidcard.appendChild(el));
  1963. vidcard.onclick = ()=>open(`https://live.bilibili.com/${info.lives.roomid}`)
  1964. }))
  1965. })
  1966. ].forEach(el=>container.appendChild(el));
  1967. }
  1968. async function addBtn(info,container){
  1969. container.style.display="flex";
  1970. container.style.flexDirection = "column";
  1971. container.style.position = "sticky";
  1972. container.style.bottom = 0;
  1973. container.style.background = getBgColor();
  1974. container.innerHTML = "";
  1975. if(!noactions){
  1976. if(info.attribute===0){
  1977. container.appendChild(await makeDom("button", btn => {
  1978. btn.className = "CKFOMAN-toolbar-btns red";
  1979. btn.style.margin = "4px 0";
  1980. btn.innerHTML = "立刻关注";
  1981. btn.onclick = async e => {
  1982. btn.innerHTML = "正在关注...";
  1983. btn.setAttribute("disabled",true)
  1984. btn.classList.add("grey");
  1985. const res = await batchOperateUser([info.mid],RELE_ACTION.FOLLOW);
  1986. if(!res.ok){
  1987. log(res)
  1988. btn.innerHTML = "关注失败";
  1989. btn.removeAttribute("disabled")
  1990. btn.classList.remove("grey");
  1991. }else{
  1992. datas.mappings[info.mid].attribute = 1;
  1993. btn.remove();
  1994. addBtn(datas.mappings[info.mid],container);
  1995. }
  1996. }
  1997. }))
  1998. container.appendChild(await makeDom("button", btn => {
  1999. btn.className = "CKFOMAN-toolbar-btns blue";
  2000. btn.style.margin = "4px 0";
  2001. btn.innerHTML = "悄悄关注";
  2002. btn.onclick = async e => {
  2003. btn.innerHTML = "正在关注...";
  2004. btn.setAttribute("disabled",true)
  2005. btn.classList.add("grey");
  2006. const res = await batchOperateUser([info.mid],RELE_ACTION.WHISPER);
  2007. if(!res.ok){
  2008. log(res)
  2009. btn.innerHTML = "关注失败";
  2010. btn.removeAttribute("disabled")
  2011. btn.classList.remove("grey");
  2012. }else{
  2013. datas.mappings[info.mid].attribute = 1;
  2014. btn.remove();
  2015. addBtn(datas.mappings[info.mid],container);
  2016. }
  2017. }
  2018. }))
  2019. }else{
  2020. container.appendChild(await makeDom("button", btn => {
  2021. btn.className = "CKFOMAN-toolbar-btns red";
  2022. btn.style.margin = "4px 0";
  2023. btn.innerHTML = "立刻取关(谨慎)";
  2024. btn.onclick = async e => {
  2025. btn.innerHTML = "正在取关...";
  2026. btn.setAttribute("disabled",true)
  2027. btn.classList.add("grey");
  2028. const res = await unfollowUser(info.mid);
  2029. if(!res.ok){
  2030. log(res);
  2031. btn.innerHTML = "取关失败";
  2032. btn.removeAttribute("disabled")
  2033. btn.classList.remove("grey");
  2034. }else{
  2035. datas.mappings[info.mid].attribute = 0;
  2036. btn.remove();
  2037. addBtn(datas.mappings[info.mid],container);
  2038. }
  2039. }
  2040. }))
  2041. if(info.attribute===1){
  2042. container.appendChild(await makeDom("button", btn => {
  2043. btn.className = "CKFOMAN-toolbar-btns blue";
  2044. btn.style.margin = "4px 0";
  2045. btn.innerHTML = "转为普通关注(不保留关注时间)";
  2046. btn.onclick = async e => {
  2047. btn.innerHTML = "正在转换...";
  2048. btn.setAttribute("disabled",true)
  2049. btn.classList.add("grey");
  2050. const res = await convertToFollow([info.mid]);
  2051. if(!res.ok){
  2052. log(res)
  2053. btn.innerHTML = "关注失败";
  2054. btn.removeAttribute("disabled")
  2055. btn.classList.remove("grey");
  2056. }else{
  2057. datas.mappings[info.mid].attribute = 1;
  2058. datas.mappings[info.mid].isWhisper = false;
  2059. btn.remove();
  2060. if(datas.dommappings[info.mid+""]&& datas.dommappings[info.mid+""] instanceof HTMLElement){
  2061. datas.dommappings[info.mid+""].replaceWith(await upinfoline(datas.mappings[info.mid]));
  2062. }
  2063. //addBtn(datas.mappings[info.mid],container);
  2064. hideModal();
  2065. }
  2066. }
  2067. }))
  2068. }else{
  2069. container.appendChild(await makeDom("button", btn => {
  2070. btn.className = "CKFOMAN-toolbar-btns blue";
  2071. btn.style.margin = "4px 0";
  2072. btn.innerHTML = "转为悄悄关注(不保留关注时间)";
  2073. btn.onclick = async e => {
  2074. btn.innerHTML = "正在悄悄关注...";
  2075. btn.setAttribute("disabled",true)
  2076. btn.classList.add("grey");
  2077. const res = await convertToWhisper([info.mid]);
  2078. if(!res.ok){
  2079. log(res)
  2080. btn.innerHTML = "关注失败";
  2081. btn.removeAttribute("disabled")
  2082. btn.classList.remove("grey");
  2083. }else{
  2084. datas.mappings[info.mid].attribute = 2;
  2085. datas.mappings[info.mid].isWhisper = true;
  2086. btn.remove();
  2087. if(datas.dommappings[info.mid+""]&& datas.dommappings[info.mid+""] instanceof HTMLElement){
  2088. datas.dommappings[info.mid+""].replaceWith(await upinfoline(datas.mappings[info.mid]));
  2089. }
  2090. //addBtn(datas.mappings[info.mid],container);
  2091. hideModal();
  2092. }
  2093. }
  2094. }))
  2095. }
  2096. }
  2097. }
  2098. container.appendChild(await makeDom("button", btn => {
  2099. btn.className = "CKFOMAN-toolbar-btns";
  2100. btn.style.margin = "4px 0";
  2101. btn.innerHTML = "个人主页";
  2102. btn.onclick = () => open(`https://space.bilibili.com/${info.mid}`)
  2103. }))
  2104. container.appendChild(await makeDom("button", btn => {
  2105. btn.className = "CKFOMAN-toolbar-btns";
  2106. btn.style.margin = "4px 0";
  2107. btn.innerHTML = "隐藏";
  2108. btn.onclick = () => hideModal();
  2109. }))
  2110. }
  2111. const btns = document.createElement("div");
  2112. await addBtn(info,btns);
  2113. container.appendChild(btns);
  2114. }));
  2115. }
  2116. const createGroupInfoModal = async () => {
  2117. hideModal();
  2118. await wait(300);
  2119. openModal("分组管理", await makeDom("div", async container=>{
  2120. container.appendChild(await makeDom("div", tip => {
  2121. tip.style.fontWeight = "bold";
  2122. tip.innerHTML = `若修改过分组信息,建议刷新页面再进行其他操作。`;
  2123. }))
  2124. container.appendChild(divider());
  2125. const taglistdom = document.createElement('div');
  2126. taglistdom.className = "CKFOMAN-scroll-list";
  2127. taglistdom.style.width = "100%";
  2128. taglistdom.style.maxHeight = "calc(50vh - 100px)";
  2129. const refreshList = async ()=>renderTagListTo(taglistdom,[],async (e,data)=>{
  2130. if(e.target.tagName==="INPUT") return;
  2131. if(['0','-10'].includes(data.tagid+'')) return;
  2132. let dom = e.path.filter(it=>it['classList']&&it.classList.contains('CKFOMAN-data-inforow'))[0];
  2133. if(!dom) return log('no target');
  2134. if(dom.hasAttribute('data-del-pending')){
  2135. if(dom.removePendingTimer) clearTimeout(dom.removePendingTimer);
  2136. removeGroup(data.tagid).then(()=>refreshList());
  2137. //cfg.infobarTemplate = `共读取 ${datas.fetched} 条关注 (已修改分组,<a href="javascript:void(0)" onclick="openFollowManager(true)">点此重新加载</a>)`;
  2138. await renderListTo(get("#CKFOMAN-MAINLIST"),datas.followings,true);
  2139. resetInfoBar();
  2140. }else{
  2141. dom.setAttribute('data-del-pending','waiting');
  2142. let namedom = dom.querySelector('.CKFOMAN-data-inforow-name');
  2143. if(!namedom) return;
  2144. let text = namedom.innerHTML;
  2145. namedom.innerHTML = '再次点击以移除'.fontcolor('red');
  2146. dom.removePendingTimer = setTimeout(()=>{
  2147. if(dom.hasAttribute('data-del-pending')) dom.removeAttribute('data-del-pending');
  2148. if(dom.removePendingTimer) clearTimeout(dom.removePendingTimer);
  2149. namedom.innerHTML = text;
  2150. },5000);
  2151. }
  2152. },true);
  2153. container.appendChild(taglistdom);
  2154. container.appendChild(await makeDom("div", async btns => {
  2155. btns.style.display = "flex";
  2156. [
  2157. await makeDom("button", btn => {
  2158. btn.className = "CKFOMAN-toolbar-btns";
  2159. btn.innerHTML = "添加分组";
  2160. btn.style.height = "30px";
  2161. btn.onclick = async () => {
  2162. const tagname = prompt("请输入新分组的标题");
  2163. if(!tagname) return;
  2164. createGroup(tagname).then(()=>refreshList());
  2165. };
  2166. }),
  2167. await makeDom("button", btn => {
  2168. btn.className = "CKFOMAN-toolbar-btns";
  2169. btn.style.height = "30px";
  2170. btn.innerHTML = "关闭";
  2171. btn.onclick = () => hideModal();
  2172. }),
  2173. ].forEach(el => btns.appendChild(el));
  2174. }))
  2175. refreshList();
  2176. }))
  2177. }
  2178. const createGroupChangeModal = async (mode='copy'/*move*/) => {
  2179. hideModal();
  2180. await wait(300);
  2181. refreshChecked();
  2182. let uids = datas.checked;
  2183. let users = [];
  2184. let groups = [];
  2185. let act = mode==='copy'?'复制':'移动';
  2186. for(let uid of uids){
  2187. users.push(datas.mappings[uid]);
  2188. let tags = datas.mappings[uid].tag;
  2189. tags && tags.forEach(t=>groups.includes(t)||groups.push(t))
  2190. }
  2191. log(users,groups);
  2192. openModal("分组修改:"+act, await makeDom("div", async container=>{
  2193. container.appendChild(await makeDom("div", tip => {
  2194. tip.style.fontWeight = "bold";
  2195. tip.innerHTML = `若修改过分组信息,建议刷新页面再进行其他操作。`;
  2196. }))
  2197. container.appendChild(divider());
  2198. const taglistdom = document.createElement('div');
  2199. taglistdom.className = "CKFOMAN-scroll-list";
  2200. taglistdom.style.width = "100%";
  2201. taglistdom.style.maxHeight = "calc(50vh - 100px)";
  2202. const refreshList = async ()=>renderTagListTo(taglistdom,mode==='copy'?[]:groups,async (e,data)=>{
  2203. const row = e.path.filter(el=>el.classList?.contains('CKFOMAN-data-inforow'));
  2204. if(row.length){
  2205. const cb = row[0].querySelector("input[type='checkbox']");
  2206. if(cb) cb.checked = !cb.checked
  2207. }
  2208. },false);
  2209. container.appendChild(taglistdom);
  2210. container.appendChild(await makeDom("div", async btns => {
  2211. btns.style.display = "flex";
  2212. [
  2213. await makeDom("button", btn => {
  2214. btn.className = "CKFOMAN-toolbar-btns";
  2215. btn.style.height = "30px";
  2216. btn.innerHTML = "管理分组 (Beta)";
  2217. btn.onclick = async () => createGroupInfoModal();
  2218. }),
  2219. await makeDom("button", btn => {
  2220. btn.className = "CKFOMAN-toolbar-btns";
  2221. btn.style.height = "30px";
  2222. btn.innerHTML = "取消";
  2223. btn.onclick = () => hideModal();
  2224. }),
  2225. await makeDom("button", btn => {
  2226. btn.className = "CKFOMAN-toolbar-btns";
  2227. btn.style.height = "30px";
  2228. btn.innerHTML = "确定";
  2229. btn.onclick = async () => {
  2230. const allOptions = [...document.querySelectorAll('.CKFOMAN-data-inforow-toggle[data-tagid]')]
  2231. const selections = allOptions.map((option)=>{
  2232. return {tagid:parseInt(option.getAttribute('data-tagid')),checked:option.checked}
  2233. })
  2234. const checked = selections.filter((selection) => selection.checked)
  2235. await alertModal("正在处理...", `正在${act}成员到新分组,请稍候`);
  2236. if(checked.length===0) checked.push({tagid:0,checked:true});
  2237. switch(mode){
  2238. case 'copy':
  2239. copyUserToGroup(uids,checked.map(c=>c.tagid));
  2240. break;
  2241. case 'move':
  2242. moveUserToGroup(uids,checked.map(c=>c.tagid));
  2243. break;
  2244. // default:
  2245. // moveUserToDefaultGroup(uids);
  2246. }
  2247. await renderListTo(get("#CKFOMAN-MAINLIST"),datas.followings,true);
  2248. hideModal();
  2249. cfg.infobarTemplate = ()=>`共读取 ${datas.fetched} 条关注 (已修改分组,<a href="javascript:void(0)" onclick="openFollowManager(true)">点此重新加载</a>)`;
  2250. resetInfoBar();
  2251. }
  2252. }),
  2253. ].forEach(el => btns.appendChild(el));
  2254. }))
  2255. refreshList();
  2256. }))
  2257. }
  2258. const makeButtons = (btns = []) => {
  2259. const dom = CKTools.domHelper;
  2260. return dom('div', {
  2261. css: {
  2262. 'display': 'flex',
  2263. 'flex-direction': 'column'
  2264. },
  2265. init: el => {
  2266. for (const btncfg of btns) {
  2267. let opt = Object.assign({
  2268. text: '按钮',
  2269. extras: '',
  2270. init: () => { },
  2271. onclick: () => { }
  2272. }, btncfg);
  2273. dom('button', {
  2274. classnames: ['CKFOMAN-toolbar-btns', ...opt.extras],
  2275. text: opt.text, init: opt.init, listeners: { click: opt.onclick },
  2276. append: el
  2277. })
  2278. }
  2279. }
  2280. });
  2281. }
  2282. const openNewFilterGuideScreen = async () => {
  2283. const dom = CKTools.domHelper;
  2284. hideModal();
  2285. await wait(300);
  2286. let newFilterModuleInstalled = unsafeWindow.FoManPlugins && unsafeWindow.FoManPlugins.FilterReborn;
  2287. openModal("全新的筛选模块", dom('div', {
  2288. childs: [
  2289. dom('p', {
  2290. text: "新的筛选模块支持更多、更灵活的筛选方式,配置方式更加直观,选择器组合方式更加自由,可以实现更高级的批量筛选。"
  2291. }),
  2292. dom('p', {
  2293. text: "但是新版本选择器并非完美,目前还在初步测试中,不能保证稳定性。"
  2294. }),
  2295. dom('p', {
  2296. text:"正在检测...",
  2297. init: el => {
  2298. if (newFilterModuleInstalled) {
  2299. el.innerText = "你已安装新模块,点击启动立刻打开使用新模块";
  2300. } else {
  2301. el.innerText = "新模块是可选附加模块之一,你需要前往页面点击安装,然后刷新页面生效。点击前往安装立刻打开安装页面。";
  2302. }
  2303. }
  2304. }),
  2305. makeButtons([
  2306. {
  2307. text: newFilterModuleInstalled?"启动":"前往安装",
  2308. onclick: async e => {
  2309. hideModal();
  2310. if (newFilterModuleInstalled) {
  2311. hideModal();
  2312. await wait(300);
  2313. unsafeWindow.FoManPlugins.FilterReborn({
  2314. datas,
  2315. info: cfg,
  2316. domHelper:CKTools.domHelper,
  2317. mdi,
  2318. get, getAll, wait, log,
  2319. addStyle, clearStyles,
  2320. alertModal, hideModal,
  2321. checkers: {
  2322. isNearly, isLongAgo, isInvalid,
  2323. isFans, isWhisper, isHardCoreMember,
  2324. isSpecial: d=>d.special === 1,
  2325. isVerified: d=>d.official_verify.type>0,
  2326. isVIP: d => d.vip.vipType === 0,
  2327. isNormalVIP: d => d.vip.vipType === 1,
  2328. isYearVIP: d=>d.vip.vipType!==1&&d.vip.vipType!==0,
  2329. },
  2330. getGroups: () => [...datas.tags],
  2331. select: (uid, status = true) => toggleSwitch(uid, status),
  2332. getSelected: () => refreshChecked().filter(id => !isNaN(id)).map(id => datas.followings[+id]).filter(el => !!el),
  2333. clearSelected: () => {
  2334. datas.checked = [];
  2335. [...getAll(`input.CKFOMAN-data-inforow-toggle[checked]`)].map(el => el.checked = false);
  2336. }
  2337. }).catch(e => {
  2338. alertModal("模块加载失败", "出现了一个错误,不能加载模块。", "确定");
  2339. console.error(e);
  2340. });
  2341. } else {
  2342. // open('about:blank');
  2343. if(!cfg.I_KNOW_WHAT_IM_DOING)alertModal("很抱歉","此模块尚未发布,请等待下个版本更新。","确定");
  2344. }
  2345. }
  2346. },
  2347. {
  2348. text: "取消",
  2349. onclick: e => hideModal()
  2350. }
  2351. ])
  2352. ]
  2353. }))
  2354. }
  2355. const createBlockOrFollowModal = async (isBlock = true) => {
  2356. hideModal();
  2357. await wait(300);
  2358. refreshChecked();
  2359. if (datas.checked.length === 0) {
  2360. if(!cfg.I_KNOW_WHAT_IM_DOING)alertModal("无法继续", "你没有选中任何项,请选中一些项然后再进行操作。", "确认");
  2361. return;
  2362. }
  2363. const ui = {
  2364. action: isBlock ? "拉黑" : "关注",
  2365. title: isBlock ? "批量拉黑" : "批量关注",
  2366. desc: isBlock ? "确认要拉黑的用户列表。<br>他们不会从你的关注中消失。" : "确认要关注的用户列表。<br>重复关注可能导致你变成新粉丝。",
  2367. }
  2368. openModal(ui.title, await makeDom("div", async container => {
  2369. container.appendChild(await makeDom("div", tip => {
  2370. tip.style.fontWeight = "bold";
  2371. tip.innerHTML = ui.desc;
  2372. }))
  2373. container.appendChild(divider());
  2374. container.appendChild(await makeDom("div", async checkedlistdom => {
  2375. checkedlistdom.className = "CKFOMAN-scroll-list";
  2376. checkedlistdom.style.width = "100%";
  2377. checkedlistdom.style.maxHeight = "calc(50vh - 100px)";
  2378. const checkedList = [];
  2379. for (let uid of datas.checked) {
  2380. if (uid in datas.mappings)
  2381. checkedList.push(datas.mappings[uid])
  2382. }
  2383. await renderListTo(checkedlistdom, checkedList, false);
  2384. }))
  2385. container.appendChild(await makeDom("div", async btns => {
  2386. btns.style.display = "flex";
  2387. [
  2388. await makeDom("button", btn => {
  2389. btn.className = "CKFOMAN-toolbar-btns red";
  2390. btn.innerHTML = "确认";
  2391. btn.onclick = async e => {
  2392. if (datas.checked.length === 0)
  2393. if(!cfg.I_KNOW_WHAT_IM_DOING)return alertModal("无需继续", "你没有选中任何项。", "确定");
  2394. const finalList = datas.checked;
  2395. await alertModal("正在" + ui.action, `正在${ui.action}${finalList.length}个关注...`);
  2396. const result = await batchOperateUser(finalList, isBlock?RELE_ACTION.BLOCK:RELE_ACTION.FOLLOW);
  2397. if (result.ok) {
  2398. await alertModal(ui.action + "完成", `${finalList.length}个关注全部${ui.action}成功!`, "确定");
  2399. return createMainWindow(true);
  2400. } else {
  2401. if ("data" in result) {
  2402. if (result.data !== null && "failed_fids" in result.data)
  2403. await alertModal(ui.action + "完成,但部分失败", `尝试${ui.action}了${finalList.length}个关注,但是有${result.data.failed_fids.length}个${ui.action}失败:
  2404. <br>
  2405. <textarea readonly onclick="this.select()">${result.data.failed_fids.join(',')}</textarea>`, "确定");
  2406. else
  2407. await alertModal(ui.action + "失败", `尝试${ui.action}了${finalList.length}个关注但失败了,原因:<br><pre>${result.res}</pre>`, "确定");
  2408. return createMainWindow(true);
  2409. } else {
  2410. await alertModal(ui.action + "失败", `尝试${ui.action}了${finalList.length}个关注但失败了,原因:<br><pre>${result.res}</pre>`, "确定");
  2411. return createMainWindow(true);
  2412. }
  2413. }
  2414. }
  2415. }),
  2416. await makeDom("button", btn => {
  2417. btn.className = "CKFOMAN-toolbar-btns";
  2418. btn.innerHTML = "取消";
  2419. btn.onclick = e => hideModal();
  2420. }),
  2421. ].forEach(el => btns.appendChild(el));
  2422. }))
  2423. }))
  2424. }
  2425. const createOtherSpaceAlert = () => cfg.I_KNOW_WHAT_IM_DOING||alertModal("无法执行操作", "此功能只能在你的个人空间使用,当前是在别人的空间。", "确定");
  2426. const createUnfollowModal = async () => {
  2427. refreshChecked();
  2428. if (datas.checked.length === 0) {
  2429. if(!cfg.I_KNOW_WHAT_IM_DOING)alertModal("取消关注", `你没有勾选任何人,所以无法取关。请勾选后再点击取关按钮。`, "知道了")
  2430. } else
  2431. hideModal();
  2432. await wait(300);
  2433. openModal("取关这些Up?", await makeDom("div", async container => {
  2434. container.appendChild(await makeDom("div", tip => {
  2435. tip.style.color = "red";
  2436. tip.style.fontWeight = "bold";
  2437. tip.innerHTML = `请注意,一旦你确认这个操作,没有任何方法可以撤销!<br>就算你重新关注,也算是新粉丝的哦!`;
  2438. }))
  2439. container.appendChild(await makeDom("div", delaySettings => {
  2440. delaySettings.style.color = "blue";
  2441. delaySettings.style.fontWeight = "bold";
  2442. delaySettings.innerHTML = `操作间隔:<input id="CKFOMAN-form-delay" type="number" step="0.01" value="${datas.settings.batchOperationDelay}" />`;
  2443. }))
  2444. container.appendChild(divider());
  2445. container.appendChild(await makeDom("div", async unfolistdom => {
  2446. unfolistdom.className = "CKFOMAN-scroll-list";
  2447. unfolistdom.style.width = "100%";
  2448. unfolistdom.style.maxHeight = "calc(50vh - 100px)";
  2449. const unfolist = [];
  2450. for (let unfoid of datas.checked) {
  2451. if (unfoid in datas.mappings)
  2452. unfolist.push(datas.mappings[unfoid])
  2453. }
  2454. await renderListTo(unfolistdom, unfolist, false);
  2455. }))
  2456. container.appendChild(await makeDom("div", async btns => {
  2457. btns.style.display = "flex";
  2458. [
  2459. await makeDom("button", btn => {
  2460. btn.className = "CKFOMAN-toolbar-btns red";
  2461. btn.innerHTML = "确认";
  2462. btn.onclick = e => {
  2463. const delayDom = get("#CKFOMAN-form-delay");
  2464. if(delayDom) {
  2465. try{
  2466. let delay = parseFloat(delayDom.value);
  2467. datas.settings.batchOperationDelay = Math.max(delay,0);
  2468. }catch{}
  2469. }
  2470. doUnfollowChecked()
  2471. }
  2472. }),
  2473. await makeDom("button", btn => {
  2474. btn.className = "CKFOMAN-toolbar-btns";
  2475. btn.innerHTML = "取消";
  2476. btn.onclick = e => hideModal();
  2477. }),
  2478. ].forEach(el => btns.appendChild(el));
  2479. }))
  2480. }))
  2481. }
  2482. const applyFilters = async config => {// TODO: pending a code refactor
  2483. setInfoBar(`正在处理 ...`);
  2484. await alertModal("请稍等", "正在应用选择的筛选器...");
  2485. const cfg = {
  2486. clear: config.clear || "0",
  2487. invalid: config.invalid || "-2",
  2488. vip: config.vip || "-2",
  2489. official: config.official || "-2",
  2490. fans: config.fans || "-2",
  2491. groups: config.groups || "-4",
  2492. special: config.special || "-2",
  2493. beforetime: {
  2494. enabled: config.beforetime.enabled || false,
  2495. before: config.beforetime.before || (new Date).getTime()
  2496. },
  2497. aftertime: {
  2498. enabled: config.aftertime.enabled || false,
  2499. after: config.aftertime.after || 0
  2500. }
  2501. };
  2502. if (
  2503. cfg.clear === "0"
  2504. && cfg.invalid === "-2"
  2505. && cfg.vip === "-2"
  2506. && cfg.official === "-2"
  2507. && cfg.fans === "-2"
  2508. && cfg.groups === "-4"
  2509. && cfg.special === "-2"
  2510. && cfg.beforetime.enabled === false
  2511. && cfg.aftertime.enabled === false
  2512. ) {
  2513. resetInfoBar();
  2514. return;
  2515. }
  2516. if (cfg.clear === "0") {
  2517. datas.checked = [];
  2518. }
  2519. const filters = {};
  2520. if (cfg.invalid !== "-2") {
  2521. filters.invalid = cfg.invalid === "1";
  2522. }
  2523. if (cfg.vip !== "-2") {
  2524. filters.vip = cfg.vip;
  2525. }
  2526. if (cfg.official !== "-2") {
  2527. filters.official = cfg.official;
  2528. }
  2529. if (cfg.fans !== "-2") {
  2530. filters.fans = cfg.fans;
  2531. }
  2532. if (cfg.special !== "-2") {
  2533. filters.special = cfg.special;
  2534. }
  2535. if (cfg.groups !== "-4") {
  2536. filters.groups = cfg.groups;
  2537. }
  2538. if (cfg.beforetime.enabled) {
  2539. filters.beforetime = parseInt(cfg.beforetime.before);
  2540. }
  2541. if (cfg.aftertime.enabled) {
  2542. filters.aftertime = parseInt(cfg.aftertime.after);
  2543. }
  2544. let checked = [];
  2545. try {
  2546. userloop: for (let mid in datas.mappings) {
  2547. const uid = parseInt(mid);
  2548. const user = datas.mappings[mid];
  2549. log(uid, user);
  2550. for (let filter in filters) {
  2551. const value = filters[filter];
  2552. switch (filter) {
  2553. case "invalid":
  2554. if (isInvalid(user) !== value) continue userloop;
  2555. break;
  2556. case "vip":
  2557. if (user.vip.vipType != value) continue userloop;
  2558. break;
  2559. case "official":
  2560. if (user.official_verify.type != value) continue userloop;
  2561. break;
  2562. case "fans":
  2563. if (user.attribute == value) continue userloop;
  2564. break;
  2565. case "special":
  2566. if (user.special != value) continue userloop;
  2567. break;
  2568. case "groups":
  2569. switch (value) {
  2570. case "-3":
  2571. if (user.tag !== null) continue userloop;
  2572. break;
  2573. case "-2":
  2574. if (user.tag === null) continue userloop;
  2575. break;
  2576. default:
  2577. if (!((user.tag instanceof Array) && user.tag.includes(parseInt(value)))) continue userloop;
  2578. }
  2579. break;
  2580. case "beforetime":
  2581. if (parseInt(user.mtime + "000") > value) continue userloop;
  2582. break;
  2583. case "aftertime":
  2584. if (parseInt(user.mtime + "000") < value) continue userloop;
  2585. break;
  2586. }
  2587. }
  2588. checked.push(uid);
  2589. if (!datas.checked.includes(uid) && !datas.checked.includes(uid + "")) {
  2590. datas.checked.push(uid);
  2591. }
  2592. }
  2593. setInfoBar("正在将筛选应用到列表...");
  2594. await wait(1);
  2595. datas.followings.forEach(it=>toggleSwitch(it.mid,datas.checked.includes(parseInt(it.mid))));
  2596. setInfoBar("正在按已选中优先排序...");
  2597. await wait(1);
  2598. datas.followings.sort((x, y) => {
  2599. const xint = (datas.checked.includes(x.mid + "") || datas.checked.includes(parseInt(x.mid))) ? 1 : 0;
  2600. const yint = (datas.checked.includes(y.mid + "") || datas.checked.includes(parseInt(y.mid))) ? 1 : 0;
  2601. return yint - xint;
  2602. })
  2603. await renderListTo(get("#CKFOMAN-MAINLIST"),datas.followings,true);
  2604. hideModal();
  2605. } catch (e) {
  2606. alertModal("抱歉", "筛选时出现错误,未能完成筛选。");
  2607. log(e);
  2608. }
  2609. resetInfoBar();
  2610. return checked;
  2611. }
  2612. const createMainWindow = async (forceRefetch = false) => {
  2613. showPanel();
  2614. setInfoBar("正在准备获取关注数据...");
  2615. await createScreen(await makeDom("div", dom => {
  2616. dom.style.position = "fixed";
  2617. dom.style.left = "50%";
  2618. dom.style.top = "50%";
  2619. dom.style.transform = "translate(-50%,-50%)";
  2620. dom.style.textAlign = "center";
  2621. dom.innerHTML = `<h2><i class="mdi mdi-account-search-outline" style="color:cornflowerblue"></i><br>正在获取数据</h2>请稍等片刻,不要关闭窗口。`;
  2622. }));
  2623. if (!(await cacheGroupList())) alertModal("警告", "分组数据获取失败。", "确定");
  2624. return getFollowings(forceRefetch)
  2625. .then(async () => {
  2626. return createScreen(await makeDom("div", async screen => {
  2627. const toolbar = await makeDom("div", async toolbar => {
  2628. toolbar.style.display = "flex";
  2629. toolbar.appendChild(await makeDom("button", btn => {
  2630. btn.className = "CKFOMAN-toolbar-btns";
  2631. btn.innerHTML = '批量操作 <i class="mdi mdi-18px mdi-chevron-down"></i>';
  2632. //btn.style.background = "#e91e63";
  2633. btn.onclick = async e => {
  2634. await openModal("批量操作", await makeDom("div", async container => {
  2635. container.style.alignContent = "stretch";
  2636. [
  2637. datas.isSelf?await makeDom("button", async btn => {
  2638. btn.className = "CKFOMAN-toolbar-btns";
  2639. btn.style.margin = "4px 0";
  2640. btn.innerHTML = '取关选中';
  2641. btn.onclick = () => createUnfollowModal();
  2642. }):null,
  2643. datas.isSelf?await makeDom("button", async btn => {
  2644. btn.className = "CKFOMAN-toolbar-btns";
  2645. btn.style.margin = "4px 0";
  2646. btn.innerHTML = '复制到分组';
  2647. btn.onclick = () => createGroupChangeModal('copy');
  2648. }):null,
  2649. datas.isSelf?await makeDom("button", async btn => {
  2650. btn.className = "CKFOMAN-toolbar-btns";
  2651. btn.style.margin = "4px 0";
  2652. btn.innerHTML = '修改分组';
  2653. btn.onclick = () => createGroupChangeModal('move');
  2654. }):null,
  2655. await makeDom("button", async btn => {
  2656. btn.className = "CKFOMAN-toolbar-btns";
  2657. btn.style.margin = "4px 0";
  2658. btn.innerHTML = '批量拉黑(测试)';
  2659. btn.onclick = () => createBlockOrFollowModal(true);
  2660. }),
  2661. (() => {
  2662. if (!datas.isSelf) {
  2663. return makeDom("button", async btn => {
  2664. btn.className = "CKFOMAN-toolbar-btns";
  2665. btn.style.margin = "4px 0";
  2666. btn.innerHTML = '批量关注(测试)';
  2667. btn.onclick = () => createBlockOrFollowModal(false);
  2668. })
  2669. } else return null;
  2670. })(),
  2671. divider(),
  2672. await makeDom("button", async btn => {
  2673. btn.className = "CKFOMAN-toolbar-btns";
  2674. btn.innerHTML = '返回';
  2675. btn.onclick = () => hideModal();
  2676. }),
  2677. ].forEach(el => el && container.appendChild(el));
  2678. }));
  2679. };
  2680. }))
  2681. toolbar.appendChild(await makeDom("button", btn => {
  2682. btn.className = "CKFOMAN-toolbar-btns";
  2683. btn.innerHTML = '全选';
  2684. btn.onclick = e => {
  2685. setInfoBar("正在处理全选...");
  2686. const all = getAll(".CKFOMAN-data-inforow-toggle");
  2687. if (all) {
  2688. [...all].forEach(it => {
  2689. it.checked = true;
  2690. it.onchange();
  2691. });
  2692. }
  2693. refreshChecked();
  2694. resetInfoBar();
  2695. }
  2696. }))
  2697. toolbar.appendChild(await makeDom("button", btn => {
  2698. btn.className = "CKFOMAN-toolbar-btns";
  2699. btn.innerHTML = '反选';
  2700. btn.onclick = e => {
  2701. setInfoBar("正在处理反选...");
  2702. const all = getAll(".CKFOMAN-data-inforow-toggle");
  2703. if (all) {
  2704. [...all].forEach(it => {
  2705. it.checked = !it.checked;
  2706. it.onchange();
  2707. });
  2708. }
  2709. refreshChecked();
  2710. resetInfoBar();
  2711. }
  2712. }))
  2713. toolbar.appendChild(await makeDom("button", btn => {
  2714. btn.className = "CKFOMAN-toolbar-btns";
  2715. btn.innerHTML = '全不选';
  2716. btn.onclick = e => {
  2717. setInfoBar("正在处理取选...");
  2718. const all = getAll(".CKFOMAN-data-inforow-toggle");
  2719. if (all) {
  2720. [...all].forEach(it => {
  2721. it.checked = false;
  2722. it.onchange();
  2723. });
  2724. }
  2725. refreshChecked();
  2726. resetInfoBar();
  2727. }
  2728. }))
  2729. toolbar.appendChild(await makeDom("button", btn => {
  2730. btn.className = "CKFOMAN-toolbar-btns";
  2731. btn.innerHTML = '间选';
  2732. btn.onclick = e => {
  2733. setInfoBar("正在处理间选...");
  2734. const all = getAll(".CKFOMAN-data-inforow-toggle");
  2735. if (all) {
  2736. let shouldCheck = false;
  2737. for (let el of [...all]) {
  2738. if (el.checked === true) {
  2739. shouldCheck = !shouldCheck;
  2740. } else {
  2741. if (shouldCheck) setToggleStatus(el.getAttribute("data-targetmid"), true);
  2742. }
  2743. }
  2744. }
  2745. resetInfoBar();
  2746. }
  2747. }))
  2748. toolbar.appendChild(await makeDom("button", btn => {
  2749. btn.className = "CKFOMAN-toolbar-btns";
  2750. btn.innerHTML = '筛选 <i class="mdi mdi-18px mdi-chevron-down"></i>';
  2751. btn.onclick = async e => {
  2752. //alertModal("施工中", "此功能尚未实现!", "返回");
  2753. openModal("筛选", await makeDom("div", async container => {
  2754. const filtersid = "CKFOMAN-filters";
  2755. [
  2756. await makeDom("div", async tip => {
  2757. tip.innerHTML = "勾选要生效的筛选器"
  2758. }),
  2759. cfg.enableNewModules?await makeDom("div", async tip => {
  2760. tip.innerHTML = "👉尝鲜新版筛选器";
  2761. tip.style.color = "#00a0e9";
  2762. tip.onclick = () => openNewFilterGuideScreen();
  2763. }):null,
  2764. divider(),
  2765. await makeDom("form", async filters => {
  2766. filters.id = filtersid;
  2767. filters.style.display = "flex";
  2768. filters.style.textAlign = "center";
  2769. filters.style.flexDirection = "column";
  2770. filters.style.flexWrap = "nowrap";
  2771. filters.style.alignContent = "center";
  2772. filters.style.justifyContent = "space-between";
  2773. filters.style.alignItems = "stretch";
  2774. [
  2775. await makeDom("select", async select => {
  2776. select.id = filtersid + "-clear";
  2777. select.name = "val-clear";
  2778. [
  2779. await makeDom("option", opt => {
  2780. opt.value = "0";
  2781. opt.innerHTML = "应用筛选器时取消已选择项目"
  2782. }),
  2783. await makeDom("option", opt => {
  2784. opt.value = "1";
  2785. opt.innerHTML = "应用筛选器时保留已选择项目"
  2786. }),
  2787. ].forEach(s => select.appendChild(s));
  2788. }),
  2789. await makeDom("div", div => div.innerHTML = "+"),
  2790. await makeDom("select", async select => {
  2791. select.id = filtersid + "-invalid";
  2792. select.name = "val-invalid";
  2793. [
  2794. await makeDom("option", opt => {
  2795. opt.value = "-2";
  2796. opt.innerHTML = "不使用注销账户选择器"
  2797. }),
  2798. await makeDom("option", opt => {
  2799. opt.value = "0";
  2800. opt.innerHTML = "正常账户"
  2801. }),
  2802. await makeDom("option", opt => {
  2803. opt.value = "1";
  2804. opt.innerHTML = "已注销账户"
  2805. }),
  2806. ].forEach(s => select.appendChild(s));
  2807. }),
  2808. await makeDom("div", div => div.innerHTML = "+"),
  2809. await makeDom("select", async select => {
  2810. select.id = filtersid + "-special";
  2811. select.name = "val-special";
  2812. [
  2813. await makeDom("option", opt => {
  2814. opt.value = "-2";
  2815. opt.innerHTML = "不使用特别关注选择器"
  2816. }),
  2817. await makeDom("option", opt => {
  2818. opt.value = "0";
  2819. opt.innerHTML = "非特别关注"
  2820. if(!datas.isSelf) opt.disabled = true;
  2821. }),
  2822. await makeDom("option", opt => {
  2823. opt.value = "1";
  2824. opt.innerHTML = "特别关注"
  2825. if(!datas.isSelf) opt.disabled = true;
  2826. }),
  2827. ].forEach(s => select.appendChild(s));
  2828. }),
  2829. await makeDom("div", div => div.innerHTML = "+"),
  2830. await makeDom("select", async select => {
  2831. select.id = filtersid + "-vip";
  2832. select.name = "val-vip";
  2833. [
  2834. await makeDom("option", opt => {
  2835. opt.value = "-2";
  2836. opt.innerHTML = "不使用会员选择器"
  2837. }),
  2838. await makeDom("option", opt => {
  2839. opt.value = "0";
  2840. opt.innerHTML = "没有大会员的用户"
  2841. }),
  2842. await makeDom("option", opt => {
  2843. opt.value = "1";
  2844. opt.innerHTML = "月度大会员用户"
  2845. }),
  2846. await makeDom("option", opt => {
  2847. opt.value = "6";
  2848. opt.innerHTML = "年度大会员用户"
  2849. }),
  2850. ].forEach(s => select.appendChild(s));
  2851. }),
  2852. await makeDom("div", div => div.innerHTML = "+"),
  2853. await makeDom("select", async select => {
  2854. select.id = filtersid + "-official";
  2855. select.name = "val-official";
  2856. [
  2857. await makeDom("option", opt => {
  2858. opt.value = "-2";
  2859. opt.innerHTML = "不使用认证账户选择器"
  2860. }),
  2861. await makeDom("option", opt => {
  2862. opt.value = "0";
  2863. opt.innerHTML = "没有认证的用户"
  2864. }),
  2865. await makeDom("option", opt => {
  2866. opt.value = "1";
  2867. opt.innerHTML = "认证用户"
  2868. }),
  2869. ].forEach(s => select.appendChild(s));
  2870. }),
  2871. await makeDom("div", div => div.innerHTML = "+"),
  2872. await makeDom("select", async select => {
  2873. select.id = filtersid + "-fans";
  2874. select.name = "val-fans";
  2875. [
  2876. await makeDom("option", opt => {
  2877. opt.value = "-2";
  2878. opt.innerHTML = "不使用互粉选择器"
  2879. }),
  2880. await makeDom("option", opt => {
  2881. opt.value = "2";
  2882. opt.innerHTML = "单项关注的用户"
  2883. if(!datas.isSelf) opt.disabled = true;
  2884. }),
  2885. await makeDom("option", opt => {
  2886. opt.value = "6";
  2887. opt.innerHTML = "互粉用户"
  2888. if(!datas.isSelf) opt.disabled = true;
  2889. }),
  2890. ].forEach(s => select.appendChild(s));
  2891. }),
  2892. await makeDom("div", div => div.innerHTML = "+"),
  2893. await makeDom("select", async select => {
  2894. select.id = filtersid + "-groups";
  2895. select.name = "val-groups";
  2896. [
  2897. await makeDom("option", opt => {
  2898. opt.value = "-4";
  2899. opt.innerHTML = "不使用分组选择器"
  2900. }),
  2901. await makeDom("option", opt => {
  2902. opt.value = "-3";
  2903. opt.innerHTML = "没有分组的用户"
  2904. if(!datas.isSelf) opt.disabled = true;
  2905. }),
  2906. await makeDom("option", opt => {
  2907. opt.value = "-2";
  2908. opt.innerHTML = "已有分组的用户"
  2909. if(!datas.isSelf) opt.disabled = true;
  2910. }),
  2911. ].forEach(s => select.appendChild(s));
  2912. if (datas.isSelf && Object.keys(datas.tags).length > 0) {
  2913. select.appendChild(await makeDom("option", opt => {
  2914. opt.innerHTML = "------------";
  2915. opt.disabled = true;
  2916. }));
  2917. for (let tag of Object.values(datas.tags)) {
  2918. select.appendChild(await makeDom("option", opt => {
  2919. opt.innerHTML = tag.name;
  2920. opt.value = tag.tagid;
  2921. }))
  2922. }
  2923. }
  2924. }),
  2925. divider(),
  2926. await makeDom("label", async label => {
  2927. label.setAttribute("for", filtersid + "-beforetime");
  2928. label.innerHTML = "在什么时间前关注:";
  2929. }),
  2930. await makeDom("input", async choose => {
  2931. choose.id = filtersid + "-beforetime";
  2932. choose.name = "val-beforetime";
  2933. choose.setAttribute("type", "datetime-local");
  2934. }),
  2935. divider(),
  2936. await makeDom("label", async label => {
  2937. label.setAttribute("for", filtersid + "-aftertime");
  2938. label.innerHTML = "在什么时间后关注:";
  2939. }),
  2940. await makeDom("input", async choose => {
  2941. choose.id = filtersid + "-aftertime";
  2942. choose.name = "val-aftertime";
  2943. choose.setAttribute("type", "datetime-local");
  2944. }),
  2945. ].forEach(el => filters.appendChild(el));
  2946. }),
  2947. divider(),
  2948. await makeDom("div", async btns => {
  2949. btns.style.display = "flex";
  2950. btns.style.flexDirection = "row";
  2951. btns.style.flexWrap = "nowrap";
  2952. btns.style.alignContent = "stretch";
  2953. btns.style.justifyContent = "space-around";
  2954. btns.style.alignItems = "stretch";
  2955. [
  2956. await makeDom("button", btn => {
  2957. btn.className = "CKFOMAN-toolbar-btns";
  2958. btn.innerHTML = "应用";
  2959. btn.onclick = async () => {
  2960. const form = get("#" + filtersid);
  2961. const config = {
  2962. clear: form['val-clear'].value + "",
  2963. invalid: form['val-invalid'].value + "",
  2964. vip: form['val-vip'].value + "",
  2965. official: form['val-official'].value + "",
  2966. fans: form['val-fans'].value + "",
  2967. groups: form['val-groups'].value + "",
  2968. special: form['val-special'].value + "",
  2969. beforetime: {
  2970. enabled: form['val-beforetime'].value.length > 0,
  2971. before: form['val-beforetime'].valueAsNumber
  2972. },
  2973. aftertime: {
  2974. enabled: form['val-aftertime'].value.length > 0,
  2975. after: form['val-aftertime'].valueAsNumber
  2976. }
  2977. };
  2978. await applyFilters(config);
  2979. hideModal();
  2980. }
  2981. }),
  2982. await makeDom("button", btn => {
  2983. btn.className = "CKFOMAN-toolbar-btns";
  2984. btn.innerHTML = "取消";
  2985. btn.onclick = () => hideModal();
  2986. }),
  2987. ].forEach(el => btns.appendChild(el));
  2988. })
  2989. ].forEach(el => el&&container.appendChild(el));
  2990. }))
  2991. }
  2992. }))
  2993. toolbar.appendChild(await makeDom("button", btn => {
  2994. btn.className = "CKFOMAN-toolbar-btns";
  2995. btn.innerHTML = '排序 <i class="mdi mdi-18px mdi-chevron-down"></i>';
  2996. btn.onclick = async e => {
  2997. openModal("选择排序方式", await makeDom("div", async select => {
  2998. select.style.alignContent = "stretch";
  2999. select.style.flexDirection = "row";
  3000. select.id = "CKFOMAN-sortbtns-container";
  3001. [
  3002. await makeDom("button", btn => {
  3003. btn.className = "CKFOMAN-toolbar-btns CKFOMAN-sortbtns";
  3004. btn.innerHTML = "已选中优先";
  3005. btn.onclick = async e => {
  3006. setInfoBar("正在按已选中优先排序...");
  3007. await alertModal("正在排序...", "请稍等...");
  3008. refreshChecked();
  3009. datas.followings.sort((x, y) => {
  3010. const xint = (datas.checked.includes(x.mid + "") || datas.checked.includes(parseInt(x.mid))) ? 1 : 0;
  3011. const yint = (datas.checked.includes(y.mid + "") || datas.checked.includes(parseInt(y.mid))) ? 1 : 0;
  3012. return yint - xint;
  3013. })
  3014. await renderListTo(get("#CKFOMAN-MAINLIST"),datas.followings,true);
  3015. hideModal();
  3016. }
  3017. }),
  3018. await makeDom("button", btn => {
  3019. btn.className = "CKFOMAN-toolbar-btns CKFOMAN-sortbtns";
  3020. btn.innerHTML = "按最新关注";
  3021. btn.onclick = async e => {
  3022. setInfoBar("正在按最新关注排序...");
  3023. await alertModal("正在排序...", "请稍等...");
  3024. refreshChecked();
  3025. datas.followings.sort((x, y) => parseInt(y.mtime) - parseInt(x.mtime))
  3026. await renderListTo(get("#CKFOMAN-MAINLIST"),datas.followings,true);
  3027. hideModal();
  3028. }
  3029. }),
  3030. await makeDom("button", btn => {
  3031. btn.className = "CKFOMAN-toolbar-btns CKFOMAN-sortbtns";
  3032. btn.innerHTML = "按最早关注";
  3033. btn.onclick = async e => {
  3034. setInfoBar("正在按最早关注排序...");
  3035. await alertModal("正在排序...", "请稍等...");
  3036. refreshChecked();
  3037. datas.followings.sort((x, y) => parseInt(x.mtime) - parseInt(y.mtime))
  3038. await renderListTo(get("#CKFOMAN-MAINLIST"));
  3039. hideModal();
  3040. }
  3041. }),
  3042. await makeDom("button", btn => {
  3043. btn.className = "CKFOMAN-toolbar-btns CKFOMAN-sortbtns";
  3044. btn.innerHTML = "大会员优先";
  3045. btn.onclick = async e => {
  3046. setInfoBar("正在按大会员优先排序...");
  3047. await alertModal("正在排序...", "请稍等...");
  3048. refreshChecked();
  3049. datas.followings.sort((x, y) => parseInt(y.vip.vipType) - parseInt(x.vip.vipType))
  3050. await renderListTo(get("#CKFOMAN-MAINLIST"),datas.followings,true);
  3051. hideModal();
  3052. }
  3053. }),
  3054. await makeDom("button", btn => {
  3055. btn.className = "CKFOMAN-toolbar-btns CKFOMAN-sortbtns";
  3056. btn.innerHTML = "无会员优先";
  3057. btn.onclick = async e => {
  3058. setInfoBar("正在按无会员优先排序...");
  3059. await alertModal("正在排序...", "请稍等...");
  3060. refreshChecked();
  3061. datas.followings.sort((x, y) => parseInt(x.vip.vipType) - parseInt(y.vip.vipType))
  3062. await renderListTo(get("#CKFOMAN-MAINLIST"),datas.followings,true);
  3063. hideModal();
  3064. }
  3065. }),
  3066. await makeDom("button", btn => {
  3067. btn.className = "CKFOMAN-toolbar-btns CKFOMAN-sortbtns";
  3068. btn.innerHTML = "认证优先";
  3069. btn.onclick = async e => {
  3070. setInfoBar("正在按认证优先排序...");
  3071. await alertModal("正在排序...", "请稍等...");
  3072. refreshChecked();
  3073. datas.followings.sort((x, y) => parseInt(y.official_verify.type) - parseInt(x.official_verify.type))
  3074. await renderListTo(get("#CKFOMAN-MAINLIST"),datas.followings,true);
  3075. hideModal();
  3076. }
  3077. }),
  3078. await makeDom("button", btn => {
  3079. btn.className = "CKFOMAN-toolbar-btns CKFOMAN-sortbtns";
  3080. btn.innerHTML = "无认证优先";
  3081. btn.onclick = async e => {
  3082. setInfoBar("正在按无认证优先排序...");
  3083. await alertModal("正在排序...", "请稍等...");
  3084. refreshChecked();
  3085. datas.followings.sort((x, y) => parseInt(x.official_verify.type) - parseInt(y.official_verify.type))
  3086. await renderListTo(get("#CKFOMAN-MAINLIST"),datas.followings,true);
  3087. hideModal();
  3088. }
  3089. }),
  3090. await makeDom("button", btn => {
  3091. btn.className = "CKFOMAN-toolbar-btns CKFOMAN-sortbtns";
  3092. btn.innerHTML = "已注销优先";
  3093. btn.onclick = async e => {
  3094. setInfoBar("正在按已注销优先排序...");
  3095. await alertModal("正在排序...", "请稍等...");
  3096. refreshChecked();
  3097. datas.followings.sort((x, y) => {
  3098. const xint = isInvalid(x) ? 1 : 0;
  3099. const yint = isInvalid(y) ? 1 : 0;
  3100. return yint - xint;
  3101. })
  3102. await renderListTo(get("#CKFOMAN-MAINLIST"),datas.followings,true);
  3103. hideModal();
  3104. }
  3105. }),
  3106. await makeDom("button", btn => {
  3107. btn.className = "CKFOMAN-toolbar-btns CKFOMAN-sortbtns";
  3108. btn.innerHTML = "特别关注优先";
  3109. btn.onclick = async e => {
  3110. setInfoBar("正在按特别关注优先排序...");
  3111. await alertModal("正在排序...", "请稍等...");
  3112. refreshChecked();
  3113. datas.followings.sort((x, y) => parseInt(y.special) - parseInt(x.special))
  3114. await renderListTo(get("#CKFOMAN-MAINLIST"),datas.followings,true);
  3115. hideModal();
  3116. }
  3117. }),
  3118. await makeDom("button", btn => {
  3119. btn.className = "CKFOMAN-toolbar-btns CKFOMAN-sortbtns";
  3120. btn.innerHTML = "互相关注优先";
  3121. btn.onclick = async e => {
  3122. setInfoBar("正在按互相关注优先排序...");
  3123. await alertModal("正在排序...", "请稍等...");
  3124. refreshChecked();
  3125. datas.followings.sort((x, y) => parseInt(y.attribute) - parseInt(x.attribute))
  3126. await renderListTo(get("#CKFOMAN-MAINLIST"),datas.followings,true);
  3127. hideModal();
  3128. }
  3129. }),
  3130. //divider(),
  3131. await makeDom("button", btn => {
  3132. btn.className = "CKFOMAN-toolbar-btns CKFOMAN-sortbtns";
  3133. btn.innerHTML = "不修改 | 取消";
  3134. btn.onclick = e => hideModal();
  3135. })
  3136. ].forEach(el => select.appendChild(el));
  3137. }));
  3138. }
  3139. }))
  3140. toolbar.appendChild(await makeDom("button", btn => {
  3141. btn.className = "CKFOMAN-toolbar-btns";
  3142. btn.innerHTML = '更多 <i class="mdi mdi-18px mdi-chevron-down"></i>';
  3143. btn.onclick = async e => {
  3144. openModal("更多...", await makeDom("div", async select => {
  3145. select.style.alignContent = "stretch";
  3146. [
  3147. await makeDom("button", btn => {
  3148. btn.className = "CKFOMAN-toolbar-btns";
  3149. btn.style.margin = "4px 0";
  3150. btn.innerHTML = "快速选中...";
  3151. btn.onclick = async e => {
  3152. hideModal();
  3153. await wait(300);
  3154. openModal("快速选中", await makeDom("div", async select => {
  3155. select.style.alignContent = "stretch";
  3156. [
  3157. await makeDom("button", btn => {
  3158. btn.className = "CKFOMAN-toolbar-btns";
  3159. btn.style.margin = "4px 0";
  3160. btn.innerHTML = "加选: 悄悄关注用户";
  3161. btn.onclick = async e => {
  3162. setInfoBar("正在处理加选");
  3163. await alertModal("正在处理...", "请稍等...");
  3164. for (let d of datas.followings) {
  3165. if (d.attribut===1||d.isWhisper) {
  3166. toggleSwitch(d.mid, true);
  3167. }
  3168. }
  3169. resetInfoBar();
  3170. hideModal();
  3171. }
  3172. }),
  3173. await makeDom("button", btn => {
  3174. btn.className = "CKFOMAN-toolbar-btns";
  3175. btn.style.margin = "4px 0";
  3176. btn.innerHTML = "加选: 所有已注销用户";
  3177. btn.onclick = async e => {
  3178. setInfoBar("正在处理加选");
  3179. await alertModal("正在处理...", "请稍等...");
  3180. for (let d of datas.followings) {
  3181. if (isInvalid(d)) {
  3182. toggleSwitch(d.mid, true);
  3183. }
  3184. }
  3185. resetInfoBar();
  3186. hideModal();
  3187. }
  3188. }),
  3189. await makeDom("button", btn => {
  3190. btn.className = "CKFOMAN-toolbar-btns";
  3191. btn.style.margin = "4px 0";
  3192. btn.innerHTML = "加选: 所有两年前的关注";
  3193. btn.onclick = async e => {
  3194. setInfoBar("正在处理加选");
  3195. await alertModal("正在处理...", "请稍等...");
  3196. for (let d of datas.followings) {
  3197. if (isLongAgo(d.mtime)) {
  3198. toggleSwitch(d.mid, true);
  3199. }
  3200. }
  3201. resetInfoBar();
  3202. hideModal();
  3203. }
  3204. }),
  3205. await makeDom("button", btn => {
  3206. btn.className = "CKFOMAN-toolbar-btns";
  3207. btn.style.margin = "4px 0";
  3208. btn.innerHTML = "加选: 所有两个月内的关注";
  3209. btn.onclick = async e => {
  3210. setInfoBar("正在处理加选");
  3211. await alertModal("正在处理...", "请稍等...");
  3212. for (let d of datas.followings) {
  3213. if (isNearly(d.mtime)) {
  3214. toggleSwitch(d.mid, true);
  3215. }
  3216. }
  3217. resetInfoBar();
  3218. hideModal();
  3219. }
  3220. }),
  3221. divider(),
  3222. await makeDom("button", btn => {
  3223. btn.className = "CKFOMAN-toolbar-btns";
  3224. btn.style.margin = "4px 0";
  3225. btn.innerHTML = "减选: 悄悄关注";
  3226. btn.onclick = async e => {
  3227. setInfoBar("正在处理减选");
  3228. await alertModal("正在处理...", "请稍等...");
  3229. for (let d of datas.followings) {
  3230. if (d.attribute===1||d.isWhisper) {
  3231. toggleSwitch(d.mid, false);
  3232. }
  3233. }
  3234. resetInfoBar();
  3235. hideModal();
  3236. }
  3237. }),
  3238. await makeDom("button", btn => {
  3239. btn.className = "CKFOMAN-toolbar-btns";
  3240. btn.style.margin = "4px 0";
  3241. btn.innerHTML = "减选: 所有两年前的关注";
  3242. btn.onclick = async e => {
  3243. setInfoBar("正在处理减选");
  3244. await alertModal("正在处理...", "请稍等...");
  3245. for (let d of datas.followings) {
  3246. if (isLongAgo(d.mtime)) {
  3247. toggleSwitch(d.mid, false);
  3248. }
  3249. }
  3250. resetInfoBar();
  3251. hideModal();
  3252. }
  3253. }),
  3254. await makeDom("button", btn => {
  3255. btn.className = "CKFOMAN-toolbar-btns";
  3256. btn.style.margin = "4px 0";
  3257. btn.innerHTML = "减选: 所有两个月内的关注";
  3258. btn.onclick = async e => {
  3259. setInfoBar("正在处理减选");
  3260. await alertModal("正在处理...", "请稍等...");
  3261. for (let d of datas.followings) {
  3262. if (isNearly(d.mtime)) {
  3263. toggleSwitch(d.mid, false);
  3264. }
  3265. }
  3266. resetInfoBar();
  3267. hideModal();
  3268. }
  3269. }),
  3270. await makeDom("button", btn => {
  3271. btn.className = "CKFOMAN-toolbar-btns";
  3272. btn.style.margin = "4px 0";
  3273. btn.innerHTML = "减选: 所有有大会员的关注";
  3274. btn.onclick = async e => {
  3275. setInfoBar("正在处理减选");
  3276. await alertModal("正在处理...", "请稍等...");
  3277. const hasVIP = d => {
  3278. return d.vip.vipType !== 0;
  3279. }
  3280. for (let d of datas.followings) {
  3281. if (hasVIP(d)) {
  3282. toggleSwitch(d.mid, false);
  3283. }
  3284. }
  3285. resetInfoBar();
  3286. hideModal();
  3287. }
  3288. }),
  3289. await makeDom("button", btn => {
  3290. btn.className = "CKFOMAN-toolbar-btns";
  3291. btn.style.margin = "4px 0";
  3292. btn.innerHTML = "减选: 所有认证账号的关注";
  3293. btn.onclick = async e => {
  3294. setInfoBar("正在处理减选");
  3295. await alertModal("正在处理...", "请稍等...");
  3296. const isVerified = d => {
  3297. return d.official_verify.type > 0;
  3298. }
  3299. for (let d of datas.followings) {
  3300. if (isVerified(d)) {
  3301. toggleSwitch(d.mid, false);
  3302. }
  3303. }
  3304. resetInfoBar();
  3305. hideModal();
  3306. }
  3307. }),
  3308. await makeDom("button", btn => {
  3309. btn.className = "CKFOMAN-toolbar-btns";
  3310. btn.style.margin = "4px 0";
  3311. btn.innerHTML = "减选: 所有特别关注的关注";
  3312. btn.onclick = async e => {
  3313. setInfoBar("正在处理减选");
  3314. await alertModal("正在处理...", "请稍等...");
  3315. const isSpecial = d => {
  3316. return d.special === 1;
  3317. }
  3318. for (let d of datas.followings) {
  3319. if (isSpecial(d)) {
  3320. toggleSwitch(d.mid, false);
  3321. }
  3322. }
  3323. resetInfoBar();
  3324. hideModal();
  3325. }
  3326. }),
  3327. await makeDom("button", btn => {
  3328. btn.className = "CKFOMAN-toolbar-btns";
  3329. btn.style.margin = "4px 0";
  3330. btn.innerHTML = "减选: 所有互相关注的关注";
  3331. btn.onclick = async e => {
  3332. setInfoBar("正在处理减选");
  3333. await alertModal("正在处理...", "请稍等...");
  3334. for (let d of datas.followings) {
  3335. if (isFans(d)) {
  3336. toggleSwitch(d.mid, false);
  3337. }
  3338. }
  3339. resetInfoBar();
  3340. hideModal();
  3341. }
  3342. }),
  3343. await makeDom("button", btn => {
  3344. btn.className = "CKFOMAN-toolbar-btns";
  3345. btn.style.margin = "4px 0";
  3346. btn.innerHTML = "减选: 所有有分组的关注";
  3347. btn.onclick = async e => {
  3348. setInfoBar("正在处理减选");
  3349. await alertModal("正在处理...", "请稍等...");
  3350. const hasGroup = d => {
  3351. return d.tag !== null;
  3352. }
  3353. for (let d of datas.followings) {
  3354. if (hasGroup(d)) {
  3355. toggleSwitch(d.mid, false);
  3356. }
  3357. }
  3358. resetInfoBar();
  3359. hideModal();
  3360. }
  3361. }),
  3362. divider(),
  3363. await makeDom("button", btn => {
  3364. btn.className = "CKFOMAN-toolbar-btns";
  3365. btn.style.margin = "4px 0";
  3366. btn.innerHTML = "不修改 | 取消";
  3367. btn.onclick = e => hideModal();
  3368. })
  3369. ].forEach(el => select.appendChild(el));
  3370. }));
  3371. }
  3372. }),
  3373. divider(),
  3374. await makeDom("button", btn => {
  3375. btn.className = "CKFOMAN-toolbar-btns";
  3376. btn.style.margin = "4px 0";
  3377. btn.innerHTML = "管理分组 (增加/删除) (Beta)";
  3378. if (!datas.isSelf) {
  3379. btn.classList.add("grey");
  3380. btn.disabled = true;
  3381. btn.title = "非个人空间,无法操作。";
  3382. btn.onclick = () => createOtherSpaceAlert();
  3383. } else btn.onclick = e => createGroupInfoModal();
  3384. }),
  3385. divider(),
  3386. await makeDom("button", btn => {
  3387. btn.className = "CKFOMAN-toolbar-btns";
  3388. btn.style.margin = "4px 0";
  3389. refreshChecked();
  3390. if (datas.checked.length > 0)
  3391. btn.innerHTML = "导出所有选中的UID列表..."
  3392. else
  3393. btn.innerHTML = "导出所有关注的UID列表...";
  3394. btn.onclick = async e => {
  3395. let list;
  3396. if (datas.checked.length > 0)
  3397. list = datas.checked.join(',');
  3398. else
  3399. list = Object.keys(datas.mappings).join(',');
  3400. let mtitle = "";
  3401. if(await copy(list)){
  3402. mtitle+="✅ 内容已经自动复制到剪贴板, 你可以粘贴到别处";
  3403. }else{
  3404. mtitle+="请单击列表并按Ctrl+C手动复制";
  3405. }
  3406. unsafeWindow.CKFOMAN_EXPORTUIDS = list;
  3407. unsafeWindow.CKFOMAN_EXPORTTOFILE = ()=>{
  3408. download("export_uids.txt",unsafeWindow.CKFOMAN_EXPORTUIDS);
  3409. }
  3410. mtitle+=`,或者:<button class="CKFOMAN-toolbar-btns" onclick="CKFOMAN_EXPORTTOFILE()">保存为文件</button>`
  3411. await alertModal("导出UID", `
  3412. ${mtitle}
  3413. <br>
  3414. <textarea readonly style="width: 400px;" onclick="this.select()" >${list}</textarea>
  3415. `, "确定");
  3416. resetInfoBar();
  3417. }
  3418. }),
  3419. await makeDom("button", btn => {
  3420. btn.className = "CKFOMAN-toolbar-btns";
  3421. btn.style.margin = "4px 0";
  3422. btn.innerHTML = "从UID列表导入关注...";
  3423. if (!datas.isSelf) {
  3424. btn.classList.add("grey");
  3425. btn.disabled = true;
  3426. btn.title = "非个人空间,无法操作。";
  3427. btn.onclick = () => createOtherSpaceAlert();
  3428. } else
  3429. btn.onclick = async e => {
  3430. hideModal();
  3431. await wait(300);
  3432. openModal("导入UID", await makeDom("div", async modaldiv => {
  3433. [
  3434. await makeDom("tip", tip => tip.innerHTML = "请输入导入的UID列表,用英文半角逗号','分割"),
  3435. await makeDom("textarea", input => {
  3436. input.id = "CKFOMAN-import-textarea";
  3437. input.placeholder = "1111111,2222222,3333333..."
  3438. }),
  3439. divider(),
  3440. await makeDom("div", async btns => {
  3441. btns.style.display = "flex";
  3442. [
  3443. await makeDom("button", btn => {
  3444. btn.className = "CKFOMAN-toolbar-btns orange";
  3445. btn.innerHTML = "批量关注";
  3446. btn.onclick = async e => {
  3447. const value = get("#CKFOMAN-import-textarea").value;
  3448. if (value.length === 0) {
  3449. await alertModal("无法导入", "空白数据", "确定");
  3450. return;
  3451. }
  3452. setInfoBar("正在验证导入");
  3453. await alertModal("正在导入", "正在处理刚刚输入的列表,请稍等...");
  3454. const parts = value.split(',');
  3455. const finalList = [];
  3456. const followed = Object.keys(datas.mappings);
  3457. for (let part of parts) {
  3458. if (part.trim().length === 0) {
  3459. log(part, "is empty, skipped");
  3460. } else if (part.trim().match(/[^0-9]/) === null) {
  3461. const int = parseInt(part.trim());
  3462. if (followed.includes(int) || followed.includes(int + "")) {
  3463. log(part, "has already followed, skipped");
  3464. } else if (int <= 0) {
  3465. log(part, "smaller than zero, skipped");
  3466. } else {
  3467. finalList.push(int);
  3468. }
  3469. } else {
  3470. log(part, "is not a number, skipped");
  3471. }
  3472. }
  3473. await alertModal("正在导入", `正在导入${finalList.length}个关注...`);
  3474. const result = await batchOperateUser(finalList, RELE_ACTION.FOLLOW);
  3475. if (result.ok) {
  3476. await alertModal("导入完成", `${finalList.length}个关注全部导入成功!`, "确定");
  3477. return createMainWindow(true);
  3478. } else {
  3479. if ("data" in result) {
  3480. if (result.data !== null && "failed_fids" in result.data)
  3481. await alertModal("导入完成,但部分失败", `尝试导入了${finalList.length}个关注,但是有${result.data.failed_fids.length}个导入失败:
  3482. <br>
  3483. <textarea readonly onclick="this.select()">${result.data.failed_fids.join(',')}</textarea>`, "确定");
  3484. else
  3485. await alertModal("导入失败", `尝试导入了${finalList.length}个关注但失败了,原因:<br><pre>${result.res}</pre>`, "确定");
  3486. return createMainWindow(true);
  3487. } else {
  3488. await alertModal("导入失败", `尝试导入了${finalList.length}个关注但失败了,原因:<br><pre>${result.res}</pre>`, "确定");
  3489. return createMainWindow(true);
  3490. }
  3491. }
  3492. };
  3493. }),
  3494. await makeDom("button", btn => {
  3495. btn.className = "CKFOMAN-toolbar-btns";
  3496. btn.innerHTML = "取消操作";
  3497. btn.onclick = e => hideModal();
  3498. })
  3499. ].forEach(el => btns.appendChild(el));
  3500. })
  3501. ].forEach(el => modaldiv.appendChild(el));
  3502. }));
  3503. }
  3504. }),
  3505. await makeDom("button", btn => {
  3506. btn.className = "CKFOMAN-toolbar-btns";
  3507. btn.style.margin = "4px 0";
  3508. btn.innerHTML = "基于UID列表批量取关...";
  3509. if (!datas.isSelf) {
  3510. btn.classList.add("grey");
  3511. btn.disabled = true;
  3512. btn.title = "非个人空间,无法操作。";
  3513. btn.onclick = () => createOtherSpaceAlert();
  3514. } else
  3515. btn.onclick = async e => {
  3516. hideModal();
  3517. await wait(300);
  3518. openModal("取关UID", await makeDom("div", async modaldiv => {
  3519. [
  3520. await makeDom("tip", tip => tip.innerHTML = "请输入取关的UID列表,用英文半角逗号','分割"),
  3521. await makeDom("textarea", input => {
  3522. input.id = "CKFOMAN-import-textarea";
  3523. input.placeholder = "1111111,2222222,3333333..."
  3524. }),
  3525. divider(),
  3526. await makeDom("div", async btns => {
  3527. btns.style.display = "flex";
  3528. [
  3529. await makeDom("button", btn => {
  3530. btn.className = "CKFOMAN-toolbar-btns orange";
  3531. btn.innerHTML = "批量取关";
  3532. btn.onclick = async e => {
  3533. const value = get("#CKFOMAN-import-textarea").value;
  3534. if (value.length === 0) {
  3535. await alertModal("无法取关", "空白数据", "确定");
  3536. return;
  3537. }
  3538. setInfoBar("正在验证数据");
  3539. await alertModal("正在取关", "正在处理刚刚输入的列表,请稍等...");
  3540. const parts = value.split(',');
  3541. const finalList = [];
  3542. const followed = Object.keys(datas.mappings);
  3543. for (let part of parts) {
  3544. if (part.trim().length === 0) {
  3545. log(part, "is empty, skipped");
  3546. } else if (part.trim().match(/[^0-9]/) === null) {
  3547. const int = parseInt(part.trim());
  3548. if (!followed.includes(int) && !followed.includes(int + "")) {
  3549. log(part, "has not been followed, skipped");
  3550. } else if (int <= 0) {
  3551. log(part, "smaller than zero, skipped");
  3552. } else {
  3553. finalList.push(int);
  3554. }
  3555. } else {
  3556. log(part, "is not a number, skipped");
  3557. }
  3558. }
  3559. await alertModal("正在取关", `正在取消${finalList.length}个关注...`);
  3560. const result = await unfollowUsers(finalList);
  3561. if (result.ok) {
  3562. await alertModal("批量取关完成", `${finalList.length}个关注全部取关成功!`, "确定");
  3563. return createMainWindow(true);
  3564. } else {
  3565. if ("data" in result) {
  3566. if (result.data !== null && "failed_fids" in result.data)
  3567. await alertModal("批量取关完成,但部分失败", `尝试移除了${finalList.length}个关注,但是有${result.data.failed_fids.length}个移除失败:
  3568. <br>
  3569. <textarea readonly onclick="this.select()">${result.data.failed_fids.join(',')}</textarea>`, "确定");
  3570. else
  3571. await alertModal("批量取关失败", `尝试移除了${finalList.length}个关注但失败了,原因:<br><pre>${result.res}</pre>`, "确定");
  3572. return createMainWindow(true);
  3573. } else {
  3574. await alertModal("批量取关失败", `尝试移除了${finalList.length}个关注但失败了,原因:<br><pre>${result.res}</pre>`, "确定");
  3575. return createMainWindow(true);
  3576. }
  3577. }
  3578. };
  3579. }),
  3580. await makeDom("button", btn => {
  3581. btn.className = "CKFOMAN-toolbar-btns";
  3582. btn.innerHTML = "取消操作";
  3583. btn.onclick = e => hideModal();
  3584. })
  3585. ].forEach(el => btns.appendChild(el));
  3586. })
  3587. ].forEach(el => modaldiv.appendChild(el));
  3588. }));
  3589. }
  3590. }),
  3591. divider(),
  3592. await makeDom("button", btn => {
  3593. btn.className = "CKFOMAN-toolbar-btns";
  3594. btn.style.margin = "4px 0";
  3595. btn.innerHTML = "重新载入列表";
  3596. btn.onclick = async e => {
  3597. await alertModal("重新载入列表", "正在重新载入列表。此重载不会重新获取数据。");
  3598. datas.dommappings = {};
  3599. await renderListTo(get("#CKFOMAN-MAINLIST"),datas.followings,false);
  3600. resetInfoBar();
  3601. hideModal();
  3602. }
  3603. }),
  3604. await makeDom("button", btn => {
  3605. btn.className = "CKFOMAN-toolbar-btns";
  3606. btn.style.margin = "4px 0";
  3607. btn.innerHTML = "重新载入数据";
  3608. btn.onclick = async e => {
  3609. await alertModal("重新载入数据", "正在重新载入数据和列表。将会重新获取所有数据。");
  3610. datas.dommappings = {};
  3611. await createMainWindow(true);
  3612. hideModal();
  3613. }
  3614. }),
  3615. await makeDom("div", div => {
  3616. div.style.margin = "4px 0";
  3617. const size = CacheManager.getSize();
  3618. div.innerHTML = "ℹ 本地缓存空间已占用 " + size + " MB。";
  3619. if(size < 1.8){
  3620. div.innerHTML += "无需处理。定期整理缓存可以减少空间占用。";
  3621. }else if (size < 2.5) {
  3622. div.innerHTML += "<b>建议整理缓存。</b>";
  3623. } else {
  3624. div.innerHTML += "<b>建议整理或清理缓存以避免缓存空间超出配额。</b>";
  3625. }
  3626. div.onclick = e => showCacheQuotaModal();
  3627. }),
  3628. await makeDom("button", btn => {
  3629. btn.className = "CKFOMAN-toolbar-btns";
  3630. btn.style.margin = "4px 0";
  3631. btn.innerHTML = "整理缓存";
  3632. btn.onclick = async e => {
  3633. await alertModal("整理缓存", "正在整理缓存并移除额外数据,稍后会重新加载。");
  3634. CacheManager.prune();
  3635. await alertModal("重新载入数据", "正在重新载入数据和列表。");
  3636. datas.dommappings = {};
  3637. await createMainWindow();
  3638. hideModal();
  3639. }
  3640. }),
  3641. await makeDom("button", btn => {
  3642. btn.className = "CKFOMAN-toolbar-btns";
  3643. btn.style.margin = "4px 0";
  3644. btn.innerHTML = "清空缓存";
  3645. btn.onclick = async e => {
  3646. await alertModal("清空全部缓存", "正在清空全部缓存,稍后会自动重新加载所有数据。");
  3647. CacheManager.clean();
  3648. await alertModal("重新载入数据", "正在重新载入数据和列表。将会重新获取所有数据。");
  3649. datas.dommappings = {};
  3650. await createMainWindow(true);
  3651. hideModal();
  3652. }
  3653. }),
  3654. await makeDom("button", btn => {
  3655. btn.className = "CKFOMAN-toolbar-btns";
  3656. btn.style.margin = "4px 0";
  3657. btn.innerHTML = "关于和反馈";
  3658. btn.onclick = async e => {
  3659. await alertModal("关于 “关注管理器 FoMan”", (await makeDom("div", async div => {
  3660. div.style.textAlign = "left";
  3661. div.style.width = "400px";
  3662. [
  3663. document.createElement("br"),
  3664. await makeDom("i", i => {
  3665. i.className = "mdi mdi-48px mdi-broom"
  3666. i.style.color = "#0091ea";
  3667. i.style.textAlign = "center";
  3668. i.style.display = "block";
  3669. i.style.width = "fit-content";
  3670. i.style.margin = "0 auto";
  3671. }),
  3672. document.createElement("br"),
  3673. await makeDom("p", span =>
  3674. span.innerHTML = `版本: v${cfg.VERSION}<br>`
  3675. + `License: GPLv3<br>`
  3676. + `作者: CKylinMC`
  3677. ),
  3678. await makeDom("p", span =>
  3679. span.innerHTML = `脚本首页: <a href="https://gf.qytechs.cn/zh-CN/scripts/428895">GreasyFork</a> | <a href="https://github.com/CKylinMC/UserJS">Github</a>`
  3680. ),
  3681. divider(),
  3682. await makeDom("p", span =>
  3683. span.innerHTML = `如果出现问题,请前往GreasyFork反馈区或Github Issues进行反馈,如果好用,还请给我一个好评!十分感谢!`
  3684. ),
  3685. document.createElement("br"),
  3686. ].forEach(el => div.appendChild(el));
  3687. })).outerHTML, "确定");
  3688. resetInfoBar();
  3689. }
  3690. }),
  3691. divider(),
  3692. await makeDom("button", btn => {
  3693. btn.className = "CKFOMAN-toolbar-btns";
  3694. btn.style.margin = "4px 0";
  3695. btn.innerHTML = "返回";
  3696. btn.onclick = e => hideModal();
  3697. })
  3698. ].forEach(el => select.appendChild(el));
  3699. }));
  3700. }
  3701. }))
  3702. });
  3703. const list = await makeDom("div", async list => {
  3704. list.className = "CKFOMAN-scroll-list";
  3705. list.id = "CKFOMAN-MAINLIST";
  3706. await renderListTo(list,datas.followings,!forceRefetch);
  3707. })
  3708. screen.appendChild(toolbar);
  3709. screen.appendChild(list);
  3710. }));
  3711. })
  3712. .catch(async (e) => {
  3713. log(e);
  3714. setInfoBar();
  3715. let errtitle = "获取数据失败";
  3716. let errdesc = "请尝试刷新页面重试";
  3717. log(datas.fetchstat);
  3718. switch(datas.fetchstat){
  3719. case "GUEST-LIMIT":
  3720. errtitle = "访客限制";
  3721. errdesc = "由于访客限制,获取数据失败。"
  3722. break;
  3723. case "PERMS-DENIED":
  3724. errtitle = "无权查看";
  3725. errdesc = "由于当前空间主人已设置访客不可见,因此无法查看到任何信息。"
  3726. break;
  3727. }
  3728. createScreen(await makeDom("div", dom => {
  3729. dom.style.position = "fixed";
  3730. dom.style.left = "50%";
  3731. dom.style.top = "50%";
  3732. dom.style.transform = "translate(-50%,-50%)";
  3733. dom.style.textAlign = "center";
  3734. dom.innerHTML = `<h2><i class="mdi mdi-alert-remove" style="color:orangered;font-size: xx-large"></i><br>${errtitle}</h2>${errdesc}`;
  3735. }));
  3736. })
  3737. }
  3738. const setToggleStatus = (mid, status = false, operateDom = true) => {
  3739. if (operateDom) {
  3740. const selection = getAll(`input.CKFOMAN-data-inforow-toggle[data-targetmid="${mid}"]`);
  3741. if (selection) {
  3742. for (let el of selection) {
  3743. el.checked = status;
  3744. }
  3745. }
  3746. }
  3747. if (status) {
  3748. if (!datas.checked.includes(mid + "") && !datas.checked.includes(parseInt(mid)))
  3749. datas.checked.push(mid);
  3750. } else {
  3751. if (datas.checked.includes(mid + "")) datas.checked.splice(datas.checked.indexOf(mid + ""), 1);
  3752. else if (datas.checked.includes(parseInt(mid))) datas.checked.splice(datas.checked.indexOf(parseInt(mid)), 1);
  3753. }
  3754. resetInfoBar();
  3755. }
  3756. const renderListTo = async (dom, datalist = datas.followings, cacheAndreuse = false) => {
  3757. setInfoBar("正在渲染列表...");
  3758. await wait(1);
  3759. const isMainList = cacheAndreuse||datalist===datas.followings;
  3760. dom.innerHTML = '';
  3761. const getDomForData = async it=>{
  3762. if(cacheAndreuse&&(datas.dommappings[it.mid+""]&& datas.dommappings[it.mid+""] instanceof HTMLElement)) return datas.dommappings[it.mid+""];
  3763. return upinfoline(it);
  3764. }
  3765. for (let it of datalist) {
  3766. const upinfolinedom = await getDomForData(it);
  3767. dom.appendChild(upinfolinedom);
  3768. if(isMainList) datas.dommappings[it.mid+""] = upinfolinedom;
  3769. }
  3770. resetInfoBar();
  3771. }
  3772. const renderTagListTo = async (dom,selectedId=[],cb = ()=>{},inManager = true) => {
  3773. setInfoBar("正在渲染列表...");
  3774. await wait(100);
  3775. dom.innerHTML = '';
  3776. for (let it of Object.values(datas.tags)) {
  3777. log(it);
  3778. dom.appendChild(await taginfoline(it,cb,selectedId.includes(it.tagid),inManager,inManager));
  3779. }
  3780. resetInfoBar();
  3781. }
  3782. const createScreen = async (content) => {
  3783. getContainer().innerHTML = '';
  3784. getContainer().appendChild(content);
  3785. }
  3786.  
  3787. const callAlertWindow = () => {
  3788. cfg.closedByBlocker++;
  3789. if (cfg.I_KNOW_WHAT_IM_DOING) return hideModal();
  3790. cfg.disableCloseModalFromBlockWindow = true;
  3791. const waitTimer = cfg.debug ? 10 : 5;
  3792. alertModal("等一下,这不是正确的关闭方式!",
  3793. `点击空白处可以关闭弹窗,但是有些窗口下这样可能会导致未知问题,<b>请尽量减少使用此方式关闭弹窗。</b>${cfg.debug ? "<br><br><i>修改脚本第53行附近的'I_KNOW_WHAT_IM_DOING:false'的false为true可以永久阻止此弹窗出现直到下一次更新。</i>" : ""}<br><br>此消息每页面只会显示一次,此窗口 ${waitTimer} 秒后自动关闭。<br><progress value=0 max=100 style="width: 100%;height: 4px" id='CKFOMAN-TIMERPROGRESS'></progress>`);
  3794. wait(10).then(async () => {
  3795. await CKTools.waitForDom('#CKFOMAN-TIMERPROGRESS');
  3796. const interval = setInterval(() => {
  3797. const pg = CKTools.get('#CKFOMAN-TIMERPROGRESS');
  3798. if (!pg) return (log('pg not found',pg??null),clearInterval(interval));
  3799. pg.value = pg.value + (cfg.debug?1:2);
  3800. if(pg>100) return (log('pg is full',pg??null),clearInterval(interval));
  3801. },100);
  3802. });
  3803. wait((waitTimer * 1000)+100).then(() => {
  3804. cfg.disableCloseModalFromBlockWindow = false;
  3805. hideModal();
  3806. });
  3807. }
  3808.  
  3809. const closeModalFromBlockWindow = () => {
  3810. if (cfg.disableCloseModalFromBlockWindow) return;
  3811. if (!cfg.closedByBlocker) {
  3812. cfg.closedByBlocker = 1;
  3813. } else if (cfg.closedByBlocker == 3) {
  3814. callAlertWindow();
  3815. } else {
  3816. cfg.closedByBlocker++;
  3817. closeModal();
  3818. }
  3819. }
  3820.  
  3821. const blockWindow = (block = true) => {
  3822. addStyle(`
  3823. #CKFOMAN-blockWindow{
  3824. z-index: 99005;
  3825. display: block;
  3826. background: #00000080;
  3827. opacity: 0;
  3828. transition: all .3s;
  3829. position: fixed;
  3830. left: 0;
  3831. top: 0;
  3832. width: 100vw;
  3833. height: 100vh;
  3834. }
  3835. #CKFOMAN-blockWindow.hide{
  3836. pointer-events: none;
  3837. opacity: 0;
  3838. }
  3839. #CKFOMAN-blockWindow.show{
  3840. opacity: 1;
  3841. }
  3842. `, "CKFOMAN-blockWindow-css", "unique");
  3843. let dom = get("#CKFOMAN-blockWindow");
  3844. if (!dom) {
  3845. dom = document.createElement("div");
  3846. dom.id = "CKFOMAN-blockWindow";
  3847. dom.className = "hide";
  3848. document.body.appendChild(dom);
  3849. }
  3850. dom.onclick = e => closeModalFromBlockWindow();
  3851. datas.preventUserCard = block;
  3852. if (block) {
  3853. dom.className = "show";
  3854. } else {
  3855. dom.className = "hide";
  3856. }
  3857. }
  3858.  
  3859. const injectSideBtn = () => {
  3860. addStyle(`
  3861. #CKFOMAN-floatbtn{
  3862. box-sizing: border-box;
  3863. z-index: 9999;
  3864. position: fixed;
  3865. left: -15px;
  3866. width: 30px;
  3867. height: 30px;
  3868. background: black;
  3869. opacity: 0.8;
  3870. color: white;
  3871. cursor: pointer;
  3872. border-radius: 50%;
  3873. text-align: right;
  3874. line-height: 24px;
  3875. border: solid 3px #00000000;
  3876. transition: opacity .3s 1s, background .3s, color .3s, left .3s, border .3s;
  3877. top: 120px;
  3878. top: 30vh;
  3879. }
  3880. #CKFOMAN-floatbtn::after,#CKFOMAN-floatbtn::before{
  3881. z-index: 9990;
  3882. content: "关注管理器";
  3883. pointer-events: none;
  3884. position: fixed;
  3885. left: -20px;
  3886. height: 30px;
  3887. background: black;
  3888. opacity: 0;
  3889. color: white;
  3890. cursor: pointer;
  3891. border-radius: 8px;
  3892. padding: 0 12px;
  3893. text-align: right;
  3894. line-height: 30px;
  3895. transition: all .3s;
  3896. top: 123px;
  3897. top: 30vh;
  3898. }
  3899. #CKFOMAN-floatbtn::after{
  3900. content: "← 关注管理器";
  3901. animation:CKFOMAN-tipsOut forwards 5s 3.5s;
  3902. }
  3903. #CKFOMAN-floatbtn:hover::before{
  3904. left: 30px;
  3905. opacity: 1;
  3906. }
  3907. #CKFOMAN-floatbtn:hover{
  3908. border: solid 3px black;
  3909. transition: opacity .3s 0s, background .3s, color .3s, left .3s, border .3s;
  3910. background: white;
  3911. color: black;
  3912. opacity: 1;
  3913. left: -5px;
  3914. }
  3915. #CKFOMAN-floatbtn.hide{
  3916. left: -40px;
  3917. }
  3918. @keyframes CKFOMAN-tipsOut{
  3919. 5%,95%{
  3920. opacity: 1;
  3921. left: 20px;
  3922. }
  3923. 0%,100%{
  3924. left: -20px;
  3925. opacity: 0;
  3926. }
  3927. }
  3928. `, "CKFOMAN-floatbtn-css", "unique");
  3929.  
  3930. const toggle = document.createElement("div");
  3931. toggle.id = "CKFOMAN-floatbtn";
  3932. toggle.innerHTML = `<i class="mdi mdi-18px mdi-wrench" style="display: inline-block;transform: rotateY(180deg) translateX(3px);"></i>`;
  3933. toggle.onclick = () => createMainWindow();
  3934. document.body.appendChild(toggle);
  3935. }
  3936.  
  3937. const startInject = () => {
  3938. if(!unsafeWindow.FoManPlugins){
  3939. unsafeWindow.FoManPlugins = {}
  3940. }
  3941. initModal();
  3942. // unsafeWindow.addEventListener("message", event => {
  3943. // if (!event.data) return;
  3944. // if (!(event.data instanceof String)) return;
  3945. // if (event.data.startsWith("CKFOMANSTATUSCHANGES|")) {
  3946. // log(event.data)
  3947. // const parts = event.data.split("|");
  3948. // setToggleStatus(parts[1], parts[2] === "1");
  3949. // }
  3950. // })
  3951. injectSideBtn();
  3952. if (cfg.debug) {
  3953. unsafeWindow.CKFOMAN_DBG = {
  3954. cfg, datas
  3955. }
  3956. }
  3957. unsafeWindow.openFollowManager = forceRefetch=>createMainWindow(forceRefetch);
  3958. };
  3959.  
  3960. startInject();
  3961. })();

QingJ © 2025

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