智友邦论坛增强

自动签到、自动回复、自动无缝翻页、回到顶部(右键点击两侧空白处)、清理置顶帖子、简化附件兑换/下载、清理帖子标题〖XXX〗【XXX】文字

安装此脚本?
作者推荐脚本

您可能也喜欢智友邦论坛美化

安装此脚本
  1. // ==UserScript==
  2. // @name 智友邦论坛增强
  3. // @version 1.1.9
  4. // @author X.I.U
  5. // @description 自动签到、自动回复、自动无缝翻页、回到顶部(右键点击两侧空白处)、清理置顶帖子、简化附件兑换/下载、清理帖子标题〖XXX〗【XXX】文字
  6. // @icon http://bbs.zhiyoo.net/favicon.ico
  7. // @match *://bbs.zhiyoo.net/*
  8. // @match *://www.zhiyoo.net/search.php*
  9. // @grant GM_xmlhttpRequest
  10. // @grant GM_registerMenuCommand
  11. // @grant GM_unregisterMenuCommand
  12. // @grant GM_openInTab
  13. // @grant GM_getValue
  14. // @grant GM_setValue
  15. // @grant GM_notification
  16. // @license GPL-3.0 License
  17. // @run-at document-end
  18. // @namespace https://gf.qytechs.cn/scripts/412362
  19. // @supportURL https://github.com/XIU2/UserScript
  20. // @homepageURL https://github.com/XIU2/UserScript
  21. // ==/UserScript==
  22.  
  23. (function() {
  24. 'use strict';
  25. var menu_ALL = [
  26. ['menu_autoReply', '自动回复', '自动回复', true],
  27. ['menu_pageLoading', '自动无缝翻页', '自动无缝翻页', true],
  28. ['menu_backToTop', '回到顶部(右键点击两侧空白处)', '回到顶部', true],
  29. ['menu_cleanTopPost', '清理置顶帖子', '清理置顶帖子', true],
  30. ['menu_cleanPostTitle', '清理帖子标题开头〖〗【】文字', '清理帖子标题开头〖〗【】文字', true],
  31. ['menu_qianDaoRedirectURL', '当前页面设为签到后重定向地址', '已设置当前页面为签到后重定向地址', 'http://bbs.zhiyoo.net/forum.php?mod=forumdisplay&fid=42&filter=author&orderby=dateline']
  32. ], menu_ID = [];
  33. for (let i=0;i<menu_ALL.length;i++){ // 如果读取到的值为 null 就写入默认值
  34. if (GM_getValue(menu_ALL[i][0]) == null){GM_setValue(menu_ALL[i][0], menu_ALL[i][3])};
  35. }
  36. registerMenuCommand();
  37.  
  38. // 注册(不可用)脚本菜单
  39. function registerMenuCommand() {
  40. if (menu_ID.length > menu_ALL.length){ // 如果菜单ID数组多于菜单数组,说明不是首次添加菜单,需要卸载所有脚本菜单
  41. for (let i=0;i<menu_ID.length;i++){
  42. GM_unregisterMenuCommand(menu_ID[i]);
  43. }
  44. }
  45. for (let i=0;i<menu_ALL.length;i++){ // 循环注册(不可用)脚本菜单
  46. menu_ALL[i][3] = GM_getValue(menu_ALL[i][0]);
  47. if (menu_ALL[i][0] == 'menu_qianDaoRedirectURL') {
  48. menu_ID[i] = GM_registerMenuCommand(`#️⃣ ${menu_ALL[i][1]}`, function(){GM_setValue(`${menu_ALL[i][0]}`, location.href);GM_notification({text: `${menu_ALL[i][2]}`, timeout: 3000});})
  49. } else {
  50. menu_ID[i] = GM_registerMenuCommand(`${menu_ALL[i][3]?'✅':'❌'} ${menu_ALL[i][1]}`, function(){menu_switch(`${menu_ALL[i][3]}`,`${menu_ALL[i][0]}`,`${menu_ALL[i][2]}`)});
  51. }
  52. }
  53. menu_ID[menu_ID.length] = GM_registerMenuCommand('💬 反馈 & 建议', function () {window.GM_openInTab('https://github.com/XIU2/UserScript#xiu2userscript', {active: true,insert: true,setParent: true});window.GM_openInTab('https://gf.qytechs.cn/zh-CN/scripts/412362/feedback', {active: true,insert: true,setParent: true});});
  54. }
  55.  
  56. // 菜单开关
  57. function menu_switch(menu_status, Name, Tips) {
  58. if (menu_status == 'true'){
  59. GM_setValue(`${Name}`, false);
  60. GM_notification({text: `已关闭 [${Tips}] 功能\n(点击刷新网页后生效)`, timeout: 3500, onclick: function(){location.reload();}});
  61. }else{
  62. GM_setValue(`${Name}`, true);
  63. GM_notification({text: `已开启 [${Tips}] 功能\n(点击刷新网页后生效)`, timeout: 3500, onclick: function(){location.reload();}});
  64. }
  65. registerMenuCommand(); // 重新注册(不可用)脚本菜单
  66. };
  67.  
  68. // 返回菜单值
  69. function menu_value(menuName) {
  70. for (let menu of menu_ALL) {
  71. if (menu[0] == menuName) {
  72. return menu[3]
  73. }
  74. }
  75. }
  76.  
  77.  
  78. // 随机回复帖子的内容
  79. var replyList = [
  80. "感谢楼主分享的内容!",
  81. "感谢分享!给你点赞!",
  82. "感谢分享!论坛因你更精彩!",
  83. "看看隐藏内容是什么!谢谢!",
  84. "先下载看看好不好用!",
  85. "楼主一生平安!好人一生平安!",
  86. "搞机上智友提问下资源!",
  87. "马克!智友邦你搞机!",
  88. "你说的观点我也很支持!",
  89. "楼主太棒了!我先下为敬!",
  90. "给楼主点赞,希望继续分享!",
  91. "感谢智友帮论坛,感谢LZ热心分享!",
  92. "感谢楼主分享优质内容,希望继续努力!",
  93. "下载试用一下,如果用着不错就给楼主顶贴!",
  94. "这么好的东西!感谢楼主分享!感谢智友帮论坛!",
  95. "希望楼主继续分享更多好用的东西!谢谢!",
  96. "看到楼主这么努力分享,我只能顶个贴感谢一下了!",
  97. "好东西,拿走了,临走顶个贴感谢一下楼主!",
  98. "这就非常给力了!感谢分享!",
  99. "厉害了!先收藏,再回复!谢谢!"
  100. ];
  101.  
  102. // 帖子数量,避免重复清理帖子列表中帖子标题开头的〖XXX〗【XXX】文字,用于提高效率
  103. var postNum = 0;
  104.  
  105. // 检查是否登陆
  106. var loginStatus = false;
  107. if (document.querySelector('.Quater_user.logined')){
  108. loginStatus = true;
  109. }
  110.  
  111. // 默认 ID 为 0
  112. var curSite = {SiteTypeID: 0};
  113.  
  114. // 自动翻页规则
  115. let DBSite = {
  116. forumdisplay: {
  117. SiteTypeID: 1,
  118. pager: {
  119. nextLink: '//a[@class="nxt"][@href]',
  120. pageElement: 'css;table#threadlisttableid > tbody[id^="normalthread_"]',
  121. HT_insert: ['css;table#threadlisttableid', 2],
  122. replaceE: 'css;div.pg'
  123. }
  124. },
  125. search: {
  126. SiteTypeID: 2,
  127. pager: {
  128. nextLink: '//a[@class="nxt"][@href]',
  129. pageElement: 'css;div#threadlist > ul',
  130. HT_insert: ['css;div#threadlist', 2],
  131. replaceE: 'css;div.pg'
  132. }
  133. }
  134. };
  135.  
  136. // 用于脚本内部判断当前 URL 类型
  137. let SiteType = {
  138. FORUMDISPLAY: DBSite.forumdisplay.SiteTypeID, // 各板块帖子列表
  139. SEARCH: DBSite.search.SiteTypeID // 搜索结果列表
  140. };
  141.  
  142. var attachmentHrefTime = 0;
  143. curSite.pageUrl = ""; // 下一页URL
  144.  
  145. var patt_thread = /\/thread-\d+-\d+\-\d+.html/, // 匹配 /thread-XXX-X-X.html 帖子正则表达式
  146. patt_search = /\/thread-\d+-\d+\-\d+.html/, // 匹配搜索结果列表正则表达式
  147. patt_posttitle = /^〖.+〗(:)?|^【.+】(:)?/, // 匹配帖子标题中的〖XXX〗【XXX】正则表达式
  148. patt_attachment_href = /(?<=\\').+(?=\\')/
  149.  
  150. if (location.pathname === '/plugin.php'){
  151. switch(getQueryVariable("id"))
  152. {
  153. case 'dsu_paulsign:sign': // 被重定向到签到页面
  154. qiandao(); // 自动签到
  155. break;
  156. case 'piaobo_attachment': // 兑换附件后的提示页面
  157. attachmentBack(); // 立即返回帖子
  158. break;
  159. case 'threed_attach:downld': // 附件下载页面
  160. goPan(); // 跳转至网盘页
  161. break;
  162. }
  163. }else if(location.pathname === '/forum.php'){
  164. switch(getQueryVariable("mod"))
  165. {
  166. case 'viewthread': // 浏览帖子内容
  167. showHide(); // 先看看是否有隐藏内容,如果已显示则定位到隐藏内容区域,如果没有隐藏内容,则啥都不干
  168. autoReply(); // 自动回复(有隐藏内容才会回复),回复过就定位到底部(隐藏内容区域)
  169. var attachmentHref_Interval = setInterval(attachmentHref,100); // 兑换附件按钮改为直链(不再弹出确认提示框)
  170. break;
  171. case 'forumdisplay': // 浏览帖子列表
  172. curSite = DBSite.forumdisplay; // 帖子列表页(自动翻页)
  173. cleanTop(); // 清理置顶帖子
  174. cleanPostTitle(); // 清理帖子列表中帖子标题开头的〖XXX〗【XXX】文字
  175. pageLoading(); // 自动无缝翻页
  176. break;
  177. }
  178. backToTop(); // 回到顶部(右键点击两侧空白处)
  179. }else if(location.pathname === '/search.php'){
  180. curSite = DBSite.search; // 搜索结果列表页(自动翻页)
  181. pageLoading(); // 自动无缝翻页
  182. }else if (patt_thread.test(location.pathname)){ // 对于 /thread-XXX-X-X.html 这种帖子页面也和上面一样
  183. showHide();
  184. autoReply();
  185. }
  186.  
  187.  
  188. // 自动签到
  189. function qiandao(){
  190. if (loginStatus){
  191. if(document.getElementById('yl'))
  192. {
  193. document.getElementById('yl').click();
  194. document.querySelector('td.tr3.tac div a').click();
  195. }
  196. setTimeout(location.href=menu_value('menu_qianDaoRedirectURL'), 2000); // 跳转到指定URL
  197. }
  198. }
  199.  
  200.  
  201. // 自动回复
  202. function autoReply(){
  203. if (!menu_value('menu_autoReply')) return
  204. if (loginStatus){
  205. // 存在隐藏内容,自动回复
  206. if (document.getElementsByClassName("showhide").length == 0){
  207. writeReply();
  208. // 如果使用了我的 [智友邦美化] 脚本,则定位至底部,反之定位至隐藏内容区域
  209. if (document.getElementById("fastpostmessage").offsetParent == null){
  210. setTimeout(function(){window.scrollTo(0,99999999)}, 1000);
  211. }else{
  212. setTimeout(function(){window.scrollTo(0,document.querySelector('.showhide').offsetTop)}, 1000);
  213. }
  214. }
  215. }
  216. }
  217.  
  218.  
  219. // 写入自动回复内容
  220. function writeReply(){
  221. let textarea = document.getElementById("fastpostmessage");
  222. if (textarea){
  223. // 随机写入回复内容
  224. textarea.value = textarea.value + replyList[Math.floor((Math.random()*replyList.length))] + replyList[Math.floor((Math.random()*replyList.length))];
  225. //console.log(`${textarea.value}`)
  226. let fastpostsubmit = document.getElementById("fastpostsubmit");
  227. if (fastpostsubmit){
  228. setTimeout(function(){fastpostsubmit.click()}, 200);
  229. }
  230. }
  231. }
  232.  
  233.  
  234. // 定位到隐藏内容区域
  235. function showHide(){
  236. if (loginStatus){
  237. // 如果已显示隐藏内容,则定位到隐藏内容区域
  238. // 如果没有发现已显示隐藏内容,就不定位了
  239. if (document.getElementsByClassName("showhide").length > 0){
  240. // 如果使用了我的 [智友邦美化] 脚本,则定位至底部,反之定位至隐藏内容区域
  241. if (document.getElementById("fastpostmessage").offsetParent == null){
  242. setTimeout(function(){window.scrollTo(0,99999999)}, 1000);
  243. }else{
  244. setTimeout(function(){window.scrollTo(0,document.querySelector('.showhide').offsetTop)}, 1000);
  245. }
  246. }
  247. }
  248. }
  249.  
  250.  
  251. // 回到顶部(右键点击空白处)
  252. function backToTop() {
  253. if (!menu_value('menu_backToTop')) return
  254. document.getElementById("nv_forum").oncontextmenu = function(event){
  255. if (event.target==this) {
  256. event.preventDefault();
  257. window.scrollTo(0,0)
  258. }
  259. }
  260. }
  261.  
  262.  
  263. // 清理置顶帖子
  264. function cleanTop(){
  265. if (!menu_value('menu_cleanTopPost')) return
  266. let showhide = document.querySelectorAll("a.showhide.y");
  267. if (showhide.length > 0){
  268. showhide.forEach(el=>el.click());
  269. }
  270. }
  271.  
  272.  
  273. // 兑换附件后立即返回
  274. function attachmentBack() {
  275. let attachmentback = document.querySelector('#messagetext p.alert_btnleft a');
  276. if (attachmentback){
  277. attachmentback.click();
  278. }
  279. }
  280.  
  281.  
  282. // 附件下载页直接跳转至网盘
  283. function goPan() {
  284. let gopan = document.querySelector('.threed_panbox .panframe .pan_left p a');
  285. if (gopan){
  286. location.href=gopan.href;
  287. }
  288. }
  289.  
  290.  
  291. // 兑换附件按钮改为直链(不再弹出确认提示框)
  292. function attachmentHref() {
  293. attachmentHrefTime += 1; // 计算该函数执行次数
  294. let attachmenthref = document.querySelector('.tab_button .button a');
  295. if (attachmenthref && attachmenthref.href == "javascript:;"){
  296. let attachmenthref_href = attachmenthref.onclick.toString();
  297. attachmenthref.href = attachmenthref_href.match(patt_attachment_href)[0];
  298. attachmenthref.onclick = null;
  299. }
  300. if (attachmentHrefTime == 50 || document.getElementsByClassName("showhide").length > 0){ // 当该函数执行超过50次(5秒),或没有隐藏内容时停止定时执行
  301. clearInterval(attachmentHref_Interval)
  302. }
  303. }
  304.  
  305.  
  306. // 清理帖子列表中帖子标题开头的〖XXX〗【XXX】文字
  307. function cleanPostTitle(){
  308. if (!menu_value('menu_cleanPostTitle')) return
  309. let cleanposttitle = document.querySelectorAll("a.s.xst");
  310. if (cleanposttitle.length > 0){
  311. for(let num = postNum;num<cleanposttitle.length;num++){
  312. cleanposttitle[num].innerText = cleanposttitle[num].innerText.replace(patt_posttitle, ``);
  313. postNum += 1;
  314. }
  315. }
  316. }
  317.  
  318.  
  319. // 自动无缝翻页
  320. function pageLoading() {
  321. if (!menu_value('menu_pageLoading')) return
  322. if (curSite.SiteTypeID > 0){
  323. windowScroll(function (direction, e) {
  324. if (direction === 'down') { // 下滑才准备翻页
  325. let scrollTop = document.documentElement.scrollTop || window.pageYOffset || document.body.scrollTop;
  326. let scrollDelta = 666;
  327. if (document.documentElement.scrollHeight <= document.documentElement.clientHeight + scrollTop + scrollDelta) {
  328. ShowPager.loadMorePage();
  329. }
  330. }
  331. });
  332. }
  333. }
  334.  
  335.  
  336. // 滚动条事件
  337. function windowScroll(fn1) {
  338. var beforeScrollTop = document.documentElement.scrollTop,
  339. fn = fn1 || function () {};
  340. setTimeout(function () { // 延时执行,避免刚载入到页面就触发翻页事件
  341. window.addEventListener('scroll', function (e) {
  342. var afterScrollTop = document.documentElement.scrollTop,
  343. delta = afterScrollTop - beforeScrollTop;
  344. if (delta == 0) return false;
  345. fn(delta > 0 ? 'down' : 'up', e);
  346. beforeScrollTop = afterScrollTop;
  347. }, false);
  348. }, 1000)
  349. }
  350.  
  351.  
  352. // 修改自 https://gf.qytechs.cn/scripts/14178 , https://github.com/machsix/Super-preloader
  353. var ShowPager = {
  354. getFullHref: function (e) {
  355. if (e != null && e.nodeType === 1 && e.href && e.href.slice(0,4) === 'http') return e.href;
  356. return '';
  357. },
  358. createDocumentByString: function (e) {
  359. if (e) {
  360. if ('HTML' !== document.documentElement.nodeName) return (new DOMParser).parseFromString(e, 'application/xhtml+xml');
  361. var t;
  362. try { t = (new DOMParser).parseFromString(e, 'text/html');} catch (e) {}
  363. if (t) return t;
  364. if (document.implementation.createHTMLDocument) {
  365. t = document.implementation.createHTMLDocument('ADocument');
  366. } else {
  367. try {((t = document.cloneNode(!1)).appendChild(t.importNode(document.documentElement, !1)), t.documentElement.appendChild(t.createElement('head')), t.documentElement.appendChild(t.createElement('body')));} catch (e) {}
  368. }
  369. if (t) {
  370. var r = document.createRange(),
  371. n = r.createContextualFragment(e);
  372. r.selectNodeContents(document.body);
  373. t.body.appendChild(n);
  374. for (var a, o = { TITLE: !0, META: !0, LINK: !0, STYLE: !0, BASE: !0}, i = t.body, s = i.childNodes, c = s.length - 1; c >= 0; c--) o[(a = s[c]).nodeName] && i.removeChild(a);
  375. return t;
  376. }
  377. } else console.error('没有找到要转成 DOM 的字符串');
  378. },
  379. loadMorePage: function () {
  380. if (curSite.pager) {
  381. let curPageEle = getElementByXpath(curSite.pager.nextLink);
  382. var url = this.getFullHref(curPageEle);
  383. //console.log(`${url} ${curSite.pageUrl}`);
  384. if(url === '') return;
  385. if(curSite.pageUrl === url) return;// 不会重复加载相同的页面
  386. curSite.pageUrl = url;
  387. // 读取下一页的数据
  388. curSite.pager.startFilter && curSite.pager.startFilter();
  389. GM_xmlhttpRequest({
  390. url: url,
  391. method: "GET",
  392. timeout: 5000,
  393. onload: function (response) {
  394. try {
  395. var newBody = ShowPager.createDocumentByString(response.responseText);
  396. let pageElems = getAllElements(curSite.pager.pageElement, newBody, newBody);
  397. let toElement = getAllElements(curSite.pager.HT_insert[0])[0];
  398. if (pageElems.length >= 0) {
  399. let addTo = "beforeend";
  400. if (curSite.pager.HT_insert[1] == 1) addTo = "beforebegin";
  401. // 插入新页面元素
  402. pageElems.forEach(function (one) {
  403. toElement.insertAdjacentElement(addTo, one);
  404. });
  405. // 清理帖子列表中帖子标题开头的〖XXX〗【XXX】文字
  406. cleanPostTitle();
  407. // 替换待替换元素
  408. try {
  409. let oriE = getAllElements(curSite.pager.replaceE);
  410. let repE = getAllElements(curSite.pager.replaceE, newBody, newBody);
  411. if (oriE.length === repE.length) {
  412. for (var i = 0; i < oriE.length; i++) {
  413. oriE[i].outerHTML = repE[i].outerHTML;
  414. }
  415. }
  416. } catch (e) {
  417. console.log(e);
  418. }
  419. }
  420. } catch (e) {
  421. console.log(e);
  422. }
  423. }
  424. });
  425. }
  426. },
  427. };
  428. function getElementByCSS(css, contextNode = document) {
  429. return contextNode.querySelector(css);
  430. }
  431. function getAllElementsByCSS(css, contextNode = document) {
  432. return [].slice.call(contextNode.querySelectorAll(css));
  433. }
  434. function getElementByXpath(xpath, contextNode, doc = document) {
  435. contextNode = contextNode || doc;
  436. try {
  437. const result = doc.evaluate(xpath, contextNode, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null);
  438. // 应该总是返回一个元素节点
  439. return result.singleNodeValue && result.singleNodeValue.nodeType === 1 && result.singleNodeValue;
  440. } catch (err) {
  441. throw new Error(`Invalid xpath: ${xpath}`);
  442. }
  443. }
  444. function getAllElementsByXpath(xpath, contextNode, doc = document) {
  445. contextNode = contextNode || doc;
  446. const result = [];
  447. try {
  448. const query = doc.evaluate(xpath, contextNode, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
  449. for (let i = 0; i < query.snapshotLength; i++) {
  450. const node = query.snapshotItem(i);
  451. // 如果是 Element 节点
  452. if (node.nodeType === 1) result.push(node);
  453. }
  454. } catch (err) {
  455. throw new Error(`无效 Xpath: ${xpath}`);
  456. }
  457. return result;
  458. }
  459. function getAllElements(selector, contextNode = undefined, doc = document, win = window, _cplink = undefined) {
  460. if (!selector) return [];
  461. contextNode = contextNode || doc;
  462. if (typeof selector === 'string') {
  463. if (selector.search(/^css;/i) === 0) {
  464. return getAllElementsByCSS(selector.slice(4), contextNode);
  465. } else {
  466. return getAllElementsByXpath(selector, contextNode, doc);
  467. }
  468. } else {
  469. const query = selector(doc, win, _cplink);
  470. if (!Array.isArray(query)) {
  471. throw new Error('getAllElements 返回错误类型');
  472. } else {
  473. return query;
  474. }
  475. }
  476. }
  477.  
  478.  
  479. // 获取GET参数
  480. function getQueryVariable(variable) {
  481. var query = window.location.search.substring(1);
  482. var vars = query.split("&");
  483. for (var i=0;i<vars.length;i++) {
  484. var pair = vars[i].split("=");
  485. if(pair[0] == variable){return pair[1];}
  486. }
  487. return(false);
  488. }
  489. })();

QingJ © 2025

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