🔥拓展增强🔥妖火网插件

发帖ubb增强、回帖ubb增强、回帖表情增强、查看贴子显示用户等级增强、手动吃肉增强、自动加载更多帖子、自动加载更多回复、一键自动上传图床、支持个性化菜单配置

  1. // ==UserScript==
  2. // @name 🔥拓展增强🔥妖火网插件
  3. // @namespace https://yaohuo.me/
  4. // @version 3.10.0
  5. // @description 发帖ubb增强、回帖ubb增强、回帖表情增强、查看贴子显示用户等级增强、手动吃肉增强、自动加载更多帖子、自动加载更多回复、一键自动上传图床、支持个性化菜单配置
  6. // @author 龙少c(id:20469)开发,参考其他大佬:外卖不用券(id:23825)、侯莫晨、Swilder-M
  7. // @match *://yaohuo.me/*
  8. // @match *://*.yaohuo.me/*
  9. // @icon https://yaohuo.me/css/favicon.ico
  10. // @run-at document-end
  11. // @license MIT
  12. // ==/UserScript==
  13.  
  14. /*
  15. 功能描述:
  16. 1. 贴子列表、贴子页、回复页支持自动加载更多、最大加载数量内自动加载(默认 150 条可单独配置),达到最大加载数量则不会自动加载。
  17. 2. 自己手动进入肉帖吃肉(默认关闭)
  18. 3. 吃过的肉帖。自己手动提交会弹窗提示已吃过肉,是否确认提交
  19. 4. 自动记忆所有吃过的肉帖,在配置天数内(默认一天可单独配置),吃过的肉帖不会重复吃肉。
  20. 5. 贴子页显示楼主等级(默认打开)
  21. 6. 增加可配置菜单,默认移动端开启,悬浮在右上角,PC 端不打开。所有功能都能单独开启和关闭,兼容其他插件,可以和其他插件一起使用,关闭本插件的相同功能即可。
  22. 7. 回帖表情增强,可配置默认是否展开
  23. 8. 回帖ubb增强,可配置默认是否展开
  24. 9. 发帖ubb增强
  25. 10. 发帖、回帖页面增加自动上传图床功能,支持手动选择图片上传、复制图片上传、截图上传、拖拽图片上传
  26. */
  27.  
  28. (function () {
  29. "use strict";
  30.  
  31. // 实现简易版替换用到的jquery,全部换成原生js太麻烦
  32. let $, jQuery;
  33. $ = jQuery = myJquery();
  34.  
  35. let settingData = {
  36. // 是否显示站内图标
  37. isShowSettingIcon: true,
  38. // 是否关闭站内勋章
  39. isCloseMedal: false,
  40. // 站内密码
  41. websitePassword: "",
  42. // 是否开启自动吃肉,手动进去肉帖自动吃肉
  43. isAutoEat: false,
  44. // 帖子里是否显示用户等级
  45. isShowLevel: true,
  46. // 刷新时间间隔
  47. timeInterval: 60,
  48. // 设置肉帖过期时间,过期前不会再自动吃肉
  49. expiredDays: 1,
  50. // 是否自动加载下一页
  51. isLoadNextPage: false,
  52. // 加载按钮方式: more / nextPage
  53. loadNextPageType: isMobile() ? "more" : "nextPage",
  54. // 一页最大的加载数量,超过数量就不会自动加载
  55. maxLoadNum: 150,
  56. // 滑块range最小和最大值
  57. minNumRange: 50,
  58. maxNumRange: 500,
  59. numStep: 50,
  60. // 设置悬浮按钮位置
  61. settingBtnLeft: 0,
  62. settingBtnTop: 0,
  63. // 按钮的响应试大小,单位vw,默认18vw
  64. settingIconResponsiveSize: 18,
  65. // 按钮最大的大小,单位px,默认100px
  66. settingIconMaxSize: 100,
  67.  
  68. // 自动加载页面时是否执行尾部
  69. isExecTrail: true,
  70.  
  71. // 是否增加发帖ubb
  72. isAddNewPostUBB: true,
  73. // 是否增加回帖ubb
  74. isAddReplyUBB: true,
  75. // 是否增加回帖表情
  76. isAddReplyFace: true,
  77. // 是否默认展开表情
  78. isUnfoldFace: false,
  79. // 是否默认展开表情
  80. isUnfoldUbb: false,
  81. // 是否自动上传到图床
  82. isUploadImage: false,
  83. // 上传图床token
  84. imageBedType: "极速图床",
  85. inkToken: "",
  86. meetToken: "",
  87. speedFreeToken: "",
  88. };
  89. let yaohuo_userData = null;
  90. // 数据初始化
  91. initSetting();
  92.  
  93. let {
  94. isAutoEat,
  95. isShowLevel,
  96. timeInterval,
  97. expiredDays,
  98. isLoadNextPage,
  99. isExecTrail,
  100.  
  101. maxLoadNum,
  102. minNumRange,
  103. maxNumRange,
  104. numStep,
  105.  
  106. isShowSettingIcon,
  107.  
  108. settingBtnLeft,
  109. settingBtnTop,
  110.  
  111. settingIconMaxSize,
  112. settingIconResponsiveSize,
  113.  
  114. isAddNewPostUBB,
  115. isAddReplyUBB,
  116. isAddReplyFace,
  117. isUnfoldFace,
  118. isUnfoldUbb,
  119.  
  120. loadNextPageType,
  121.  
  122. isUploadImage,
  123. imageBedType,
  124. inkToken,
  125. meetToken,
  126. speedFreeToken,
  127.  
  128. websitePassword,
  129. isCloseMedal,
  130. } = yaohuo_userData;
  131.  
  132. // 存储吃过肉的id,如果吃过肉则不会重复吃肉
  133. let autoEatList = getItem("autoEatList");
  134. // 回复页
  135. const viewPage = ["/bbs/book_re.aspx", "/bbs/book_view.aspx"];
  136. // 帖子列表页面
  137. const bbsPage = ["/bbs/book_list.aspx", "/bbs/list.aspx"];
  138. // 发帖
  139. const postPage = [
  140. "/bbs/book_view_add.aspx",
  141. "/bbs/book_view_sendmoney.aspx",
  142. "/bbs/book_view_addvote.aspx",
  143. "/bbs/book_view_addfile.aspx",
  144. "/bbs/book_view_mod.aspx",
  145. "/bbs/book_view_addURL.aspx",
  146. ];
  147. const loadNextPage = [
  148. /\/bbs\/book_re.aspx/,
  149. /\/bbs\/book_list.aspx/,
  150. /\/bbs\/list.aspx/,
  151. /\/bbs-.*\.html/,
  152. ];
  153. // 404
  154. const notFoundPage = ["/404.htm"];
  155. const faceList = [
  156. "踩.gif",
  157. "狂踩.gif",
  158. "淡定.gif",
  159. "囧.gif",
  160. "不要.gif",
  161. "重拳出击.gif",
  162. "砳砳.gif",
  163. "滑稽砳砳.gif",
  164. "沙发.gif",
  165. "汗.gif",
  166. "亲亲.gif",
  167. "太开心.gif",
  168. "酷.gif",
  169. "思考.gif",
  170. "发呆.gif",
  171. "得瑟.gif",
  172. "哈哈.gif",
  173. "泪流满面.gif",
  174. "放电.gif",
  175. "困.gif",
  176. "超人.gif",
  177. "害羞.gif",
  178. "呃.gif",
  179. "哇哦.gif",
  180. "要死了.gif",
  181. "谢谢.gif",
  182. "抓狂.gif",
  183. "无奈.gif",
  184. "不好笑.gif",
  185. "呦呵.gif",
  186. "感动.gif",
  187. "喜欢.gif",
  188. "疑问.gif",
  189. "委屈.gif",
  190. "你不行.gif",
  191. "流口水.gif",
  192. "潜水.gif",
  193. "咒骂.gif",
  194. "耶耶.gif",
  195. "被揍.gif",
  196. "抱走.gif",
  197. ];
  198. const diyFaceList = [{"url":"http:\/\/static2.51gonggui.com\/FhBfMfl4sGC3QJVTMaLqEKkE90Ia#.gif","name":"摸鱼"},{"url":"http:\/\/static2.51gonggui.com\/FmNyrjU8Wq0m3PiwHQJwDhHdv-EJ#.gif","name":"稽舞"},{"url":"http:\/\/static2.51gonggui.com\/FoKvdu89eiq0q-24IfOM2mFB0vIq#.gif","name":"色稽"},{"url":"http:\/\/static2.51gonggui.com\/FrZ6GDJiOAz3pp4e5_8uSShSXXXk#.gif","name":"撒娇"},{"url":"http:\/\/static2.51gonggui.com\/FiZiSSyXSa8eCzwOXmIfOOpfA_7a#.gif","name":"稽狗"},{"url":"http:\/\/static2.51gonggui.com\/FqNDzswUNJ-AsSHXyxXB4Qm1X0y-#.gif","name":"没钱"},{"url":"http:\/\/static2.51gonggui.com\/Fsq-HyBc5lP6vZY_qeWofOM9mRVH#.gif","name":"骚舞"},{"url":"http:\/\/static2.51gonggui.com\/FhCk4emkrO9f8ICFxKlm8wBcTOgT#.gif","name":"吃屎"},{"url":"http:\/\/static2.51gonggui.com\/FkEHwSlEfQ7bWya6-wg366Xy91qW#.gif","name":"鄙视"},{"url":"http:\/\/static2.51gonggui.com\/Fi2hY7M9DPgD9s0aCWemwk2iYUDW#.gif","name":"听歌"},{"url":"http:\/\/static2.51gonggui.com\/Fhry6EpdUBqFCt3OOyQTkLZMZGFR#.gif","name":"伸头"},{"url":"http:\/\/static2.51gonggui.com\/FhgYnWJ-apnyjSXOpInJhLbfUQFY#.gif","name":"鼓掌"},{"url":"http:\/\/static2.51gonggui.com\/FvSxOEIhyA7ID1J8emIME7tBT7Io#.gif","name":"踢腿"},{"url":"http:\/\/static2.51gonggui.com\/FunDHky9UKkB-4zj-bfSb82u81Xg#.gif","name":"男同"},{"url":"http:\/\/static2.51gonggui.com\/FgXUeACmKWWMDT9hrpVAnQp4dCqF#.gif","name":"手枪"},{"url":"http:\/\/static2.51gonggui.com\/Fg_qtra3abNozPxaoEMVKO7VIsuX#.gif","name":"拍头"},{"url":"http:\/\/static2.51gonggui.com\/FnNg1vOiuOlSe7WFWRyNZfO_4H3U#.gif","name":"躺平"},{"url":"http:\/\/static2.51gonggui.com\/Fj7WAkv87tpL1I26WQgSaXlsyYBL#.gif","name":"追稽"},{"url":"http:\/\/static2.51gonggui.com\/FgwFBazeUavJcw-SL7FS6wUkcUTk#.gif","name":"司稽"},{"url":"http:\/\/static2.51gonggui.com\/FjXNVx-MUgAVq62aNqekSPOUjDAC#.gif","name":"乞讨"},{"url":"http:\/\/static2.51gonggui.com\/FjudMlJdd8dLXuGjyASN7JldAxqe#.gif","name":"跪稽"},{"url":"http:\/\/static2.51gonggui.com\/Fm8DQQwyYthk8Q97ZLScgCDXsv4_#.gif","name":"刀你"},{"url":"http:\/\/static2.51gonggui.com\/FqTaBgs1l8bqeDYBxcWzxF4Wgt6_#.gif","name":"冲刺"},{"url":"http:\/\/static2.51gonggui.com\/Fmw152FIzN1gpFrbCKlp7cmqlCxc#.gif","name":"转圈"},{"url":"http:\/\/static2.51gonggui.com\/Fmf5aWS5yqycKebxTno7un53h9HW#.gif","name":"吃稽"},{"url":"http:\/\/static2.51gonggui.com\/FhUkLD2khZ7hn1uzArWkT47Pd9jq#.gif","name":"犯贱"},{"url":"http:\/\/static2.51gonggui.com\/FihrjZwpB1jMdOF9QvtQG3J32z4q#.gif","name":"牛掰"},{"url":"http:\/\/static2.51gonggui.com\/FlX6e1Ip6Z8gvl7lkimmCifwBhFt#.gif","name":"拥抱"},{"url":"http:\/\/static2.51gonggui.com\/FoIs-hNK7fhW8jwxEgDLRxARFcve#.gif","name":"拍头"},{"url":"http:\/\/static2.51gonggui.com\/Fgx4XlxG9461Y_TJsg0hGxPTylYi#.gif","name":"摇头"},{"url":"http:\/\/static2.51gonggui.com\/Fvrng91QU_PKY9Uwat77VTVouj5k#.gif","name":"挠头"},{"url":"http:\/\/static2.51gonggui.com\/FkyiMRaJI1BfuA6T3w4Z9mJh1qbg#.gif","name":"上学"},{"url":"http:\/\/static2.51gonggui.com\/FpZEifxiFGs1BWtHjFsk5tJJNKSE#.gif","name":"流汗"},{"url":"http:\/\/static2.51gonggui.com\/FiBZZ6mBTB5R5bu5lGkybboOwLwm#.gif","name":"摩擦"},{"url":"http:\/\/static2.51gonggui.com\/FmMtly844_wS6LfLLtLSwgzcXSqg#.gif","name":"喝饮料"},{"url":"http:\/\/static2.51gonggui.com\/FqyckEvAxFVyD1SmA9m2jInv_Crb#.gif","name":"猛狗"},{"url":"http:\/\/static2.51gonggui.com\/FmfsKjv4ymuWR80UGY-sea-I_Ey5#.gif","name":"妲己"},{"url":"http:\/\/static2.51gonggui.com\/FkEmzRCL3eJGlgkHHnHTy94sXwE1#.gif","name":"街舞"},{"url":"http:\/\/static2.51gonggui.com\/FgiAIOkFg8qG3UZKQx24ImVDrDRj#.gif","name":"功德"},{"url":"http:\/\/static2.51gonggui.com\/Fl2Zonx2Y8z-xZrSQnGBWzsnRKC9#.gif","name":"晃饮料"},{"url":"http:\/\/static2.51gonggui.com\/FvMXbnIX8RavSBAhflxf1zomD1ov#.gif","name":"扇子"},{"url":"http:\/\/static2.51gonggui.com\/FmD3h-QCVdJ-ehjLh8_G-nQzynuv#.gif","name":"膜拜"},{"url":"http:\/\/static2.51gonggui.com\/FoGXe8yRSIomTZFM78TZVyP-kwlz#.gif","name":"醒醒"},{"url":"http:\/\/static2.51gonggui.com\/Fim_ZRiJugrWJkDtq4SlqbOziuZ3#.gif","name":"巴掌"},{"url":"http:\/\/static2.51gonggui.com\/FpVLTimqXFvRJB9PxWDKMherZoRi#.gif","name":"鼓掌"},{"url":"http:\/\/static2.51gonggui.com\/Fit100hjJ-T5RwQxeNdoVWplvNvU#.gif","name":"该死"},{"url":"http:\/\/static2.51gonggui.com\/FkeVK5icB5-Pc7mbZitDTX1AqfNO#.gif","name":"红酒"},{"url":"http:\/\/static2.51gonggui.com\/FnjJRSH3_CLjYyyQzVjD8mtY-PdB#.gif","name":"开心"},{"url":"http:\/\/static2.51gonggui.com\/Foqd_tGWrk-ARnNrt-XraMCDzhUS#.gif","name":"紧张"},{"url":"http:\/\/static2.51gonggui.com\/FsCE3iHM0REN077WKr0bssyKiR7Z#.gif","name":"伤心2"}];
  199. // 批量添加事件数组
  200. let addEventAry = [
  201. {
  202. id: "ubb_id",
  203. ubb: "[userid]",
  204. offset: 0,
  205. },
  206. {
  207. id: "ubb_ip",
  208. ubb: "[ip]",
  209. offset: 0,
  210. },
  211. {
  212. id: "ubb_url",
  213. ubb: "[url=网址]文字说明[/url]",
  214. offset: 6,
  215. },
  216. {
  217. id: "ubb_text",
  218. ubb: "[text]全角转半角:代码内容[/text]",
  219. offset: 0,
  220. },
  221. {
  222. id: "ubb_br",
  223. ubb: "///",
  224. offset: 0,
  225. },
  226. {
  227. id: "ubb_hr",
  228. ubb: "[hr]",
  229. offset: 0,
  230. },
  231. {
  232. id: "ubb_left",
  233. ubb: "[left]",
  234. offset: 0,
  235. },
  236. {
  237. id: "ubb_center",
  238. ubb: "[center]",
  239. offset: 0,
  240. },
  241. {
  242. id: "ubb_right",
  243. ubb: "[right]",
  244. offset: 0,
  245. },
  246. {
  247. id: "ubb_font",
  248. ubb: "[font=serif][/font]",
  249. offset: 7,
  250. },
  251. {
  252. id: "ubb_b",
  253. ubb: "[b]加粗文字[/b]",
  254. offset: 4,
  255. },
  256. {
  257. id: "ubb_i",
  258. ubb: "[i]斜体文字[/i]",
  259. offset: 4,
  260. },
  261. {
  262. id: "ubb_u",
  263. ubb: "[u]下划线文字[/u]",
  264. offset: 0,
  265. },
  266. {
  267. id: "ubb_color",
  268. ubb: "[forecolor=red]颜色文字,默认红[/forecolor]",
  269. offset: 12,
  270. },
  271. {
  272. id: "ubb_img",
  273. ubb: "[img]图片链接[/img]",
  274. offset: 6,
  275. },
  276. {
  277. id: "ubb_strike",
  278. ubb: "[strike]删除线文字[/strike]",
  279. offset: 9,
  280. },
  281. {
  282. id: "ubb_call",
  283. ubb: "[call]拨号手机号码[/call]",
  284. offset: 0,
  285. },
  286. {
  287. id: "ubb_sms",
  288. ubb: "[url=sms:手机号码?body=短信内容]点此发送[/url]",
  289. offset: 0,
  290. },
  291. {
  292. id: "ubb_now",
  293. ubb: "当前系统日期和时间:[now]",
  294. offset: 0,
  295. },
  296. {
  297. id: "ubb_codo",
  298. ubb: "倒计天:[codo]2030-01-01[/codo]",
  299. offset: 0,
  300. },
  301. {
  302. id: "ubb_audio",
  303. ubb: "[audio=X]音频直链地址[/audio]",
  304. offset: 8,
  305. },
  306. {
  307. id: "ubb_movie",
  308. ubb: "[movie=100%*100%]视频直链地址|封面图片地址[/movie]",
  309. offset: 8,
  310. },
  311. {
  312. id: "ubb_nzgsa",
  313. ubb: "[audio=X]https://file.uhsea.com/2304/3deb45e90564252bf281f47c7b47a153KJ.mp3[/audio]",
  314. offset: 0,
  315. },
  316. ];
  317.  
  318. let timer = null;
  319.  
  320. // 是否点击加载更多
  321. let isClickLoadMoreBtn = false;
  322. // 是否是新页面
  323. let isNewPage = false;
  324.  
  325. const spanstyle =
  326. "color: #fff; padding: 2px 4px; font-size: 14px; background-color: #ccc;border-radius: 10%;";
  327. const a2style =
  328. "color: #fff; padding: 2px 4px; font-size: 14px; background-color: #d19275;border-radius: 10%;";
  329. const a3style =
  330. "color: #fff; padding: 2px 4px; font-size: 14px; background-color: #66ccff;border-radius: 10%;";
  331. // ==主代码执行==
  332. (function () {
  333. // 修复网站更新样式错乱问题
  334. handleStyle();
  335. // 处理浏览器滚动条事件
  336. handleWindowScroll();
  337. // 处理窗口改变事件
  338. handleWindowResize();
  339. // 添加站内设置按钮
  340. addSettingBtn();
  341. // 自动填充密码并确认
  342. handlePassword();
  343. // 关闭勋章显示
  344. handleCloseMedal();
  345. // 如果关闭了悬浮图标,在网站首页右上角添加插件设置入口
  346. handleAddSettingText();
  347. // 加载更多按钮点击事件监听
  348. handleAddLoadMoreBtnClick();
  349. // 手动吃肉:手动进入肉帖吃
  350. handleAutoEat();
  351. // 增加回帖ubb
  352. handleAddReplyUBB();
  353. // 增加回帖表情
  354. handleAddReplyFace();
  355. // 优化回帖
  356. handleReply();
  357. // 自动上传图床功能
  358. handleUploadImage();
  359. // 增加发帖ubb
  360. handleAddNewPostUBB();
  361. // 显示用户等级
  362. handleShowUserLevel();
  363. // 处理404页面跳回新帖页面
  364. handleNotFoundPage();
  365. })();
  366.  
  367. // ==其他功能函数和方法==
  368. function handleStyle() {
  369. MY_addStyle(`
  370. .centered-container {
  371. display: block !important;
  372. }
  373. `);
  374.  
  375. let flexDivs = document.querySelectorAll('div[style*="display: flex"]');
  376.  
  377. // 遍历选中的元素并添加额外的样式
  378. for (let i = 0; i < flexDivs.length; i++) {
  379. flexDivs[i].style.flexWrap = "wrap";
  380. }
  381. }
  382. function handleCloseMedal() {
  383. if (
  384. /^\/bbs-\d+\.html|\/bbs\/book_view.aspx$/.test(
  385. window.location.pathname
  386. ) &&
  387. isCloseMedal
  388. ) {
  389. let medalImg = [...document.querySelectorAll(".subtitle > img")].slice(2);
  390. medalImg.forEach((item, index) => {
  391. if (index === 0) {
  392. item.insertAdjacentHTML(
  393. "afterend",
  394. `<a href="javascript:;">已关闭勋章显示</a>`
  395. );
  396. }
  397. item.remove();
  398. });
  399. }
  400. }
  401. function handlePassword() {
  402. let password = document.querySelector("input[type=password]");
  403. let submit = document.querySelector("input[type=submit]");
  404. if (document.title === "请输入密码") {
  405. if (!password.value) {
  406. password.value = websitePassword;
  407. }
  408. if (password.value) {
  409. submit.click();
  410. }
  411. }
  412. }
  413. function handleAddSettingText() {
  414. // 修改pc端滚动条样式
  415. if (!isMobile()) {
  416. MY_addStyle(`
  417. /*滚动条整体样式*/
  418. /*高宽分别对应横竖滚动条的尺寸*/
  419. ::-webkit-scrollbar {
  420. width: 8px;
  421. height: 8px;
  422. background-color: #F5F5F5;
  423. }
  424. /*定义滚动条轨道 内阴影+圆角*/
  425. ::-webkit-scrollbar-track
  426. {
  427. box-shadow: inset 0 0 6px rgba(0,0,0,0.1);
  428. border-radius: 3px;
  429. background-color: #F5F5F5;
  430. }
  431. /*滚动条里面小方块*/
  432. ::-webkit-scrollbar-thumb {
  433. /* height: 50px; */
  434. border-radius:3px;
  435. box-shadow: inset 0 0 6px rgba(0,0,0,.3);
  436. background-color: #b8b8b8;
  437. }
  438. ::-webkit-scrollbar-thumb:hover {
  439. /* height: 50px; */
  440. background-color: #878987;
  441. border-radius: 6px
  442. }
  443. `);
  444. }
  445.  
  446. if (!isShowSettingIcon && $(".top2").length) {
  447. $(".top2").append(
  448. `<a class="yaohuo-setting-text" style="float:right;cursor: pointer;">插件设置</a>`
  449. );
  450.  
  451. $(".yaohuo-setting-text").click(() => {
  452. setMenu();
  453. });
  454. }
  455. }
  456. function isMobile() {
  457. return /Mobile/i.test(navigator.userAgent);
  458. }
  459. function initSetting() {
  460. // 在移动设备上执行的代码
  461. if (isMobile()) {
  462. // 移动端默认显示站内设置图标
  463. settingData.isShowSettingIcon = true;
  464. } else {
  465. // 在桌面设备上执行的代码
  466. }
  467.  
  468. // 获取用户历史数据
  469. yaohuo_userData = MY_getValue("yaohuo_userData");
  470.  
  471. // 查看本地是否存在旧数据
  472. if (!yaohuo_userData) {
  473. yaohuo_userData = settingData;
  474. // MY_setValue("yaohuo_userData", yaohuo_userData);
  475. }
  476.  
  477. // 自动更新数据
  478. for (let value in settingData) {
  479. if (!yaohuo_userData.hasOwnProperty(value)) {
  480. yaohuo_userData[value] = settingData[value];
  481. MY_setValue("yaohuo_userData", yaohuo_userData);
  482. }
  483. }
  484.  
  485. initSettingBtnPosition("init");
  486. }
  487. // 更新按钮位置到最右边
  488. /**
  489. * 当按钮靠最右边时,设置按钮的left偏移
  490. * @param {'init' | 'update'} type type值为init / update
  491. */
  492. function initSettingBtnPosition(type = "update") {
  493. let { settingBtnTop, settingIconResponsiveSize, settingIconMaxSize } =
  494. yaohuo_userData;
  495. let newLeft;
  496.  
  497. // btn最大100px,根据屏幕视口取18vw
  498. newLeft = Math.floor(
  499. window.innerWidth -
  500. Math.min(
  501. (window.innerWidth / 100) * settingIconResponsiveSize,
  502. settingIconMaxSize
  503. )
  504. );
  505. if (type === "update") {
  506. const floatingDiv = $("#floating-setting-btn")[0];
  507. floatingDiv.style.left = newLeft + "px";
  508. }
  509. saveSettingBtnPosition({ left: newLeft, top: settingBtnTop });
  510. }
  511. /**
  512. * 保存设置按钮的位置
  513. * @param {Object} pos - 按钮位置信息
  514. * @param {number} pos.left - 按钮左边距
  515. * @param {number} pos.top - 按钮上边距
  516. */
  517. function saveSettingBtnPosition({ left, top }) {
  518. yaohuo_userData.settingBtnLeft = left;
  519. yaohuo_userData.settingBtnTop = top;
  520.  
  521. setItem("yaohuo_userData", yaohuo_userData);
  522. }
  523.  
  524. function addSettingBtn() {
  525. if ($("#floating-setting-btn").length) {
  526. return;
  527. }
  528.  
  529. MY_addStyle(`
  530. #floating-setting-btn {
  531. display: ${isShowSettingIcon ? "block" : "none"};
  532. max-width: ${settingIconMaxSize}px;
  533. max-height: ${settingIconMaxSize}px;
  534. width: ${settingIconResponsiveSize}vw;
  535. height: ${settingIconResponsiveSize}vw;
  536. user-select: none;
  537. box-sizing: border-box;
  538. position: fixed;
  539. z-index: 999;
  540. cursor: move;
  541. background-repeat: no-repeat;
  542. background-position: center;
  543. }
  544. #floating-setting-btn svg {
  545. width: 100%;
  546. height: 100%;
  547. opacity: 0.6;
  548. filter: drop-shadow(0px 0px 3px #666);
  549. }
  550. .overflow-hidden-scroll {
  551. overflow: hidden !important;
  552. }
  553. .touch-action-none {
  554. touch-action: none;
  555. }
  556. .add-position-static{
  557. position: static !important;
  558. }
  559. `);
  560.  
  561. let innerH = `
  562. <div id="floating-setting-btn" style="top: ${settingBtnTop}px;
  563. left: ${settingBtnLeft}px;">
  564. </svg>
  565. <svg
  566. t="1681624380345"
  567. class="icon"
  568. viewBox="0 0 1024 1024"
  569. version="1.1"
  570. xmlns="http://www.w3.org/2000/svg"
  571. p-id="11281"
  572. width="80"
  573. height="80"
  574. >
  575. <path
  576. d="M761.472 978.944H253.696c-115.328 0-208.768-93.44-208.768-208.768V262.528C45.056 147.2 138.496 53.76 253.696 53.76h507.776C876.8 53.76 970.24 147.2 970.24 262.528v507.776c0 115.2-93.44 208.64-208.768 208.64z"
  577. fill="#89898F"
  578. p-id="11282"
  579. ></path>
  580. <path
  581. d="M602.368 513.408h-189.44C334.208 513.408 270.08 449.28 270.08 370.56S334.208 227.84 412.928 227.84h189.44c78.72 0 142.848 64.128 142.848 142.848s-64.128 142.72-142.848 142.72z m-189.44-262.272c-65.92 0-119.552 53.632-119.552 119.552s53.632 119.552 119.552 119.552h189.44c65.92 0 119.552-53.632 119.552-119.552s-53.632-119.552-119.552-119.552h-189.44zM611.072 804.864H404.224c-72.448 0-131.2-58.752-131.2-131.2s58.752-131.2 131.2-131.2h206.976c72.448 0 131.2 58.752 131.2 131.2-0.128 72.576-58.88 131.2-131.328 131.2z"
  582. fill="#FFFFFF"
  583. p-id="11283"
  584. ></path>
  585. <path
  586. d="M417.28 370.56m-80.128 0a80.128 80.128 0 1 0 160.256 0 80.128 80.128 0 1 0-160.256 0Z"
  587. fill="#FFFFFF"
  588. p-id="11284"
  589. ></path>
  590. <path
  591. d="M619.392 673.792m-80.128 0a80.128 80.128 0 1 0 160.256 0 80.128 80.128 0 1 0-160.256 0Z"
  592. fill="#89898F"
  593. p-id="11285"
  594. ></path>
  595. </svg>
  596. </div>
  597. `;
  598. $("body").append(innerH);
  599.  
  600. const floatingDiv = $("#floating-setting-btn")[0];
  601.  
  602. let mouseOffsetX = 0;
  603. let mouseOffsetY = 0;
  604. let isDragging = false;
  605. let dragThreshold = 5; // 拖动阈值,即鼠标移动的距离超过多少像素才开始拖动
  606. let clickThreshold = 500; // 点击阈值,即鼠标按下的时间不超过多少毫秒才算是点击
  607.  
  608. let mouseDownTime; // 鼠标按下的时间
  609. let mouseDownX; // 鼠标按下的位置
  610. let mouseDownY;
  611.  
  612. // 鼠标事件
  613. floatingDiv.addEventListener("mousedown", onMouseDown);
  614. document.addEventListener("mousemove", throttle(onMouseMove, 15));
  615. document.addEventListener("mouseup", onMouseUp);
  616.  
  617. // 触摸事件
  618. floatingDiv.addEventListener("touchstart", onTouchStart);
  619. document.addEventListener("touchmove", throttle(onTouchMove, 5));
  620. document.addEventListener("touchend", onTouchEnd);
  621.  
  622. // 鼠标事件监听器
  623. function onMouseDown(e) {
  624. floatingDiv.style.transition = "unset";
  625.  
  626. // 记录鼠标按下的时间和位置
  627. mouseDownTime = new Date().getTime();
  628. mouseDownX = e.clientX;
  629. mouseDownY = e.clientY;
  630.  
  631. // 鼠标相对于拖动图标的偏移x和y距离
  632. mouseOffsetX = e.clientX - floatingDiv.offsetLeft;
  633. mouseOffsetY = e.clientY - floatingDiv.offsetTop;
  634. isDragging = true;
  635.  
  636. e.preventDefault();
  637. e.stopPropagation();
  638. }
  639.  
  640. function onMouseMove(e) {
  641. if (!isDragging) {
  642. return;
  643. }
  644.  
  645. const left = e.clientX - mouseOffsetX;
  646. const top = e.clientY - mouseOffsetY;
  647.  
  648. const maxLeft = window.innerWidth - floatingDiv.offsetWidth;
  649. const maxTop = window.innerHeight - floatingDiv.offsetHeight;
  650.  
  651. floatingDiv.style.left = Math.min(Math.max(0, left), maxLeft) + "px";
  652. floatingDiv.style.top = Math.min(Math.max(0, top), maxTop) + "px";
  653. }
  654.  
  655. function onMouseUp(e) {
  656. if (!isDragging) {
  657. return;
  658. }
  659.  
  660. // 拖动结束重置数据
  661. mouseOffsetX = 0;
  662. mouseOffsetY = 0;
  663. isDragging = false;
  664.  
  665. // 如果按下的时间不够长,则认为是点击事件
  666. // 如果移动的距离
  667.  
  668. let distanceX = Math.abs(e.clientX - mouseDownX);
  669. let distanceY = Math.abs(e.clientY - mouseDownY);
  670. // 计算鼠标点击的时间小于500ms,并且移动的距离少于5像素则认为时点击事件
  671. if (
  672. new Date().getTime() - mouseDownTime < clickThreshold &&
  673. distanceX < dragThreshold &&
  674. distanceY < dragThreshold
  675. ) {
  676. setMenu();
  677. return;
  678. }
  679.  
  680. // 拖动按钮自动靠边处理
  681. let newLeft;
  682. let position = floatingDiv.getBoundingClientRect();
  683. // 如果左边比右边多就靠左,否则靠右
  684. if (window.innerWidth - position.right > position.left) {
  685. newLeft = 0;
  686. } else {
  687. newLeft = window.innerWidth - position.width;
  688. }
  689.  
  690. floatingDiv.style.left = newLeft + "px";
  691. floatingDiv.style.transition = "all 0.2s ease-in-out";
  692.  
  693. // 更新悬浮图标位置信息
  694. saveSettingBtnPosition({ top: position.top, left: newLeft });
  695. }
  696.  
  697. // 触摸事件监听器
  698. function onTouchStart(e) {
  699. floatingDiv.style.transition = "unset";
  700.  
  701. // 记录鼠标按下的时间和位置
  702. mouseDownTime = new Date().getTime();
  703.  
  704. mouseDownX = e.touches[0].clientX;
  705. mouseDownY = e.touches[0].clientY;
  706.  
  707. mouseOffsetX = e.touches[0].clientX - floatingDiv.offsetLeft;
  708. mouseOffsetY = e.touches[0].clientY - floatingDiv.offsetTop;
  709. isDragging = true;
  710.  
  711. e.preventDefault();
  712. e.stopPropagation();
  713.  
  714. $("body").addClass("touch-action-none");
  715. $("body").addClass("overflow-hidden-scroll");
  716. }
  717.  
  718. function onTouchMove(e) {
  719. if (!isDragging) {
  720. return;
  721. }
  722.  
  723. const left = e.touches[0].clientX - mouseOffsetX;
  724. const top = e.touches[0].clientY - mouseOffsetY;
  725.  
  726. const maxLeft = window.innerWidth - floatingDiv.offsetWidth;
  727. const maxTop = window.innerHeight - floatingDiv.offsetHeight;
  728.  
  729. floatingDiv.style.left = Math.min(Math.max(0, left), maxLeft) + "px";
  730. floatingDiv.style.top = Math.min(Math.max(0, top), maxTop) + "px";
  731. }
  732.  
  733. function onTouchEnd(e) {
  734. if (!isDragging) {
  735. return;
  736. }
  737. $("body").removeClass("touch-action-none");
  738. $("body").removeClass("overflow-hidden-scroll");
  739.  
  740. // 拖动结束重置数据
  741. mouseOffsetX = 0;
  742. mouseOffsetY = 0;
  743. isDragging = false;
  744.  
  745. // 如果按下的时间不够长,则认为是点击事件
  746. // 如果移动的距离
  747. let touch = e.changedTouches[0];
  748. let clientX = touch.clientX;
  749. let clientY = touch.clientY;
  750.  
  751. let distanceX = Math.abs(clientX - mouseDownX);
  752. let distanceY = Math.abs(clientY - mouseDownY);
  753.  
  754. // 计算鼠标点击的时间小于500ms,并且移动的距离少于5像素则认为时点击事件
  755. if (
  756. new Date().getTime() - mouseDownTime < clickThreshold &&
  757. distanceX < dragThreshold &&
  758. distanceY < dragThreshold
  759. ) {
  760. setMenu();
  761. return;
  762. }
  763.  
  764. // 拖动按钮自动靠边处理
  765. let newLeft;
  766. let position = floatingDiv.getBoundingClientRect();
  767. // 如果左边比右边多就靠左,否则靠右
  768. if (window.innerWidth - position.right > position.left) {
  769. newLeft = 0;
  770. } else {
  771. newLeft = window.innerWidth - position.width;
  772. }
  773. floatingDiv.style.left = newLeft + "px";
  774. floatingDiv.style.transition = "all 0.2s ease-in-out";
  775.  
  776. // 更新悬浮图标位置信息
  777. saveSettingBtnPosition({ top: position.top, left: newLeft });
  778. }
  779. }
  780. // 处理窗口改变事件
  781. function handleWindowResize() {
  782. // 窗口改变重新计算悬浮按钮的位置
  783. window.addEventListener("resize", function (e) {
  784. let { settingBtnLeft } = yaohuo_userData;
  785.  
  786. if (settingBtnLeft !== 0) {
  787. initSettingBtnPosition("update");
  788. }
  789. });
  790. }
  791. function getIcon(icon, tips) {
  792. let iconConfig = {
  793. tipIcon: `<svg t="1688708359358" onclick="alert('${tips}')" class="icon tip-icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1472" width="16" height="16"><path d="M512 1024c-281.6 0-512-230.4-512-512s230.4-512 512-512 512 230.4 512 512S793.6 1024 512 1024zM512 64C262.4 64 64 262.4 64 512s198.4 448 448 448 448-198.4 448-448S761.6 64 512 64z" fill="#3D3D3D" p-id="1473"></path><path d="M614.4 313.6c25.6 25.6 38.4 51.2 38.4 89.6 0 32-6.4 57.6-25.6 76.8C627.2 492.8 608 505.6 576 531.2 563.2 544 556.8 556.8 550.4 563.2c0 6.4-6.4 12.8-6.4 25.6 0 19.2-19.2 32-32 32l0 0c-19.2 0-38.4-19.2-32-38.4 0-12.8 6.4-25.6 6.4-32 6.4-19.2 32-44.8 70.4-76.8l12.8-12.8c12.8-12.8 19.2-32 19.2-44.8 0-19.2-6.4-38.4-19.2-51.2C556.8 345.6 537.6 339.2 512 339.2c-32 0-51.2 6.4-64 25.6C441.6 371.2 435.2 384 435.2 403.2c0 19.2-19.2 32-32 32l0 0c-19.2 0-38.4-19.2-32-44.8C377.6 358.4 390.4 339.2 403.2 320c25.6-25.6 64-38.4 108.8-38.4C556.8 281.6 595.2 288 614.4 313.6zM537.6 665.6c6.4 6.4 12.8 19.2 12.8 32 0 12.8-6.4 25.6-12.8 32-12.8 6.4-19.2 12.8-32 12.8-12.8 0-25.6-6.4-32-12.8-6.4-6.4-12.8-19.2-12.8-32 0-12.8 6.4-25.6 12.8-32 6.4-6.4 19.2-12.8 32-12.8C518.4 652.8 531.2 659.2 537.6 665.6z" fill="#3D3D3D" p-id="1474"></path></svg>`,
  794. eyeIcon: `<svg
  795. viewBox="64 64 896 896"
  796. focusable="false"
  797. data-icon="eye"
  798. width="20px"
  799. height="20px"
  800. fill="currentColor"
  801. aria-hidden="true"
  802. class="toggle-password"
  803. >
  804. <path
  805. d="M942.2 486.2C847.4 286.5 704.1 186 512 186c-192.2 0-335.4 100.5-430.2 300.3a60.3 60.3 0 0 0 0 51.5C176.6 737.5 319.9 838 512 838c192.2 0 335.4-100.5 430.2-300.3 7.7-16.2 7.7-35 0-51.5zM512 766c-161.3 0-279.4-81.8-362.7-254C232.6 339.8 350.7 258 512 258c161.3 0 279.4 81.8 362.7 254C791.5 684.2 673.4 766 512 766zm-4-430c-97.2 0-176 78.8-176 176s78.8 176 176 176 176-78.8 176-176-78.8-176-176-176zm0 288c-61.9 0-112-50.1-112-112s50.1-112 112-112 112 50.1 112 112-50.1 112-112 112z"
  806. ></path>
  807. </svg>`,
  808. };
  809. return iconConfig[icon];
  810. }
  811. function setMenu() {
  812. // 避免重复添加
  813. if ($(".yaohuo-modal-mask").length) {
  814. return;
  815. }
  816. MY_addStyle(`
  817. .yaohuo-modal-mask {
  818. display: none;
  819. position: fixed;
  820. top: 0;
  821. left: 0;
  822. bottom: 0;
  823. right: 0;
  824. z-index: 1000;
  825. height: 100%;
  826. width: 100%;
  827. background-color: rgba(0, 0, 0, 0.45);
  828. }
  829. .yaohuo-wrap {
  830. padding: 20px 0;
  831. box-sizing: border-box;
  832. position: relative;
  833. margin: 0 auto;
  834. background: #fff;
  835. border-radius: 12px;
  836. top: 10%;
  837. width: 400px;
  838. max-width: 95vw;
  839. user-select: none;
  840. }
  841. .yaohuo-wrap header {
  842. padding: 0 20px;
  843. color: rgba(0, 0, 0, 0.88);
  844. font-weight: 600;
  845. font-size: 16px;
  846. line-height: 1.5;
  847. margin-bottom: 8px;
  848. }
  849. .yaohuo-wrap footer {
  850. font-size: 0px;
  851. text-align: right;
  852. margin-top: 10px;
  853. padding: 0 20px;
  854. }
  855. .yaohuo-wrap footer button {
  856. font-size: 14px;
  857. height: 32px;
  858. padding: 4px 15px;
  859. border-radius: 6px;
  860. border: 1px solid transparent;
  861. cursor: pointer;
  862. }
  863. .yaohuo-wrap footer .cancel-btn {
  864. background-color: #fff;
  865. border-color: #d9d9d9;
  866. margin-right: 8px;
  867. box-shadow: 0 2px 0 rgba(0, 0, 0, 0.02);
  868. }
  869. .yaohuo-wrap footer .ok-btn {
  870. color: #fff;
  871. background-color: #1677ff;
  872. box-shadow: 0 2px 0 rgba(5, 145, 255, 0.1);
  873. }
  874. .yaohuo-wrap ul {
  875. margin: 0;
  876. padding: 0 10px 0 20px;
  877. margin-right: 7px;
  878. max-height: 65vh;
  879. overflow: auto;
  880. }
  881. /* 设置滚动条的宽度和高度 */
  882. .yaohuo-wrap ::-webkit-scrollbar {
  883. width: 3px;
  884. }
  885.  
  886. /* 设置滚动条滑块的背景色 */
  887. .yaohuo-wrap ::-webkit-scrollbar-thumb {
  888. background-color: #999;
  889. border-radius:10px;
  890. }
  891. .yaohuo-wrap i {
  892. font-style: normal;
  893. }
  894. .yaohuo-wrap li {
  895. display: flex;
  896. justify-content: space-between;
  897. align-items: center;
  898. height: 44px;
  899. }
  900. .yaohuo-wrap li .tip-icon{
  901. vertical-align: text-top;
  902. margin-left: 5px;
  903. cursor: pointer;
  904. }
  905. .yaohuo-wrap li input[type="range"] {
  906. width: 130px;
  907. height: 1px;
  908. appearance: auto;
  909. border: none;
  910. padding: 0;
  911. }
  912. .yaohuo-wrap li select {
  913. height: 28px;
  914. line-height: 28px;
  915. font-size: 14px;
  916. width: 130px;
  917. border: 1px solid #5c5c5c;
  918. border-radius: 5px;
  919. text-align: center;
  920. outline: 0;
  921. margin-right: 0;
  922. }
  923.  
  924. .yaohuo-wrap .switch {
  925. position: relative;
  926. display: inline-block;
  927. width: 60px;
  928. height: 30px;
  929. }
  930.  
  931. .yaohuo-wrap .switch input {
  932. opacity: 0;
  933. width: 0;
  934. height: 0;
  935. }
  936.  
  937. .yaohuo-wrap .password-container{
  938. width: 60%;
  939. position: relative;
  940. }
  941.  
  942. .password-container .toggle-password {
  943. position: absolute;
  944. top: 52%;
  945. right: 6px;
  946. transform: translateY(-50%);
  947. cursor: pointer;
  948. }
  949.  
  950. .yaohuo-wrap-title{
  951. height: 38px !important;
  952. }
  953. .yaohuo-wrap-title .title-line {
  954. margin: 0px;
  955. border: none;
  956. border-top: 1px dashed #dcdcdc;
  957. width: 30%; /* 可根据需要调整宽度 */
  958. }
  959.  
  960. .yaohuo-wrap li .password-container input {
  961. width: 100%;
  962. box-sizing: border-box;
  963. height: 32px;
  964. padding-right: 28px;
  965. }
  966. .yaohuo-wrap li input[type="number"] {
  967. width: 130px;
  968. box-sizing: border-box;
  969. height: 30px;
  970. }
  971.  
  972. .yaohuo-wrap .switch label {
  973. position: absolute;
  974. cursor: pointer;
  975. top: 0;
  976. left: 0;
  977. right: 0;
  978. bottom: 0;
  979. background-color: #ccc;
  980. -webkit-transition: 0.4s;
  981. transition: 0.4s;
  982. border-radius: 34px;
  983. }
  984.  
  985. .yaohuo-wrap .switch label::before {
  986. position: absolute;
  987. content: "";
  988. height: 22px;
  989. width: 22px;
  990. left: 6px;
  991. bottom: 4px;
  992. background-color: white;
  993. transition: 0.4s;
  994. border-radius: 50%;
  995. }
  996.  
  997. .yaohuo-wrap .switch input:checked + label {
  998. background-color: #2196f3;
  999. }
  1000.  
  1001. .yaohuo-wrap .switch input:checked + label::before {
  1002. transform: translateX(26px);
  1003. }
  1004. .yaohuo-wrap hr {
  1005. margin:5px 0
  1006. }
  1007. `);
  1008. let innerH = `
  1009. <div class="yaohuo-modal-mask">
  1010. <div class="yaohuo-wrap">
  1011. <header>🔥拓展增强🔥妖火插件设置</header>
  1012. <ul>
  1013. <li class="yaohuo-wrap-title">
  1014. <hr class="title-line title-line-left" />
  1015. <b>站内设置</b>
  1016. <hr class="title-line title-line-right" />
  1017. </li>
  1018. <li>
  1019. <span>显示站内设置图标</span>
  1020. <div class="switch">
  1021. <input type="checkbox" id="isShowSettingIcon" data-key="isShowSettingIcon" />
  1022. <label for="isShowSettingIcon"></label>
  1023. </div>
  1024. </li>
  1025. <li>
  1026. <span>设置图标大小:<i class="range-num">${settingIconMaxSize}</i>px</span>
  1027. <input
  1028. type="range"
  1029. id="settingIconMaxSize"
  1030. data-key="settingIconMaxSize"
  1031. min="${40}"
  1032. value="${settingIconMaxSize}"
  1033. max="${100}"
  1034. step="${5}"
  1035. />
  1036. </li>
  1037. <li>
  1038. <span>站内密码设置</span>
  1039. <div class="password-container">
  1040. <input
  1041. type="password"
  1042. placeholder="自动填充密码并确认"
  1043. id="websitePassword"
  1044. data-key="websitePassword"
  1045. value="${websitePassword}"
  1046. />
  1047. ${getIcon("eyeIcon")}
  1048. </div>
  1049. </li>
  1050. <li>
  1051. <span>关闭勋章显示</span>
  1052. <div class="switch">
  1053. <input type="checkbox" id="isCloseMedal" data-key="isCloseMedal" />
  1054. <label for="isCloseMedal"></label>
  1055. </div>
  1056. </li>
  1057.  
  1058. <li class="yaohuo-wrap-title">
  1059. <hr class="title-line title-line-left" />
  1060. <b>图床设置</b>
  1061. <hr class="title-line title-line-right" />
  1062. </li>
  1063.  
  1064. <li>
  1065. <span>自动上传图床</span>
  1066. <div class="switch">
  1067. <input type="checkbox" id="isUploadImage" data-key="isUploadImage" />
  1068. <label for="isUploadImage"></label>
  1069. </div>
  1070. </li>
  1071. <li>
  1072. <span>图床设置</span>
  1073. <select data-key="imageBedType" id="imageBedType">
  1074. <option value="水墨图床">水墨图床</option>
  1075. <option value="极速图床">极速图床</option>
  1076. <option value="葫芦侠图床">葫芦侠图床</option>
  1077. </select>
  1078. </li>
  1079. <li>
  1080. <span><a href="https://img.ink" target="_blank">水墨图床token</a></span>
  1081. <div class="password-container">
  1082. <input
  1083. type="password"
  1084. placeholder="为空则为游客上传"
  1085. id="inkToken"
  1086. data-key="inkToken"
  1087. value="${inkToken}"
  1088. />
  1089. ${getIcon("eyeIcon")}
  1090. </div>
  1091. </li>
  1092. <li>
  1093. <span><a href="https://tucdn.wpon.cn" target="_blank">极速图床token</a></span>
  1094. <div class="password-container">
  1095. <input
  1096. type="password"
  1097. placeholder="为空则为游客上传"
  1098. id="speedFreeToken"
  1099. data-key="speedFreeToken"
  1100. value="${speedFreeToken}"
  1101. />
  1102. ${getIcon("eyeIcon")}
  1103. </div>
  1104. </li>
  1105. <li class="yaohuo-wrap-title">
  1106. <hr class="title-line title-line-left" />
  1107. <b>吃肉设置</b>
  1108. <hr class="title-line title-line-right" />
  1109. </li>
  1110. <li>
  1111. <span>手动进贴吃肉</span>
  1112. <div class="switch">
  1113. <input type="checkbox" id="isAutoEat" data-key="isAutoEat" />
  1114. <label for="isAutoEat"></label>
  1115. </div>
  1116. </li>
  1117. <li>
  1118. <span>肉帖过期时间:<i class="range-num">${expiredDays}</i>天</span>
  1119. <input
  1120. type="range"
  1121. id="expiredDays"
  1122. data-key="expiredDays"
  1123. min="${1}"
  1124. value="${expiredDays}"
  1125. max="${7}"
  1126. step="${1}"
  1127. />
  1128. </li>
  1129. <li class="yaohuo-wrap-title">
  1130. <hr class="title-line title-line-left" />
  1131. <b>回帖设置</b>
  1132. <hr class="title-line title-line-right" />
  1133. </li>
  1134. <li>
  1135. <span>回帖表情增强</span>
  1136. <div class="switch">
  1137. <input type="checkbox" id="isAddReplyFace" data-key="isAddReplyFace" />
  1138. <label for="isAddReplyFace"></label>
  1139. </div>
  1140. </li>
  1141. <li>
  1142. <span>回帖表情默认展开</span>
  1143. <div class="switch">
  1144. <input type="checkbox" id="isUnfoldFace" data-key="isUnfoldFace" />
  1145. <label for="isUnfoldFace"></label>
  1146. </div>
  1147. </li>
  1148. <hr>
  1149. <li>
  1150. <span>回帖UBB增强</span>
  1151. <div class="switch">
  1152. <input type="checkbox" id="isAddReplyUBB" data-key="isAddReplyUBB" />
  1153. <label for="isAddReplyUBB"></label>
  1154. </div>
  1155. </li>
  1156. <li>
  1157. <span>回帖UBB默认展开</span>
  1158. <div class="switch">
  1159. <input type="checkbox" id="isUnfoldUbb" data-key="isUnfoldUbb" />
  1160. <label for="isUnfoldUbb"></label>
  1161. </div>
  1162. </li>
  1163. <li class="yaohuo-wrap-title">
  1164. <hr class="title-line title-line-left" />
  1165. <b>发帖设置</b>
  1166. <hr class="title-line title-line-right" />
  1167. </li>
  1168. <li>
  1169. <span>发帖UBB增强</span>
  1170. <div class="switch">
  1171. <input type="checkbox" id="isAddNewPostUBB" data-key="isAddNewPostUBB" />
  1172. <label for="isAddNewPostUBB"></label>
  1173. </div>
  1174. </li>
  1175. <li class="yaohuo-wrap-title">
  1176. <hr class="title-line title-line-left" />
  1177. <b>自动加载设置</b>
  1178. <hr class="title-line title-line-right" />
  1179. </li>
  1180. <li>
  1181. <span>自动加载下一页</span>
  1182. <div class="switch">
  1183. <input type="checkbox" id="isLoadNextPage" data-key="isLoadNextPage" />
  1184. <label for="isLoadNextPage"></label>
  1185. </div>
  1186. </li>
  1187. <li>
  1188. <span>自动加载下一页方式</span>
  1189. <select data-key="loadNextPageType" id="loadNextPageType">
  1190. <option value="more">加载更多按钮</option>
  1191. <option value="nextPage">下一页按钮</option>
  1192. </select>
  1193. </li>
  1194. <li>
  1195. <span>自动加载最大数:<i class="range-num">${maxLoadNum}</i>个</span>
  1196. <input
  1197. id="maxLoadNum"
  1198. type="range"
  1199. data-key="maxLoadNum"
  1200. min="${minNumRange}"
  1201. value="${maxLoadNum}"
  1202. max="${maxNumRange}"
  1203. step="${numStep}"
  1204. />
  1205. </li>
  1206. <li class="yaohuo-wrap-title">
  1207. <hr class="title-line title-line-left" />
  1208. <b>显示帖子等级</b>
  1209. <hr class="title-line title-line-right" />
  1210. </li>
  1211. <li>
  1212. <span>贴子显示等级</span>
  1213. <div class="switch">
  1214. <input type="checkbox" id="isShowLevel" data-key="isShowLevel" />
  1215. <label for="isShowLevel"></label>
  1216. </div>
  1217. </li>
  1218. </ul>
  1219. <footer>
  1220. <button class="cancel-btn">取消</button>
  1221. <button class="ok-btn">确认</button>
  1222. </footer>
  1223. </div>
  1224. </div>
  1225. `;
  1226. $("body").append(innerH).addClass("overflow-hidden-scroll");
  1227.  
  1228. $(".yaohuo-modal-mask").show();
  1229.  
  1230. setSettingInputEvent("edit");
  1231.  
  1232. $(".cancel-btn").click(handleCancelBtn);
  1233. $(".ok-btn").click(handleOkBtn);
  1234. }
  1235. /**
  1236. * 设置设置菜单,点击设置打开菜单,并且回显数据,保存则保存数据
  1237. * @param {"edit" | 'save'} status edit和save两种模式
  1238. */
  1239. function setSettingInputEvent(status = "edit") {
  1240. $(".yaohuo-wrap input, .yaohuo-wrap select").each((index, item) => {
  1241. let type = item.type;
  1242. let dataKey = item.getAttribute("data-key");
  1243. switch (type) {
  1244. case "checkbox":
  1245. if (status === "edit") {
  1246. item.checked = getValue(dataKey) ? true : false;
  1247. // 根据当前的按钮选中状态处理子项的联动显示或隐藏
  1248. autoShowElement({
  1249. fatherIdAry: ["isShowSettingIcon"],
  1250. childIdAry: ["settingIconMaxSize"],
  1251. dataKey,
  1252. });
  1253. autoShowElement({
  1254. fatherIdAry: ["isLoadNextPage"],
  1255. childIdAry: ["loadNextPageType", "maxLoadNum"],
  1256. dataKey,
  1257. });
  1258. autoShowElement({
  1259. fatherIdAry: ["isAutoEat"],
  1260. childIdAry: ["expiredDays"],
  1261. dataKey,
  1262. });
  1263. autoShowElement({
  1264. fatherIdAry: ["isUploadImage"],
  1265. childIdAry: [
  1266. "imageBedType",
  1267. "inkToken",
  1268. "meetToken",
  1269. "speedFreeToken",
  1270. ],
  1271. dataKey,
  1272. });
  1273. autoShowElement({
  1274. fatherIdAry: ["isAddReplyUBB"],
  1275. childIdAry: ["isUnfoldUbb"],
  1276. dataKey,
  1277. });
  1278. autoShowElement({
  1279. fatherIdAry: ["isAddReplyFace"],
  1280. childIdAry: ["isUnfoldFace"],
  1281. dataKey,
  1282. });
  1283. } else {
  1284. setValue(dataKey, item.checked);
  1285. }
  1286. break;
  1287.  
  1288. case "range":
  1289. if (status === "edit") {
  1290. $(item).on("input propertychange", function (event) {
  1291. $(item).prev().children(".range-num").text(item.value);
  1292. });
  1293. } else {
  1294. setValue(dataKey, parseInt(item.value));
  1295. }
  1296. break;
  1297.  
  1298. case "select-one":
  1299. if (status === "edit") {
  1300. item.value = getValue(dataKey);
  1301. $(item).on("change", function (event) {
  1302. autoShowImageToken(item, dataKey);
  1303. });
  1304. autoShowImageToken(item, dataKey);
  1305. } else {
  1306. setValue(dataKey, item.value);
  1307. }
  1308. break;
  1309.  
  1310. case "number":
  1311. if (status === "edit") {
  1312. item.value = getValue(dataKey);
  1313. } else {
  1314. setValue(dataKey, item.value);
  1315. }
  1316. break;
  1317.  
  1318. case "password":
  1319. if (status === "edit") {
  1320. item.value = getValue(dataKey, "");
  1321. $(item)
  1322. .next()
  1323. .on("click", function (event) {
  1324. item.type = item.type === "password" ? "text" : "password";
  1325. });
  1326. } else {
  1327. setValue(dataKey, item.value);
  1328. }
  1329. break;
  1330.  
  1331. default:
  1332. if (status === "edit") {
  1333. item.value = getValue(dataKey, "");
  1334. } else {
  1335. setValue(dataKey, item.value);
  1336. }
  1337. break;
  1338. }
  1339. });
  1340. function autoShowImageToken(item, dataKey) {
  1341. if (dataKey === "imageBedType") {
  1342. let config = {
  1343. 水墨图床: "#inkToken",
  1344. 极速图床: "#speedFreeToken",
  1345. };
  1346. Object.keys(config).forEach((name) => {
  1347. if (item.value === name) {
  1348. $(config[name]).closest("li").show();
  1349. } else {
  1350. $(config[name]).closest("li").hide();
  1351. }
  1352. });
  1353. }
  1354. }
  1355. /**
  1356. * 根据当前的选中状态处理子项的显示或隐藏
  1357. * @param {Object} options - 选项对象
  1358. * @param {Array<string>} options.fatherIdAry - 包含父元素ID的字符串数组
  1359. * @param {Array<string>} options.childId - 子元素的ID
  1360. * @param {string} options.dataKey - 存储在父元素上的数据键名
  1361. */
  1362. function autoShowElement({ fatherIdAry, childIdAry, dataKey }) {
  1363. execFn();
  1364. fatherIdAry.forEach((item) => {
  1365. $(`#${item}`).on("change", function (event) {
  1366. execFn();
  1367. });
  1368. });
  1369. function execFn() {
  1370. if (fatherIdAry.includes(dataKey)) {
  1371. childIdAry.forEach((childId) => {
  1372. let parent = $(`#${childId}`).closest("li");
  1373.  
  1374. let isShow = fatherIdAry.some((item) =>
  1375. $(`#${item}`).prop("checked")
  1376. );
  1377. isShow ? parent.show() : parent.hide();
  1378. });
  1379. }
  1380. }
  1381. }
  1382. }
  1383. function checkSaveSetting() {
  1384. let openUploadImageBed = $("#isUploadImage").prop("checked");
  1385. let imageBedType = $("#imageBedType").prop("value");
  1386. let meetToken = $("#meetToken").prop("value");
  1387.  
  1388. // if (openUploadImageBed && imageBedType === "遇见图床" && !meetToken) {
  1389. // alert("遇见图床必须填写token");
  1390. // return false;
  1391. // }
  1392. return true;
  1393. }
  1394. function handleCancelBtn() {
  1395. $("body").removeClass("overflow-hidden-scroll");
  1396. $(".yaohuo-modal-mask").remove();
  1397. }
  1398. function handleOkBtn() {
  1399. if (!checkSaveSetting()) {
  1400. return;
  1401. }
  1402. setSettingInputEvent("save");
  1403. $("body").removeClass("overflow-hidden-scroll");
  1404. $(".yaohuo-modal-mask").hide();
  1405. MY_setValue("yaohuo_userData", yaohuo_userData);
  1406. if (!yaohuo_userData.isShowSettingIcon) {
  1407. $("#floating-setting-btn").hide();
  1408. } else {
  1409. $("#floating-setting-btn").show();
  1410. }
  1411. // 刷新页面
  1412. setTimeout(function () {
  1413. window.location.reload();
  1414. }, 300);
  1415. }
  1416.  
  1417. // 浏览器scroll事件
  1418. function handleWindowScroll() {
  1419. window.addEventListener(
  1420. "scroll",
  1421. throttle(() => {
  1422. let isPage = loadNextPage.some((item) =>
  1423. item.test(window.location.pathname)
  1424. );
  1425.  
  1426. // 处理点击加载更多后的全自动吃肉
  1427. if (isPage) {
  1428. let nextBtn = null;
  1429. // 下一页按钮的父节点
  1430. let nextPageWrap = document.querySelector(".bt2");
  1431.  
  1432. if (loadNextPageType === "more" || !nextPageWrap) {
  1433. // 加载更多加载下一页
  1434. nextBtn = document.querySelector("span[id$=show_tip]");
  1435.  
  1436. // 已经请求到数据
  1437. if (nextBtn?.innerText.includes("加载更多")) {
  1438. // 加载完成了
  1439.  
  1440. isClickLoadMoreBtn = false;
  1441. isNewPage = false;
  1442.  
  1443. // 处理自动加载更多,需要放到最后
  1444. handleLoadNextPage();
  1445. }
  1446. } else {
  1447. // 下一页按钮加载下一页
  1448.  
  1449. nextBtn = nextPageWrap.firstChild;
  1450.  
  1451. handleLoadNextPage();
  1452. }
  1453. }
  1454. }, 100)
  1455. );
  1456. }
  1457. // 返回当前列表数据的长度
  1458. function getListLength() {
  1459. let length = 0;
  1460. if ($(".listdata").length) {
  1461. length = $(".listdata").length;
  1462. } else {
  1463. length = $(".reline").length + $(".list-reply").length;
  1464. }
  1465. return length;
  1466. }
  1467. // 自动吃肉:手动进入肉帖自动吃
  1468. function handleAutoEat() {
  1469. if (/^\/bbs-.*\.html$/.test(window.location.pathname) && isAutoEat) {
  1470. const form = document.getElementsByName("f")[0];
  1471. let isAutoEatBbs = window.location.search.includes("open=new");
  1472. if (!form) {
  1473. // 如果是自动吃肉的则直接返回,并记录不可吃肉
  1474. if (isAutoEatBbs) {
  1475. autoEatCallback();
  1476. }
  1477. return;
  1478. }
  1479. const face = form.getElementsByTagName("select")[0];
  1480. const replyBtn = document.getElementsByName("g")[0];
  1481.  
  1482. const textarea = document.querySelector(".retextarea");
  1483. // 帖子标识id
  1484. let id = window.location.pathname.match(/\d+/)[0];
  1485.  
  1486. // 吃肉 必须放在后面
  1487. const fileTag = document.querySelector(
  1488. "a[href^='/bbs/book_re_addfile.aspx']"
  1489. );
  1490. let eatMeat = document.createElement("input");
  1491. eatMeat.style.float = "right";
  1492. eatMeat.type = "submit";
  1493. eatMeat.value = "一键吃肉";
  1494. eatMeat.addEventListener("click", (e) => {
  1495. e.preventDefault();
  1496. let eatWordsArr = [
  1497. "吃",
  1498. "吃吃",
  1499. "吃吃.",
  1500. "吃吃。",
  1501. "吃吃..",
  1502. "吃吃。。",
  1503. "吃了",
  1504. "吃肉",
  1505. "来吃肉",
  1506. "吃.",
  1507. "吃。",
  1508. "吃了.",
  1509. "吃了。",
  1510. "吃肉.",
  1511. "吃肉。",
  1512. "来吃肉.",
  1513. "来吃肉。",
  1514. "吃..",
  1515. "吃。。",
  1516. "吃了..",
  1517. "吃了。。",
  1518. "吃肉..",
  1519. "吃肉。。",
  1520. "来吃肉..",
  1521. "来吃肉。。",
  1522. "口乞了",
  1523. "口乞了.",
  1524. "口乞了。",
  1525. "口乞肉",
  1526. "口乞肉.",
  1527. "口乞肉。",
  1528. "口乞..",
  1529. "口乞。。",
  1530. "chile..",
  1531. "chile。。",
  1532. "7肉..",
  1533. "7肉。。",
  1534. "7了..",
  1535. "7了。。",
  1536. "肉肉肉",
  1537. "肉肉肉.",
  1538. "肉肉肉。",
  1539. "肉肉肉..",
  1540. "肉肉肉。。",
  1541. "先吃肉",
  1542. "先吃肉.",
  1543. "先吃肉。",
  1544. "先吃肉..",
  1545. "先吃肉。。",
  1546. ];
  1547.  
  1548. let index = Math.floor(Math.random() * eatWordsArr.length);
  1549. console.log("吃肉回复:", eatWordsArr[index]);
  1550. insertText(textarea, eatWordsArr[index], 0);
  1551. replyBtn.click();
  1552. autoEatCallback();
  1553. });
  1554.  
  1555. // 添加事件监听,如果吃过肉则会提示
  1556. document.getElementsByName("g")[0].addEventListener(
  1557. "click",
  1558. (e) => {
  1559. if (autoEatList[id] && !confirm("当前已经吃过肉,是否继续回复")) {
  1560. // 取消提交
  1561. textarea.value = "";
  1562. e.preventDefault();
  1563. e.stopPropagation();
  1564. }
  1565. },
  1566. true
  1567. );
  1568.  
  1569. const meatTag = document.querySelector("span.yushuzi");
  1570.  
  1571. if (!isAutoEat) {
  1572. console.log("未开启自动吃肉,可在编辑脚本进行开启");
  1573. } else {
  1574. if (meatTag == undefined) {
  1575. console.log("非肉勿7");
  1576. // autoEatCallback();
  1577. } else if (parseInt(meatTag.innerHTML) <= 0) {
  1578. console.log("无肉怎7");
  1579. // 把无肉帖添加进去
  1580. autoEatCallback();
  1581. } else {
  1582. let autoEatList = getItem("autoEatList");
  1583.  
  1584. if (!autoEatList[id]) {
  1585. console.log("有肉快7");
  1586. eatMeat.click();
  1587. } else {
  1588. console.log("已经吃过了");
  1589. autoEatCallback();
  1590. }
  1591. }
  1592. }
  1593. // 将吃肉插入到文件回帖后面
  1594. fileTag.after(eatMeat);
  1595. }
  1596. }
  1597. function insertText(obj, str, offset) {
  1598. if (document.selection) {
  1599. let sel = document.selection.createRange();
  1600. sel.text = str;
  1601. } else if (
  1602. typeof obj.selectionStart === "number" &&
  1603. typeof obj.selectionEnd === "number"
  1604. ) {
  1605. let startPos = obj.selectionStart,
  1606. endPos = obj.selectionEnd,
  1607. cursorPos = startPos,
  1608. tmpStr = obj.value;
  1609. obj.value =
  1610. tmpStr.substring(0, startPos) +
  1611. str +
  1612. tmpStr.substring(endPos, tmpStr.length);
  1613. cursorPos += str.length;
  1614. obj.selectionStart = obj.selectionEnd = cursorPos - offset;
  1615. } else {
  1616. obj.value += str;
  1617. }
  1618. obj.focus();
  1619. }
  1620. // 吃完肉的回调
  1621. function autoEatCallback() {
  1622. let id = window.location.pathname.match(/\d+/)[0];
  1623. let isAutoEatBbs = window.location.search.includes("open=new");
  1624. // let autoEatList = getItem("autoEatList");
  1625.  
  1626. autoEatList[id] = new Date().getTime();
  1627.  
  1628. setItem("autoEatList", autoEatList);
  1629. }
  1630. /**
  1631. * 返回指定天数后的一天开始的时间,例如1天,则为明天00:00:00,2天则为后天00:00:00
  1632. * 肉帖1天有效期则代表明天00:00:00过期,2天有效期则是后天00:00:00
  1633. * @param {number} time 传入一个时间戳
  1634. * @param {number} days 传入过期的天数
  1635. * @returns {number} 返回到期时间的时间戳
  1636. */
  1637.  
  1638. function timeLeft(time, days = 1) {
  1639. const target = new Date(time + (days - 1) * 24 * 60 * 60 * 1000);
  1640. target.setHours(24, 0, 0, 0);
  1641. return target.getTime();
  1642. }
  1643. // 获取值
  1644. function getItem(key, defaultValue = {}) {
  1645. if (key === "autoEatList") {
  1646. let autoEatList = MY_getValue(key, {});
  1647. // 删除过期的肉帖
  1648. deleteExpiredID(autoEatList);
  1649. // 更新肉帖数据
  1650. setItem(key, autoEatList);
  1651. return autoEatList;
  1652. }
  1653. return MY_getValue(key, {});
  1654. }
  1655. function MY_addStyle(innerHTML) {
  1656. // 创建 style 元素
  1657. let style = document.createElement("style");
  1658. style.type = "text/css";
  1659.  
  1660. // 设置样式内容
  1661. let css = innerHTML;
  1662. style.innerHTML = css;
  1663.  
  1664. // 将 style 元素添加到 head 元素中
  1665. document.head.appendChild(style);
  1666. }
  1667. function MY_setValue(key, value) {
  1668. if (typeof value === "object") {
  1669. value = JSON.stringify(value);
  1670. }
  1671. localStorage.setItem(key, value);
  1672. }
  1673. function MY_getValue(key, defaultValue = {}) {
  1674. const value = localStorage.getItem(key) || defaultValue;
  1675. try {
  1676. return JSON.parse(value);
  1677. } catch (e) {
  1678. return value;
  1679. }
  1680. }
  1681. // 设置值
  1682. function setItem(key, value) {
  1683. // if (key === "autoEatList") {
  1684. // deleteExpiredID(value); //删除过期的肉帖
  1685. // }
  1686. MY_setValue(key, value);
  1687. }
  1688. /**
  1689. * 返回yaohuo_userData里的数据
  1690. * @param {string} key 要获取值的属性
  1691. * @param {any} value 获取的值
  1692. * @returns {any}
  1693. */
  1694. function getValue(key, value) {
  1695. let setting = yaohuo_userData;
  1696. return setting[key] || value;
  1697. }
  1698. /**
  1699. * 更新yaohuo_userData里的数据
  1700. * @param {string} key 要设置值的属性
  1701. * @param {any} value 设置的值
  1702. * @returns {any}
  1703. */
  1704. function setValue(key, value) {
  1705. yaohuo_userData[key] = value;
  1706. // setItem("yaohuo_userData", yaohuo_userData);
  1707. }
  1708. // 增加发帖ubb
  1709. function handleAddNewPostUBB() {
  1710. if (postPage.includes(window.location.pathname) && isAddNewPostUBB) {
  1711. let bookContent = document.getElementsByName("book_content")[0];
  1712. bookContent.insertAdjacentHTML(
  1713. "beforebegin",
  1714. `<div class='btBox'>
  1715. <div class='bt2'>
  1716. <a style='width:25%' id='ubb_url'>超链接</a>
  1717. <a style='width:25%' id='ubb_img'>图片</a>
  1718. <a style='width:25%' id='ubb_movie'>视频</a>
  1719. <a style='width:25%' id='ubb_more'">更多...</a>
  1720. </div>
  1721. </div>
  1722. <div class='more_ubb_tools' style='display: none'>
  1723. <div class='btBox'>
  1724. <div class='bt2'>
  1725. <a style='width:25%' id='ubb_color'>颜色</a>
  1726. <a style='width:25%' id='ubb_b'">加粗</a>
  1727. <a style='width:25%' id='ubb_strike'>删除</a>
  1728. <a style='width:25%' id='ubb_font'>字体</a>
  1729. </div>
  1730. </div>
  1731. <div class='btBox'>
  1732. <div class='bt2'>
  1733. <a style='width:25%' id='ubb_u'>下划线</a>
  1734. <a style='width:25%' id='ubb_i'>斜体</a>
  1735. <a style='width:25%' id='ubb_hr'>分割线</a>
  1736. <a style='width:25%' href='https://www.yaohuo.me/bbs/book_view_ubb.aspx?action=class&siteid=1000&classid=177&page=1&backurl=bbs%2fbook_view_add.aspx%3fsiteid%3d1000%26classid%3d177' target="_blank">UBB使用</a>
  1737. </div>
  1738. </div>
  1739. <div class='btBox'>
  1740. <div class='bt2'>
  1741. <a style='width:25%' id='ubb_audio'>音频</a>
  1742. <a style='width:25%' id='ubb_left'>居左</a>
  1743. <a style='width:25%' id='ubb_center'>居中</a>
  1744. <a style='width:25%' id='ubb_right'>居右</a>
  1745. </div>
  1746. </div>
  1747. <div class='btBox'>
  1748. <div class='bt2'>
  1749. <a style='width:25%' href='https://yaohuo.me/tuchuang/' target="_blank">妖火图床</a>
  1750. <a style='width:25%' href='https://img.ink/' target="_blank">水墨图床</a>
  1751. <a style='width:25%' href='https://aapi.eu.org/dy' target="_blank">抖音解析</a>
  1752. <a style='width:25%' href='https://aapi.eu.org/ks' target="_blank">快手解析</a>
  1753. </div>
  1754. </div>
  1755. <div class='btBox'>
  1756. <div class='bt2'>
  1757. <a style='width:25%' href='https://aapi.eu.org/ppx' target="_blank">皮皮虾解析</a>
  1758. <a style='width:25%' href='https://aapi.eu.org/bili' target="_blank">b站解析</a>
  1759. <a style='width:25%' href='https://pan.whgpc.com/upload.php' target="_blank">外链网盘</a>
  1760. <a style='width:25%' href='https://urlify.cn/' target="_blank">短链生成</a>
  1761. </div>
  1762. </div>
  1763. </div>`
  1764. );
  1765.  
  1766. addEventAry.forEach((item) => {
  1767. handleEventListener(item.id, bookContent, item.ubb, item.offset);
  1768. });
  1769. document.getElementById("ubb_more").addEventListener("click", () => {
  1770. let ubb_tool = document.getElementsByClassName("more_ubb_tools")[0];
  1771. ubb_tool.style.display =
  1772. ubb_tool.style.display === "none" ? "block" : "none";
  1773. });
  1774. }
  1775. }
  1776. // 增加回帖ubb
  1777. function handleAddReplyUBB() {
  1778. if (
  1779. (/^\/bbs-.*\.html$/.test(window.location.pathname) ||
  1780. viewPage.includes(window.location.pathname)) &&
  1781. isAddReplyUBB
  1782. ) {
  1783. const form = document.getElementsByName("f")[0];
  1784. if (!form) {
  1785. return;
  1786. }
  1787. const fileTag = document.querySelector(
  1788. "a[href^='/bbs/book_re_addfile.aspx']"
  1789. );
  1790.  
  1791. // 添加ubb展开按钮
  1792. fileTag.insertAdjacentHTML(
  1793. "afterend",
  1794. `<input id="ubb_unfold" type="submit" value="${
  1795. isUnfoldUbb ? "折叠UBB" : "展开UBB"
  1796. }" style="float:right"/>`
  1797. );
  1798.  
  1799. // 妖火图床、超链接、图片
  1800. form.insertAdjacentHTML(
  1801. "beforeend",
  1802. `
  1803. <hr>
  1804. <div class="ubb_wrap" style="text-align: center;overflow: hidden;">
  1805. <span id='ubb_url' style="${spanstyle}">链接</span>
  1806. <span id='ubb_img' style="${spanstyle}">图片</span>
  1807. <span id='ubb_audio' style="${spanstyle}">音频</span>
  1808. <span id='ubb_movie' style="${spanstyle}">视频</span>
  1809. <span id='ubb_nzgsa' style="${a2style}">你真该死啊</span>
  1810.  
  1811. <br>
  1812. <span id='ubb_text' style="${spanstyle}">半角</span>
  1813. <span id='ubb_br' style="${spanstyle}">换行</span>
  1814. <span id='ubb_b' style="${spanstyle}">加粗</span>
  1815. <span id='ubb_i' style="${spanstyle}">斜体</span>
  1816.  
  1817. <span id='ubb_color' style="${spanstyle}">颜色字</span>
  1818. <span id='ubb_u' style="${spanstyle}">下划线</span>
  1819. <span id='ubb_strike' style="${spanstyle}">删除线</span>
  1820. <span id='ubb_hr' style="${spanstyle}">分割线</span>
  1821. <br>
  1822. <span id='ubb_sms' style="${spanstyle}">短信</span>
  1823. <span id='ubb_call' style="${spanstyle}">拨号</span>
  1824. <span id='ubb_now' style="${spanstyle}">时间</span>
  1825. <span id='ubb_codo' style="${spanstyle}">倒计天</span>
  1826. <br>
  1827. <a href='https://yaohuo.me/tuchuang/' target="_blank" style="${spanstyle}">图床</a>
  1828. <a href='https://aapi.eu.org/ppx' target="_blank" style="${spanstyle}">皮皮</a>
  1829. <a href='https://aapi.eu.org/bili' target="_blank" style="${spanstyle}">b站</a>
  1830. <a href='https://aapi.eu.org/dy' target="_blank" style="${spanstyle}">抖音</a>
  1831. <a href='https://aapi.eu.org/ks' target="_blank" style="${spanstyle}">快手</a>
  1832. <a href='https://pan.whgpc.com/upload.php' target="_blank" style="${spanstyle}">外链</a>
  1833. <a href='https://urlify.cn/' target="_blank" style="${spanstyle}">短链接</a>
  1834. </div>
  1835. <hr>
  1836. `
  1837. );
  1838.  
  1839. // 处理默认展开ubb
  1840. if (isUnfoldUbb) {
  1841. $(".ubb_wrap").height("auto");
  1842. } else {
  1843. $(".ubb_wrap").height(32);
  1844. }
  1845.  
  1846. // 处理折叠ubb
  1847. $("#ubb_unfold").click(function (event) {
  1848. if (this.value == "折叠UBB") {
  1849. $(".ubb_wrap").height(32);
  1850. this.value = "展开UBB";
  1851. } else {
  1852. $(".ubb_wrap").height("auto");
  1853. this.value = "折叠UBB";
  1854. }
  1855. event.preventDefault();
  1856. });
  1857.  
  1858. // 超链接
  1859. const textarea = document.querySelector("textarea");
  1860. addEventAry.forEach((item) => {
  1861. handleEventListener(item.id, textarea, item.ubb, item.offset);
  1862. });
  1863. }
  1864. }
  1865. // 增加回帖表情
  1866. function handleAddReplyFace() {
  1867. if (
  1868. (/^\/bbs-.*\.html$/.test(window.location.pathname) ||
  1869. viewPage.includes(window.location.pathname)) &&
  1870. isAddReplyFace
  1871. ) {
  1872. const form = document.getElementsByName("f")[0];
  1873. if (!form) {
  1874. return;
  1875. }
  1876. const face = form.getElementsByTagName("select")[0];
  1877. const sendmsg = form.getElementsByTagName("select")[1];
  1878. const textarea = form.getElementsByTagName("textarea")[0];
  1879. // 显示表情
  1880. textarea.insertAdjacentHTML("beforebegin", '<div id="facearea"></div>');
  1881. const facearea = document.getElementById("facearea");
  1882.  
  1883. let allFaceHtml = "";
  1884. faceList.forEach((faceStr, i) => {
  1885. let name = faceStr.split(".")[0];
  1886. allFaceHtml += `
  1887. <img
  1888. id="setFace${i}"
  1889. style="width: 32px;height: 32px"
  1890. src="face/${faceStr}"
  1891. value="${name}.gif"
  1892. />`;
  1893. });
  1894. diyFaceList.forEach((item, i) => {
  1895. allFaceHtml += `
  1896. <img
  1897. id="diyFace${i}"
  1898. data-src="${item.url}"
  1899. style="width: 32px;height: 32px"
  1900. src="${item.url}"
  1901. value="${item.name}.gif"
  1902. />`;
  1903. });
  1904. facearea.innerHTML = allFaceHtml;
  1905.  
  1906. // 添加表情展开按钮
  1907. sendmsg.insertAdjacentHTML(
  1908. "afterend",
  1909. `<span
  1910. style="${spanstyle}display:${
  1911. isUnfoldFace ? "display: block" : "display: none"
  1912. }" id="unfold"
  1913. >表情${isUnfoldFace ? "折叠" : "展开"}</span>`
  1914. );
  1915.  
  1916. // 处理点击添加表情包
  1917. facearea.onclick = function (event) {
  1918. if (event.target.tagName.toLowerCase() === "img") {
  1919. // 自定义图片
  1920. let diySrc = event.target.dataset.src;
  1921.  
  1922. if (diySrc) {
  1923. //把光标移到文本框最前面
  1924. textarea.focus();
  1925. textarea.setSelectionRange(0, 0);
  1926. insertText(textarea, `[img]${diySrc}[/img]`, 0);
  1927. } else {
  1928. // 处理图片的点击事件
  1929. face.value = event.target.getAttribute("value");
  1930. }
  1931.  
  1932. // 处理完折叠表情
  1933. $("#facearea").hide();
  1934. $("#unfold").text("表情展开");
  1935. }
  1936. };
  1937. // 处理默认展开表情
  1938. if (isUnfoldFace) {
  1939. $("#facearea").show();
  1940. } else {
  1941. $("#facearea").hide();
  1942. }
  1943. // 处理折叠表情
  1944. $("#unfold").click(function (event) {
  1945. if (this.innerText == "表情展开") {
  1946. $("#facearea").show();
  1947. this.innerText = "表情折叠";
  1948. } else {
  1949. $("#facearea").hide();
  1950. this.innerText = "表情展开";
  1951. }
  1952. });
  1953. }
  1954. }
  1955. function handleReply() {
  1956. if (
  1957. (/^\/bbs-.*\.html$/.test(window.location.pathname) ||
  1958. viewPage.includes(window.location.pathname)) &&
  1959. (isAddReplyUBB || isAddReplyFace)
  1960. ) {
  1961. // 取消回复文本框粘性定位。
  1962. $(".sticky").addClass("add-position-static");
  1963.  
  1964. // 回复页不处理
  1965. if (!window.location.pathname.includes("/bbs/book_re.aspx")) {
  1966. document
  1967. .querySelector(".recontent")
  1968. .addEventListener("click", (event) => {
  1969. if (event.target.innerText === "回") {
  1970. // 如果是回复指定楼层就定位到回复输入框
  1971. if (
  1972. /回复\d+楼/.test(document.querySelector(".sticky b")?.innerText)
  1973. ) {
  1974. window.scrollTo(0, document.querySelector(".sticky").offsetTop);
  1975. }
  1976. }
  1977. });
  1978. }
  1979. }
  1980. }
  1981. function handleUploadImage() {
  1982. if (isUploadImage) {
  1983. let textArea = document.getElementsByTagName("textarea")[0];
  1984. if (!textArea) return;
  1985. // 排除游戏大厅页面
  1986. if (/^\/games\/\w+\/index\.aspx$/.test(window.location.pathname)) return;
  1987.  
  1988. let isReplyPage =
  1989. /^\/bbs-.*\.html$/.test(window.location.pathname) ||
  1990. viewPage.includes(window.location.pathname);
  1991. MY_addStyle(`
  1992. .upload-wrap {
  1993. position: relative;
  1994. display: inline-block;
  1995. width: 100%;
  1996. box-sizing: border-box;
  1997. height: 50px;
  1998. border: 2px dashed #ccc;
  1999. border-radius: 5px;
  2000. font-size: 16px;
  2001. color: #555;
  2002. text-align: center;
  2003. cursor: pointer;
  2004. transition: all 0.3s;
  2005. }
  2006. .upload-wrap-disabled{
  2007. background: #ddd;
  2008. cursor: not-allowed;
  2009. }
  2010. .upload-wrap:hover {
  2011. border-color: #aaa;
  2012. }
  2013. .upload-wrap:focus {
  2014. outline: none;
  2015. }
  2016. .upload-input-label {
  2017. width: 100%;
  2018. height: 100%;
  2019. font-weight: bold;
  2020. display: flex;
  2021. justify-content: center;
  2022. align-items: center;
  2023. }
  2024. .upload-loading {
  2025. box-sizing: border-box;
  2026. border: 6px solid #f3f3f3;
  2027. border-top: 6px solid #3498db;
  2028. border-radius: 50%;
  2029. width: 40px;
  2030. height: 40px;
  2031. animation: spin 2s linear infinite;
  2032. margin: auto;
  2033. position: absolute;
  2034. z-index: 10;
  2035. left: 50%;
  2036. top: 50%;
  2037. margin-top: -20px;
  2038. margin-left: -20px;
  2039. }
  2040. @keyframes spin {
  2041. 0% { transform: rotate(0deg); }
  2042. 100% { transform: rotate(360deg); }
  2043. }
  2044. `);
  2045. textArea.insertAdjacentHTML(
  2046. "afterend",
  2047. `<label for="upload-input" class="upload-wrap">
  2048. <div class="upload-loading" style="display: none"></div>
  2049. <span class="upload-input-label">
  2050. <svg t="1683636826356" style="margin-right: 10px" class="icon" fill="#16baaa" viewBox="0 0 1264 1024" version="1.1"
  2051. xmlns="http://www.w3.org/2000/svg" p-id="9231" width="38" height="38">
  2052. <path d="M992.171444 312.62966C975.189616 137.155482 827.415189 0 647.529412 0 469.849434 0 323.616239 133.860922 303.679205 306.210218 131.598564 333.839271 0 482.688318 0 662.588235c0 199.596576 161.815189 361.411765 361.411765 361.411765h184.014581V692.705882H294.530793l337.939795-361.411764 337.939796 361.411764H726.132229v331.294118H933.647059v-1.555371c185.470975-15.299199 331.294118-170.426291 331.294117-359.856394 0-168.969898-116.101408-310.367302-272.769732-349.958575z" p-id="9232"></path>
  2053. </svg>
  2054. 选择或拖拽图片上传到图床
  2055. </span>
  2056. <input
  2057. type="file"
  2058. multiple
  2059. id="upload-input"
  2060. accept="image/*"
  2061. style="display: none"
  2062. />
  2063. </label>`
  2064. );
  2065.  
  2066. // 获取上传图标的 input 元素
  2067. const uploadInput = document.querySelector("#upload-input");
  2068. const uploadWrap = document.querySelector(".upload-wrap");
  2069. const uploadLoading = document.querySelector(".upload-loading");
  2070.  
  2071. uploadInput.addEventListener("change", handleFileSelect);
  2072. uploadWrap.addEventListener("dragover", handleDragOver);
  2073. uploadWrap.addEventListener("drop", handleDrop);
  2074. textArea.addEventListener("paste", handlePaste);
  2075.  
  2076. // 剪贴板事件
  2077. async function handlePaste(event) {
  2078. const clipboardData =
  2079. event.clipboardData || event.originalEvent.clipboardData;
  2080. const items = clipboardData.items;
  2081.  
  2082. handleUploadStatus("start");
  2083. const files = [];
  2084.  
  2085. for (const item of items) {
  2086. if (item.type.indexOf("image") !== -1) {
  2087. const blob = item.getAsFile();
  2088. // paste 事件的处理程序是异步的,所以不能在这里直接上传,否则有多个只会读取第一个
  2089. // await uploadFile(blob);
  2090. files.push(blob);
  2091. }
  2092. }
  2093.  
  2094. // 此处处理上传
  2095. for (const item of files) {
  2096. await uploadFile(item);
  2097. }
  2098. handleUploadStatus("end");
  2099. }
  2100.  
  2101. // 上传事件
  2102. async function uploadFile(file) {
  2103. let uploadConfig = {
  2104. 水墨图床: {
  2105. url: "https://img.ink/api/upload",
  2106. name: "image",
  2107. token: inkToken || "",
  2108. },
  2109. 极速图床: {
  2110. url: "https://tucdn.wpon.cn/api/upload",
  2111. name: "image",
  2112. token: speedFreeToken || "",
  2113. },
  2114. 葫芦侠图床: {
  2115. url: "https://api.suyanw.cn/huluxia/upload.php",
  2116. name: "file",
  2117. },
  2118. };
  2119. let {
  2120. url: uploadUrl,
  2121. name: uploadName,
  2122. token: uploadToken,
  2123. } = uploadConfig[imageBedType];
  2124.  
  2125. let formData = new FormData();
  2126. formData.append(uploadName, file);
  2127. try {
  2128. let response;
  2129. if (imageBedType === "葫芦侠图床") {
  2130. response = await fetch(uploadUrl, {
  2131. method: "POST",
  2132. body: formData,
  2133. });
  2134. } else {
  2135. response = await fetch(uploadUrl, {
  2136. method: "POST",
  2137. headers: {
  2138. token: uploadToken || "",
  2139. },
  2140. body: formData,
  2141. });
  2142. }
  2143.  
  2144. const res = await response.json();
  2145.  
  2146. let { code, url, data, msg } = res;
  2147.  
  2148. if (code === 200 || code === 0 || url) {
  2149. // 处理葫芦侠图床直接取url,其他取data.url
  2150. if (!url) {
  2151. url = data.url;
  2152. }
  2153. if (url) {
  2154. // 如果是回帖页面把光标移到文本框最前面
  2155. if (isReplyPage) {
  2156. textArea.focus();
  2157. textArea.setSelectionRange(0, 0);
  2158. }
  2159.  
  2160. insertText(textArea, `[img]${url}[/img]`, 0);
  2161. }
  2162. } else {
  2163. alert(msg);
  2164. }
  2165. } catch (error) {
  2166. alert(error);
  2167. console.error("上传失败:", error);
  2168. }
  2169. }
  2170.  
  2171. // 选择文件change事件
  2172. async function handleFileSelect(event) {
  2173. const files = event.target.files;
  2174. handleUploadStatus("start");
  2175. for (const file of files) {
  2176. await uploadFile(file);
  2177. }
  2178. handleUploadStatus("end");
  2179. }
  2180. // 拖拽事件
  2181. function handleDragOver(event) {
  2182. event.stopPropagation();
  2183. event.preventDefault();
  2184. event.dataTransfer.dropEffect = "copy";
  2185. }
  2186.  
  2187. async function handleDrop(event) {
  2188. event.stopPropagation();
  2189. event.preventDefault();
  2190.  
  2191. const files = event.dataTransfer.files;
  2192. handleUploadStatus("start");
  2193. for (const file of files) {
  2194. if (file.type.indexOf("image") !== -1) {
  2195. await uploadFile(file);
  2196. }
  2197. }
  2198. handleUploadStatus("end");
  2199. }
  2200. /**
  2201. * 处理上传状态
  2202. * @param {'start' | 'end'} type 处理的状态
  2203. */
  2204. function handleUploadStatus(type) {
  2205. if (type === "start") {
  2206. uploadWrap.classList.toggle("upload-wrap-disabled");
  2207. uploadInput.disabled = true;
  2208. uploadLoading.style.display = "block";
  2209. }
  2210. if (type === "end") {
  2211. uploadWrap.classList.toggle("upload-wrap-disabled");
  2212. uploadInput.disabled = false;
  2213. uploadLoading.style.display = "none";
  2214. uploadInput.value = "";
  2215. }
  2216. }
  2217. }
  2218. }
  2219. // 处理404页面跳回新帖页面
  2220. function handleNotFoundPage() {
  2221. if (notFoundPage.includes(window.location.pathname)) {
  2222. history.go(-2);
  2223. // let year = new Date().getFullYear();
  2224. // location.href = `/bbs/book_list.aspx?gettotal=${year}&action=new`;
  2225. }
  2226. }
  2227. /**
  2228. * 删除过期的帖子
  2229. * @param {number|string} value 存储肉帖的对象
  2230. */
  2231. function deleteExpiredID(value) {
  2232. let nowTime = new Date().getTime();
  2233.  
  2234. Object.keys(value).forEach((key) => {
  2235. let lastTime = value[key];
  2236. if (nowTime > timeLeft(lastTime, expiredDays)) {
  2237. delete value[key];
  2238. }
  2239. });
  2240. }
  2241. // 获取用户等级
  2242. function handleShowUserLevel() {
  2243. if (!/^\/bbs-.*\.html$/.test(window.location.pathname) || !isShowLevel) {
  2244. return;
  2245. }
  2246.  
  2247. let user_id =
  2248. document.getElementsByClassName("subtitle")[0].firstElementChild.href;
  2249.  
  2250. function success(rp) {
  2251. let lv_zz = /<\/b>(\S*)级/;
  2252. let lv_text = rp.match(lv_zz)?.[1] || "0";
  2253. // console.log(lv_text);
  2254. addLvTip(lv_text);
  2255. }
  2256.  
  2257. function fail(code) {
  2258. console.log("error");
  2259. }
  2260.  
  2261. let request = new XMLHttpRequest();
  2262.  
  2263. request.onreadystatechange = function () {
  2264. if (request.readyState === 4) {
  2265. if (request.status === 200) {
  2266. return success(request.responseText);
  2267. } else {
  2268. return fail(request.status);
  2269. }
  2270. } else {
  2271. }
  2272. };
  2273. request.open("GET", user_id);
  2274. //request.responseType = 'document';
  2275. request.send();
  2276.  
  2277. function addLvTip(lv) {
  2278. let info_d = document.getElementsByClassName("subtitle")[0];
  2279. let user_name_d = info_d.children[1];
  2280. console.log(user_name_d);
  2281.  
  2282. let lv_d = document.createElement("div");
  2283. lv_d.innerText = "Lv " + lv;
  2284. lv_d.style =
  2285. "display:inline;margin-left:10px; text-align:center; margin-right:10px;color:#ff4234;font-size:17px;border-radius: 30px;";
  2286. info_d.insertBefore(lv_d, user_name_d);
  2287. }
  2288. }
  2289. function handleAddLoadMoreBtnClick() {
  2290. // 如果打开了全自动吃肉和自动加载更多,并且在帖子列表页才添加事件
  2291. let isPage = loadNextPage.some((item) =>
  2292. item.test(window.location.pathname)
  2293. );
  2294. if (isPage) {
  2295. let loadMoreBtn = null;
  2296. let nextPageWrap = document.querySelector(".bt2");
  2297. if (loadNextPageType === "nextPage" && nextPageWrap) {
  2298. loadMoreBtn = nextPageWrap.firstChild;
  2299. } else {
  2300. loadMoreBtn = document.querySelector("#KL_loadmore");
  2301. }
  2302. loadMoreBtn?.addEventListener("click", (e) => {
  2303. isClickLoadMoreBtn = true;
  2304. isNewPage = false;
  2305. });
  2306. }
  2307. }
  2308. // 自动加载下一页
  2309. function handleLoadNextPage() {
  2310. // 处理自动加载更多
  2311. let isPage = loadNextPage.some((item) =>
  2312. item.test(window.location.pathname)
  2313. );
  2314. if (isPage && isLoadNextPage) {
  2315. let nextBtn = null;
  2316. let nextPageWrap = document.querySelector(".bt2");
  2317. // 距离按钮最大多少就会触发
  2318. let bottomMaxDistance = 250;
  2319. if (loadNextPageType === "more" || !nextPageWrap) {
  2320. nextBtn = document.querySelector("span[id$=show_tip]");
  2321. } else {
  2322. nextBtn = nextPageWrap.firstChild;
  2323. bottomMaxDistance = 150;
  2324. }
  2325. let A = nextBtn.getBoundingClientRect().bottom;
  2326. let B = document.documentElement.clientHeight;
  2327. // 获取当前列表的长度
  2328. let newLength = getListLength();
  2329.  
  2330. // 加载更多按钮距离距底部小于300px才开始加载
  2331. // 没有加载完成前不会再次加载
  2332. // 小于页面最大加载数量才会加载
  2333. if (
  2334. A <= B + bottomMaxDistance &&
  2335. !isClickLoadMoreBtn &&
  2336. newLength < maxLoadNum
  2337. ) {
  2338. nextBtn.click();
  2339. // 放到加载更多按钮里面监听,此处不处理
  2340. // isClickLoadMoreBtn = true;
  2341. // isNewPage = false;
  2342. }
  2343. }
  2344. }
  2345. /**
  2346. * 节流函数
  2347. * @param {function} fn - 要节流的函数
  2348. * @param {number} interval - 节流时间间隔(毫秒)
  2349. * @param {Object} options - 选项对象
  2350. * @param {boolean} [options.leading=true] - 是否在开始时立即执行一次函数
  2351. * @param {boolean} [options.trailing=isExecTrail] - 是否在结束时再执行一次函数
  2352. * @returns {function} 节流后的函数
  2353. */
  2354. function throttle(
  2355. fn,
  2356. interval,
  2357. { leading = true, trailing = isExecTrail } = {}
  2358. ) {
  2359. let startTime = 0;
  2360. let timer = null;
  2361.  
  2362. const _throttle = function (...args) {
  2363. return new Promise((resolve, reject) => {
  2364. try {
  2365. // 1.获取当前时间
  2366. const nowTime = new Date().getTime();
  2367.  
  2368. // 对立即执行进行控制
  2369. if (!leading && startTime === 0) {
  2370. startTime = nowTime;
  2371. }
  2372.  
  2373. // 2.计算需要等待的时间执行函数
  2374. const waitTime = interval - (nowTime - startTime);
  2375. if (waitTime <= 0) {
  2376. // console.log("执行操作fn")
  2377. if (timer) clearTimeout(timer);
  2378. const res = fn.apply(this, args);
  2379. resolve(res);
  2380. startTime = nowTime;
  2381. timer = null;
  2382. return;
  2383. }
  2384.  
  2385. // 3.判断是否需要执行尾部
  2386. if (trailing && !timer) {
  2387. timer = setTimeout(() => {
  2388. // console.log("执行timer")
  2389. const res = fn.apply(this, args);
  2390. resolve(res);
  2391. startTime = new Date().getTime();
  2392. timer = null;
  2393. }, waitTime);
  2394. }
  2395. } catch (error) {
  2396. reject(error);
  2397. }
  2398. });
  2399. };
  2400.  
  2401. _throttle.cancel = function () {
  2402. if (timer) clearTimeout(timer);
  2403. startTime = 0;
  2404. timer = null;
  2405. };
  2406.  
  2407. return _throttle;
  2408. }
  2409. /**
  2410. * 添加事件监听
  2411. * @param {string} id dom元素id
  2412. * @param {HTMLElement} textarea 插入的文本框
  2413. * @param {string} ubb 插入的内容
  2414. * @param {number} offset 插入的位置
  2415. */
  2416. function handleEventListener(id, textarea, ubb, offset) {
  2417. document.getElementById(id)?.addEventListener("click", (e) => {
  2418. if (id === "ubb_nzgsa") {
  2419. // if (textarea.value !== "") {
  2420. insertText(
  2421. textarea,
  2422. "[audio=X]https://file.uhsea.com/2304/3deb45e90564252bf281f47c7b47a153KJ.mp3[/audio]",
  2423. 0
  2424. );
  2425. // } else {
  2426. // alert("不要无意义灌水啦!");
  2427. // }
  2428. } else {
  2429. e.preventDefault();
  2430. insertText(textarea, ubb, offset);
  2431. }
  2432. });
  2433. }
  2434. /**
  2435. * 简易版jquery实现,用于替换之前写的部分语法,不引用cdn库
  2436. * @returns
  2437. */
  2438. function myJquery() {
  2439. let jQuery = function (selector) {
  2440. return new jQuery.fn.init(selector);
  2441. };
  2442.  
  2443. jQuery.fn = jQuery.prototype = {
  2444. constructor: jQuery,
  2445.  
  2446. init: function (selector) {
  2447. if (!selector) {
  2448. return this;
  2449. }
  2450.  
  2451. if (typeof selector === "string") {
  2452. let elements = document.querySelectorAll(selector);
  2453. this.length = elements.length;
  2454. for (let i = 0; i < elements.length; i++) {
  2455. this[i] = elements[i];
  2456. }
  2457. } else if (selector.nodeType) {
  2458. this[0] = selector;
  2459. this.length = 1;
  2460. } else if (selector instanceof jQuery) {
  2461. return selector;
  2462. } else if (Array.isArray(selector)) {
  2463. for (let i = 0; i < selector.length; i++) {
  2464. this[i] = selector[i];
  2465. }
  2466. this.length = selector.length;
  2467. return this;
  2468. }
  2469.  
  2470. return this;
  2471. },
  2472.  
  2473. length: 0,
  2474.  
  2475. each: function (callback) {
  2476. for (let i = 0; i < this.length; i++) {
  2477. callback.call(this[i], i, this[i]);
  2478. }
  2479.  
  2480. return this;
  2481. },
  2482.  
  2483. css: function (prop, value) {
  2484. if (typeof prop === "string") {
  2485. if (value !== undefined) {
  2486. this.each(function () {
  2487. this.style[prop] = value;
  2488. });
  2489. return this;
  2490. } else {
  2491. return this[0].style[prop];
  2492. }
  2493. } else {
  2494. for (let key in prop) {
  2495. this.each(function () {
  2496. this.style[key] = prop[key];
  2497. });
  2498. }
  2499. return this;
  2500. }
  2501. },
  2502.  
  2503. text: function (text) {
  2504. if (text !== undefined) {
  2505. this.each(function () {
  2506. this.textContent = text;
  2507. });
  2508. return this;
  2509. } else {
  2510. let result = "";
  2511. this.each(function () {
  2512. result += this.textContent;
  2513. });
  2514. return result;
  2515. }
  2516. },
  2517.  
  2518. html: function (html) {
  2519. if (html !== undefined) {
  2520. this.each(function () {
  2521. this.innerHTML = html;
  2522. });
  2523. return this;
  2524. } else {
  2525. return this[0].innerHTML;
  2526. }
  2527. },
  2528.  
  2529. append: function (content) {
  2530. if (typeof content === "string") {
  2531. this.each(function () {
  2532. this.insertAdjacentHTML("beforeend", content);
  2533. });
  2534. } else if (content.nodeType) {
  2535. this.each(function () {
  2536. this.appendChild(content);
  2537. });
  2538. } else if (content instanceof jQuery) {
  2539. this.each(function () {
  2540. let self = this;
  2541. content.each(function () {
  2542. self.appendChild(this);
  2543. });
  2544. });
  2545. }
  2546.  
  2547. return this;
  2548. },
  2549.  
  2550. addClass: function (className) {
  2551. let classNames = className.split(" ");
  2552. this.each(function () {
  2553. for (let i = 0; i < classNames.length; i++) {
  2554. if (this.classList) {
  2555. this.classList.add(classNames[i]);
  2556. } else {
  2557. let currentClasses = this.className.split(" ");
  2558. if (currentClasses.indexOf(classNames[i]) === -1) {
  2559. this.className += " " + classNames[i];
  2560. }
  2561. }
  2562. }
  2563. });
  2564.  
  2565. return this;
  2566. },
  2567.  
  2568. removeClass: function (className) {
  2569. let classNames = className.split(" ");
  2570. this.each(function () {
  2571. for (let i = 0; i < classNames.length; i++) {
  2572. if (this.classList) {
  2573. this.classList.remove(classNames[i]);
  2574. } else {
  2575. let currentClasses = this.className.split(" ");
  2576. let index = currentClasses.indexOf(classNames[i]);
  2577. if (index !== -1) {
  2578. currentClasses.splice(index, 1);
  2579. this.className = currentClasses.join(" ");
  2580. }
  2581. }
  2582. }
  2583. });
  2584.  
  2585. return this;
  2586. },
  2587.  
  2588. show: function () {
  2589. this.each(function () {
  2590. // 恢复元素之前的display属性
  2591. let classDisplay = getComputedStyle(this).getPropertyValue("display");
  2592. let display =
  2593. this.getAttribute("data-display") ||
  2594. (classDisplay === "none" ? "block" : classDisplay);
  2595. this.style.display = display ? display : "";
  2596. });
  2597.  
  2598. return this;
  2599. },
  2600.  
  2601. hide: function () {
  2602. this.each(function () {
  2603. // 记住元素之前的display属性
  2604. let display =
  2605. this.style.display ||
  2606. getComputedStyle(this).getPropertyValue("display");
  2607. if (display !== "none") {
  2608. this.setAttribute("data-display", display);
  2609. }
  2610. this.style.display = "none";
  2611. });
  2612.  
  2613. return this;
  2614. },
  2615.  
  2616. click: function (callback) {
  2617. this.each(function () {
  2618. this.addEventListener("click", callback);
  2619. });
  2620.  
  2621. return this;
  2622. },
  2623.  
  2624. on: function (event, childSelector, data, handler) {
  2625. if (typeof childSelector === "function") {
  2626. handler = childSelector;
  2627. childSelector = null;
  2628. data = null;
  2629. } else if (typeof data === "function") {
  2630. handler = data;
  2631. data = null;
  2632. }
  2633.  
  2634. this.each(function () {
  2635. let element = this;
  2636.  
  2637. let listener = function (e) {
  2638. let target = e.target;
  2639. if (
  2640. !childSelector ||
  2641. element.querySelector(childSelector) === target
  2642. ) {
  2643. handler.call(target, e, data);
  2644. }
  2645. };
  2646.  
  2647. event.split(" ").forEach(function (type) {
  2648. element.addEventListener(type, listener);
  2649. });
  2650. });
  2651.  
  2652. return this;
  2653. },
  2654.  
  2655. prev: function () {
  2656. let prevElement = null;
  2657. this.each(function () {
  2658. prevElement = this.previousElementSibling;
  2659. });
  2660.  
  2661. return jQuery(prevElement);
  2662. },
  2663.  
  2664. next: function () {
  2665. let nextElement = null;
  2666. this.each(function () {
  2667. nextElement = this.nextElementSibling;
  2668. });
  2669.  
  2670. return new jQuery(nextElement);
  2671. },
  2672.  
  2673. children: function (selector) {
  2674. let childElements = [];
  2675. this.each(function () {
  2676. let children = this.children;
  2677. for (let i = 0; i < children.length; i++) {
  2678. if (!selector || children[i].matches(selector)) {
  2679. childElements.push(children[i]);
  2680. }
  2681. }
  2682. });
  2683.  
  2684. return jQuery(childElements);
  2685. },
  2686.  
  2687. parent: function (selector) {
  2688. let parentElements = [];
  2689. this.each(function () {
  2690. let parent = this.parentElement;
  2691. if (!selector || parent.matches(selector)) {
  2692. parentElements.push(parent);
  2693. }
  2694. });
  2695. return jQuery(parentElements);
  2696. },
  2697. closest: function (selector) {
  2698. var result = [];
  2699.  
  2700. this.each(function () {
  2701. var closestElement = this.closest(selector);
  2702.  
  2703. if (closestElement) {
  2704. result.push(closestElement);
  2705. }
  2706. });
  2707.  
  2708. return new jQuery(result);
  2709. },
  2710.  
  2711. prop: function (name, value) {
  2712. if (value === undefined) {
  2713. let element = this[0] || {};
  2714. return element[name];
  2715. } else {
  2716. this.each(function () {
  2717. this[name] = value;
  2718. });
  2719.  
  2720. return this;
  2721. }
  2722. },
  2723.  
  2724. remove: function () {
  2725. this.each(function () {
  2726. this.parentElement.removeChild(this);
  2727. });
  2728.  
  2729. return this;
  2730. },
  2731.  
  2732. height: function (value) {
  2733. if (value === undefined) {
  2734. if (this[0]) {
  2735. return this[0].clientHeight;
  2736. } else {
  2737. return null;
  2738. }
  2739. } else {
  2740. this.each(function () {
  2741. this.style.height = isNaN(value) ? value : value + "px";
  2742. });
  2743. return this;
  2744. }
  2745. },
  2746. };
  2747.  
  2748. jQuery.fn.init.prototype = jQuery.fn;
  2749.  
  2750. return jQuery;
  2751. }
  2752. })();

QingJ © 2025

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