深色模式

设置页面为深色模式, 可定时开关

  1. // ==UserScript==
  2. // @name 深色模式
  3. // @namespace https://gf.qytechs.cn/zh-CN/users/1196880-ling2ling4
  4. // @version 1.3.1
  5. // @author Ling2Ling4
  6. // @description 设置页面为深色模式, 可定时开关
  7. // @license AGPL-3.0-or-later
  8. // @icon 
  9. // @match *://*/*
  10. // @run-at document-start
  11. // @grant GM_setValue
  12. // @grant GM_getValue
  13. // @grant GM_registerMenuCommand
  14. // @grant GM_notification
  15. // @noframes
  16. // @compatible chrome
  17. // @compatible edge
  18. // @compatible firefox
  19. // ==/UserScript==
  20.  
  21. (() => {
  22. "use strict";
  23. function verify_time1(newVal, oldVal, base) {
  24. const arr = newVal.trim().split(/:|:/);
  25. if (2 === arr.length && 2 === arr[0].length && 2 === arr[1].length) {
  26. const a = +arr[0],
  27. b = +arr[1];
  28. if (a >= 0 && a <= 24 && b >= 0 && b <= 59) return newVal;
  29. }
  30. return oldVal;
  31. }
  32. function getNumVerifyFn(min, max, rangeLimit = [1, 1]) {
  33. return (newVal, oldVal, base) => {
  34. if (!(newVal = +newVal) && 0 !== newVal) return oldVal;
  35. if (!1 !== min && !1 !== max) {
  36. if (rangeLimit[0] && newVal >= min) {
  37. if (rangeLimit[1] && newVal <= max) return newVal;
  38. if (!rangeLimit[1] && newVal < max) return newVal;
  39. }
  40. if (!rangeLimit[0] && newVal > min) {
  41. if (rangeLimit[1] && newVal <= max) return newVal;
  42. if (!rangeLimit[1] && newVal < max) return newVal;
  43. }
  44. } else {
  45. if (!1 === min) {
  46. if (rangeLimit[1] && newVal <= max) return newVal;
  47. if (!rangeLimit[1] && newVal < max) return newVal;
  48. }
  49. if (!1 === max) {
  50. if (rangeLimit[0] && newVal >= min) return newVal;
  51. if (!rangeLimit[0] && newVal > min) return newVal;
  52. }
  53. }
  54. return oldVal;
  55. };
  56. }
  57. const keyBase = "ll_pageDarkMode_",
  58. info = {
  59. keyBase,
  60. settingsArea: null,
  61. isDarkMode: !1,
  62. isCanRun: !0,
  63. timer: null,
  64. interval: 5e3,
  65. cssDom: null,
  66. btnHoverTxt: "点击切换深色模式",
  67. otherSettings: {
  68. oldDarkMode: {
  69. value: !1,
  70. base: !1,
  71. key: keyBase + "oldDarkMode",
  72. valType: "boolean",
  73. },
  74. },
  75. settings: {
  76. btnPosition: {
  77. value: !1,
  78. base: !1,
  79. key: keyBase + "btnPosition",
  80. groupTitle3: "按钮设置",
  81. desc: "模式切换按钮的位置",
  82. type: "基础设置",
  83. valType: "boolean",
  84. compType: "radio",
  85. valueText: { true: "左下", false: "右下" },
  86. },
  87. isHiddenBtn: {
  88. value: !1,
  89. base: !1,
  90. key: keyBase + "isHiddenBtn",
  91. desc: "是否隐藏按钮, 隐藏后鼠标移入时会重新显示",
  92. type: "基础设置",
  93. valType: "boolean",
  94. compType: "radio",
  95. valueText: { true: "隐藏", false: "显示" },
  96. },
  97. btnSize: {
  98. value: "30",
  99. base: "30",
  100. key: keyBase + "btnSize",
  101. desc: "按钮的大小",
  102. type: "基础设置",
  103. valType: "number",
  104. compType: "textarea",
  105. verify: getNumVerifyFn(20, 60),
  106. },
  107. isAutoStartStop: {
  108. value: !0,
  109. base: !0,
  110. key: keyBase + "isAutoStartStop",
  111. groupTitle3: "定时开关",
  112. desc: "是否开启定时开关功能",
  113. type: "基础设置",
  114. valType: "boolean",
  115. compType: "radio",
  116. valueText: { true: "开启", false: "关闭" },
  117. },
  118. startTime: {
  119. value: "0",
  120. base: "0",
  121. key: keyBase + "startTime",
  122. valType: "string",
  123. type: "基础设置",
  124. desc: "深色模式的自动开启时间, 0表示关闭, 按照24小时制书写, 格式为 xx:xx, 如: 20:00",
  125. compType: "textarea",
  126. verify: (newVal, oldVal, base) =>
  127. 0 == +newVal ? newVal : verify_time1(newVal, oldVal),
  128. },
  129. stopTime: {
  130. value: "0",
  131. base: "0",
  132. key: keyBase + "stopTime",
  133. valType: "string",
  134. type: "基础设置",
  135. desc: "深色模式的自动关闭时间, 0表示关闭, 按照24小时制书写",
  136. compType: "textarea",
  137. verify: (newVal, oldVal, base) =>
  138. 0 == +newVal ? newVal : verify_time1(newVal, oldVal),
  139. },
  140. startStopWay: {
  141. value: !0,
  142. base: !0,
  143. key: keyBase + "startStopWay",
  144. desc: "定时开关深色模式的方式",
  145. type: "基础设置",
  146. valType: "boolean",
  147. compType: "radio",
  148. valueText: {
  149. true: "仅在设定时刻进行开关",
  150. false: "根据设定时间段任意时刻都可开关",
  151. },
  152. },
  153. isShowTips: {
  154. value: !0,
  155. base: !0,
  156. key: keyBase + "isShowTips",
  157. desc: "定时开关时是否进行弹窗提示",
  158. type: "基础设置",
  159. valType: "boolean",
  160. compType: "radio",
  161. valueText: { true: "弹窗提示", false: "关闭弹窗" },
  162. },
  163. onlyColor: {
  164. value: !1,
  165. base: !1,
  166. key: keyBase + "onlyColor",
  167. desc: "深色模式下是否仅调整颜色而不使页面变成深色 (此时'颜色反转'设置将失效)",
  168. type: "颜色设置",
  169. valType: "boolean",
  170. compType: "radio",
  171. valueText: { true: "自定义颜色", false: "深色+自定义颜色" },
  172. },
  173. invert: {
  174. value: 1,
  175. base: 1,
  176. key: keyBase + "invert",
  177. valType: "number",
  178. type: "颜色设置",
  179. title: "颜色反转",
  180. desc: "颜色反转的程度, 深色效果主要与该设置相关. 默认1, 浏览器默认0, 取值范围0-1",
  181. compType: "textarea",
  182. verify: getNumVerifyFn(0, 1),
  183. },
  184. brightness: {
  185. value: 0.9,
  186. base: 0.9,
  187. key: keyBase + "brightness",
  188. valType: "number",
  189. type: "颜色设置",
  190. title: "亮度",
  191. desc: "亮度的大小. 默认0.9, 浏览器默认1, 取值范围0-∞",
  192. compType: "textarea",
  193. verify: getNumVerifyFn(0, !1),
  194. },
  195. contrast: {
  196. value: 1,
  197. base: 1,
  198. key: keyBase + "contrast",
  199. valType: "number",
  200. type: "颜色设置",
  201. title: "对比度",
  202. desc: "对比度的强弱. 默认1, 取值范围0-∞",
  203. compType: "textarea",
  204. verify: getNumVerifyFn(0, !1),
  205. },
  206. grayscale: {
  207. value: 0,
  208. base: 0,
  209. key: keyBase + "grayscale",
  210. valType: "number",
  211. type: "颜色设置",
  212. title: "灰度",
  213. desc: "灰度的程度. 默认0, 取值范围0-1",
  214. compType: "textarea",
  215. verify: getNumVerifyFn(0, 1),
  216. },
  217. hueRotate: {
  218. value: 0,
  219. base: 0,
  220. key: keyBase + "hueRotate",
  221. valType: "number",
  222. type: "颜色设置",
  223. title: "色调",
  224. desc: "色调的旋转变化. 默认0, 取值范围0-360",
  225. compType: "textarea",
  226. verify: getNumVerifyFn(0, 360),
  227. },
  228. saturate: {
  229. value: 1,
  230. base: 1,
  231. key: keyBase + "saturate",
  232. valType: "number",
  233. type: "颜色设置",
  234. title: "饱和度",
  235. desc: "饱和度的高低. 默认1, 取值范围0-∞",
  236. compType: "textarea",
  237. verify: getNumVerifyFn(0, !1),
  238. },
  239. sepia: {
  240. value: 0.2,
  241. base: 0.2,
  242. key: keyBase + "sepia",
  243. valType: "number",
  244. type: "颜色设置",
  245. title: "深褐色",
  246. desc: "深褐色的程度. 默认0.2, 浏览器默认0, 取值范围0-1",
  247. compType: "textarea",
  248. verify: getNumVerifyFn(0, 1),
  249. },
  250. autoDarkMode: {
  251. value: !0,
  252. base: !0,
  253. key: keyBase + "autoDarkMode",
  254. desc: "'刷新页面/打开新页面'后是否自动恢复页面的深色模式",
  255. type: "其他设置",
  256. valType: "boolean",
  257. compType: "radio",
  258. valueText: { true: "自动恢复", false: "手动开关" },
  259. groupTitle3: "自动恢复",
  260. },
  261. autoDarkModeWay: {
  262. value: !0,
  263. base: !0,
  264. key: keyBase + "autoDarkModeWay",
  265. title: "自动恢复显示模式的方式",
  266. desc: "左选项: 可使同一时间段内打开的每个页面都是相同显示模式\n右选项: 可使同一个页面打开后是上一次该页面的显示模式",
  267. type: "其他设置",
  268. valType: "boolean",
  269. compType: "radio",
  270. valueText: {
  271. true: "恢复上一次使用的显示模式",
  272. false: "恢复当前网页上一次的显示模式",
  273. },
  274. },
  275. website: {
  276. value:
  277. "*www.baidu.com*\n*www.bilibili.com*\n*message.bilibili.com*\n*space.bilibili.com*\n*weibo.com*\n*www.zhihu.com*\n*www.douyin.com*",
  278. base: "*www.baidu.com*\n*www.bilibili.com*\n*message.bilibili.com*\n*space.bilibili.com*\n*weibo.com*\n*www.zhihu.com*\n*www.douyin.com*",
  279. key: keyBase + "website",
  280. valType: "string",
  281. type: "其他设置",
  282. title: "应用的网站",
  283. desc: "以下网站可启用深色模式, 支持*通配符, 多个网站请换行书写, 仅书写*表示所有网站都可启用\n【示例】*www.bilibili.com* 可匹配B站",
  284. compType: "textarea",
  285. compH: "110px",
  286. },
  287. onlyColorWebsite: {
  288. value: "",
  289. base: "",
  290. key: keyBase + "onlyColorWebsite",
  291. valType: "string",
  292. type: "其他设置",
  293. title: "不变为深色的网站",
  294. desc: '以下网站即使启用深色模式后也不会变为深色, 而是采用"自定义颜色"模式, 支持*通配符, 多个网站请换行书写',
  295. compType: "textarea",
  296. compH: "110px",
  297. },
  298. noneInvertNodes: {
  299. value:
  300. '// B站\n.h .h-inner, .h-inner .avatar-container, bili-user-profile, .bili-im .avatar, .owner .to-top, #bilibili-player [role="comment"],\n// 百度\n#content_left h3.t, .cr-content [class*="opr-toplist"], .cr-content [class*="tag-common"]',
  301. base: '// B站\n.h .h-inner, .h-inner .avatar-container, bili-user-profile, .bili-im .avatar, .owner .to-top, #bilibili-player [role="comment"],\n// 百度\n#content_left h3.t, .cr-content [class*="opr-toplist"], .cr-content [class*="tag-common"]',
  302. key: keyBase + "noneInvertNodes",
  303. valType: "string",
  304. type: "其他设置",
  305. title: "不反转的元素",
  306. desc: "不进行颜色反转的元素, 每项用 , 分隔, 可书写css选择器. 以//开头表示行注释\n【可选】\nh1, h2, h3, h4, p, span, ul, li, i, svg, a, img, input, textarea, button, select, option, label, audio, video, ....",
  307. compType: "textarea",
  308. compH: "110px",
  309. verify: (newVal) => {
  310. "," ===
  311. (newVal = newVal.trim().replaceAll(",", ","))[
  312. newVal.length - 1
  313. ] && (newVal = newVal.slice(0, -1));
  314. return newVal
  315. .split("\n")
  316. .map((item, i) => {
  317. const t = item;
  318. return (
  319. (item = item.trim()),
  320. 0 === i
  321. ? "/" === item[0] && "/" === item[1]
  322. ? item
  323. : t
  324. : "/" === item[0] && "/" === item[1]
  325. ? ",\n" + item
  326. : "\n" + t
  327. );
  328. })
  329. .join("")
  330. .replaceAll(", ,", ",")
  331. .replaceAll(",,", ",");
  332. },
  333. },
  334. },
  335. };
  336. function getCssHtml(isDark) {
  337. const settings = info.settings,
  338. r = parseInt(settings.btnSize.value / 5),
  339. btnCss = `#${info.keyBase}btn{\nbackground:#ffffff;padding:${
  340. r - 2
  341. }px;border-radius:${r}px;position:fixed;${
  342. settings.btnPosition.value ? "left" : "right"
  343. }:-${
  344. settings.btnSize.value / 2
  345. }px;bottom:20px;z-index:1000;transition:ease 0.3s all,ease 0.5s 2s opacity;cursor:pointer;box-sizing:border-box;\n${
  346. settings.isHiddenBtn.value ? "opacity:0" : ""
  347. }}\n#${info.keyBase}btn.dark{background:#ababab}\n#${
  348. info.keyBase
  349. }btn:hover {${
  350. settings.btnPosition.value ? "left:0" : "right:0"
  351. };bottom:20px;\n${
  352. settings.isHiddenBtn.value
  353. ? "transition:ease 0.3s opacity;opacity:1"
  354. : ""
  355. }\n}\n #${info.keyBase}btn svg{display:block;fill:#addeee}\n #${
  356. info.keyBase
  357. }btn.dark svg{fill:#ffffff}`;
  358. if (!isDark) return btnCss;
  359. const onlyColorflag =
  360. verifyWebsite(settings.onlyColorWebsite.value) ||
  361. settings.onlyColor.value,
  362. invertText = onlyColorflag ? "" : `invert(${settings.invert.value})`,
  363. selectorArr = settings.noneInvertNodes.value
  364. .split("\n")
  365. .filter((item) => "/" !== (item = item.trim())[0] && "/" !== item[1]),
  366. nonoFilter = onlyColorflag
  367. ? ""
  368. : 'img,video,input,iframe,canvas,object,svg image,\n[style*="background:url"],\n[style*="background: url"],\n[style*="background-image:url"],\n[style*="background-image: url"],\n[background]{filter:invert(1)}',
  369. otherCss = onlyColorflag
  370. ? ""
  371. : `${selectorArr.join("")}{filter:invert(1)}`;
  372. return `html {\nbackground-color:#fff;\nfilter:${invertText} brightness(${settings.brightness.value}) contrast(${settings.contrast.value}) grayscale(${settings.grayscale.value}) hue-rotate(${settings.hueRotate.value}deg) saturate(${settings.saturate.value}) sepia(${settings.sepia.value});\n}\n${nonoFilter}${otherCss}${btnCss}`;
  373. }
  374. function matchUrlWithWildcard(url, pattern) {
  375. return new RegExp("^" + pattern.replace(/\*/g, ".*") + "$").test(url);
  376. }
  377. function setValue_setValue({
  378. value,
  379. base,
  380. key,
  381. verification = null,
  382. getValue = null,
  383. setValue = null,
  384. getVal = null,
  385. setVal = null,
  386. } = {}) {
  387. getValue && (getVal = getValue), setValue && (setVal = setValue);
  388. let newVal = value,
  389. oldVal = getVal ? getVal(key) : localStorage.getItem(key);
  390. return (
  391. void 0 !== base &&
  392. null == oldVal &&
  393. ((oldVal = base),
  394. "string" != typeof base && (base = JSON.stringify(base)),
  395. setVal ? setVal(key, base) : localStorage.setItem(key, base)),
  396. null !== newVal &&
  397. ("function" != typeof verification ||
  398. ((newVal = verification(newVal, oldVal, base)), null !== newVal)) &&
  399. newVal !== oldVal &&
  400. ("string" != typeof newVal && (newVal = JSON.stringify(newVal)),
  401. setVal ? setVal(key, newVal) : localStorage.setItem(key, newVal),
  402. !0)
  403. );
  404. }
  405. function getValue({
  406. base,
  407. key,
  408. valType = "string",
  409. isReSet = !0,
  410. getValue = null,
  411. setValue = null,
  412. getVal = null,
  413. setVal = null,
  414. } = {}) {
  415. getValue && (getVal = getValue), setValue && (setVal = setValue);
  416. let val = getVal ? getVal(key) : localStorage.getItem(key);
  417. return (
  418. void 0 !== base &&
  419. null == val &&
  420. ((val = base),
  421. isReSet &&
  422. ("string" != typeof base && (base = JSON.stringify(base)),
  423. setVal ? setVal(key, base) : localStorage.setItem(key, base))),
  424. (valType = valType.toLowerCase()),
  425. "string" == typeof val
  426. ? "string" === valType
  427. ? val
  428. : "boolean" === valType || "number" === valType
  429. ? JSON.parse(val)
  430. : "object" === valType
  431. ? val
  432. ? JSON.parse(val)
  433. : {}
  434. : "array" === valType
  435. ? val
  436. ? JSON.parse(val)
  437. : []
  438. : val
  439. : val
  440. );
  441. }
  442. function getData(settings, getVal = null, setVal = null) {
  443. (getVal = getVal || localStorage.getItem),
  444. (setVal = setVal || localStorage.setItem);
  445. for (const valName in settings) {
  446. const setting = settings[valName];
  447. setting.value = getValue({
  448. base: setting.base,
  449. key: setting.key,
  450. valType: setting.valType,
  451. getVal,
  452. setVal,
  453. });
  454. }
  455. return settings;
  456. }
  457. function setDarkMode(isDark = !0, modeTxt = "") {
  458. if (
  459. (((isDark) => {
  460. let dom = info.cssDom,
  461. isAdd = !1;
  462. if (!dom) {
  463. const id = info.keyBase + "css";
  464. (dom = document.head.querySelector("#" + id)),
  465. dom ||
  466. ((dom = document.createElement("style")),
  467. (dom.id = id),
  468. (info.cssDom = dom),
  469. (isAdd = !0));
  470. }
  471. dom.isDark !== isDark &&
  472. ((dom.innerHTML = getCssHtml(isDark)),
  473. (info.isDarkMode = isDark),
  474. (dom.isDark = isDark),
  475. isAdd && document.head.appendChild(dom));
  476. })(isDark),
  477. document.body)
  478. )
  479. document.body.appendChild(info.cssDom);
  480. else {
  481. const bodyObserver = new MutationObserver(() => {
  482. document.body &&
  483. (bodyObserver.disconnect(), document.body.appendChild(info.cssDom));
  484. });
  485. bodyObserver.observe(document, { childList: !0, subtree: !0 });
  486. }
  487. let logText, txt1;
  488. txt1 = isDark ? "开启" : "关闭";
  489. const settings = info.settings;
  490. modeTxt ||
  491. (settings.onlyColor.value &&
  492. (modeTxt = `'${settings.onlyColor.valueText.true}'模式`),
  493. verifyWebsite(settings.onlyColorWebsite.value) &&
  494. (modeTxt = `'${settings.onlyColor.valueText.true}'模式`)),
  495. (logText = `${txt1}${(modeTxt = modeTxt || "深色模式")}`),
  496. console.log(logText),
  497. setValue_setValue({
  498. value: isDark,
  499. base: info.otherSettings.oldDarkMode.base,
  500. key: info.otherSettings.oldDarkMode.key,
  501. getValue: GM_getValue,
  502. setValue: GM_setValue,
  503. }),
  504. setValue_setValue({
  505. value: isDark,
  506. base: info.otherSettings.oldDarkMode.base,
  507. key: info.otherSettings.oldDarkMode.key,
  508. });
  509. }
  510. function setStartStopTimer() {
  511. const settings = info.settings;
  512. if (!settings.isAutoStartStop.value) return;
  513. const startTime = settings.startTime.value,
  514. stopTime = settings.stopTime.value;
  515. if (0 == +startTime && 0 == +stopTime) return;
  516. const autoStartStop = () => {
  517. const f = (function isNeedDarkMode() {
  518. const settings = info.settings,
  519. startTime = settings.startTime.value,
  520. stopTime = settings.stopTime.value;
  521. if (0 == +startTime && 0 == +stopTime) return -1;
  522. const t = new Date(),
  523. curT = 60 * t.getHours() + t.getMinutes();
  524. let startT, stopT;
  525. if (0 != +startTime) {
  526. const tArr1 = startTime.trim().replace(":", ":").split(":");
  527. startT = 60 * +tArr1[0] + +tArr1[1];
  528. }
  529. if (0 != +stopTime) {
  530. const tArr2 = stopTime.trim().replace(":", ":").split(":");
  531. stopT = 60 * +tArr2[0] + +tArr2[1];
  532. }
  533. if (settings.startStopWay.value)
  534. return curT === startT || (curT !== stopT && -1);
  535. if (0 == +startTime) return !(curT >= stopT) && -1;
  536. if (0 == +stopTime) return curT >= startT || -1;
  537. const f = (function isTimeInRange(t, startTime, stopTime, rangeLimit) {
  538. const curH = t.getHours(),
  539. curMin = t.getMinutes(),
  540. startText = startTime.trim().replace(":", ":"),
  541. stopText = stopTime.trim().replace(":", ":"),
  542. tArr1 = startText.split(":"),
  543. tArr2 = stopText.split(":"),
  544. h1 = +tArr1[0],
  545. h2 = +tArr2[0],
  546. startT = 60 * h1 + +tArr1[1],
  547. stopT = 60 * h2 + +tArr2[1],
  548. curT = 60 * curH + curMin;
  549. if (startT < stopT)
  550. if (rangeLimit[0]) {
  551. if (rangeLimit[1]) {
  552. if (curT >= startT && curT <= stopT) return !0;
  553. } else if (curT >= startT && curT < stopT) return !0;
  554. } else if (rangeLimit[1]) {
  555. if (curT > startT && curT <= stopT) return !0;
  556. } else if (curT > startT && curT < stopT) return !0;
  557. if (startT > stopT)
  558. if (rangeLimit[0]) {
  559. if (rangeLimit[1]) {
  560. if (
  561. (curT >= startT && curT < 1440) ||
  562. (curT <= stopT && curT >= 0)
  563. )
  564. return !0;
  565. } else if (
  566. (curT >= startT && curT < 1440) ||
  567. (curT < stopT && curT >= 0)
  568. )
  569. return !0;
  570. } else if (rangeLimit[1]) {
  571. if (
  572. (curT > startT && curT < 1440) ||
  573. (curT <= stopT && curT >= 0)
  574. )
  575. return !0;
  576. } else if (
  577. (curT > startT && curT < 1440) ||
  578. (curT < stopT && curT >= 0)
  579. )
  580. return !0;
  581. return !1;
  582. })(t, startTime, stopTime, [1, 0]);
  583. return f;
  584. })();
  585. -1 !== f &&
  586. (settings.isShowTips.value &&
  587. GM_notification({
  588. title: "深色模式",
  589. text: `定时${f ? "开启" : "关闭"}深色模式`,
  590. timeout: 3e3,
  591. }),
  592. setDarkMode(f));
  593. };
  594. autoStartStop(),
  595. info.timer && clearInterval(info.timer),
  596. (info.timer = setInterval(autoStartStop, info.interval));
  597. }
  598. function verifyWebsite(websiteText, url) {
  599. if (!websiteText) return !1;
  600. url = url || location.href;
  601. return websiteText
  602. .trim()
  603. .split("\n")
  604. .some((item) => matchUrlWithWildcard(url, item));
  605. }
  606. function updateShow() {
  607. getData(info.settings, GM_getValue, GM_setValue),
  608. info.cssDom
  609. ? (info.cssDom.innerHTML = getCssHtml(info.isDarkMode))
  610. : info.isDarkMode
  611. ? (setDarkMode(!1), setDarkMode(!0))
  612. : setDarkMode(!1),
  613. createDarkModeBtn(),
  614. setStartStopTimer();
  615. }
  616. function createDarkModeBtn() {
  617. const settings = info.settings;
  618. if (info.modeBtn) {
  619. const btn = info.modeBtn;
  620. return (
  621. (btn.style.width = settings.btnSize.value + "px"),
  622. (btn.style.height = settings.btnSize.value + "px"),
  623. void (info.isDarkMode
  624. ? btn.classList.add("dark")
  625. : btn.classList.remove("dark"))
  626. );
  627. }
  628. const html = `<div id="${info.keyBase}btn" style="width:${
  629. settings.btnSize.value
  630. }px;height:${settings.btnSize.value}px" title="${
  631. info.btnHoverTxt
  632. }" class="${
  633. info.isDarkMode ? "dark" : ""
  634. }">\n <svg t="1723482089223" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="6752"><path d="M505.565962 845.481164c-14.545274 0-26.317389 11.770068-26.317389 26.316366l0 61.937654c0 14.546298 11.771091 26.316366 26.317389 26.316366 14.520715 0 26.317389-11.770068 26.317389-26.316366l0-61.937654C531.883351 857.251232 520.086677 845.481164 505.565962 845.481164z" p-id="6753"></path><path d="M735.426117 803.899116c-7.248078-12.595876-23.362081-16.936741-35.929304-9.611915-12.591783 7.247054-16.909112 23.334451-9.611915 35.925211l30.969339 53.635571c4.856611 8.458649 13.697977 13.160741 22.795169 13.160741 4.471848 0 8.994861-1.1328 13.157671-3.523243 12.593829-7.270591 16.884552-23.38664 9.612938-35.953863L735.426117 803.899116z" p-id="6754"></path><path d="M931.462932 674.318876l-53.66013-30.994921c-12.567223-7.29515-28.680203-2.979868-35.927257 9.613962-7.272637 12.593829-2.981914 28.682249 9.611915 35.953863l53.661154 30.969339c4.111644 2.416026 8.634658 3.545756 13.107529 3.545756 9.097192 0 17.937534-4.726651 22.820752-13.156648C948.348508 697.653327 944.057785 681.56593 931.462932 674.318876z" p-id="6755"></path><path d="M306.006927 794.288225c-12.567223-7.353478-28.705785-2.983961-35.953863 9.611915l-30.968315 53.633524c-7.272637 12.567223-2.955308 28.682249 9.636474 35.953863 4.112668 2.390443 8.635681 3.523243 13.107529 3.523243 9.098215 0 17.939581-4.703115 22.821775-13.160741l30.993898-53.635571C322.89148 817.621653 318.59871 801.535279 306.006927 794.288225z" p-id="6756"></path><path d="M127.675356 643.323954l-53.609988 30.994921C61.472562 681.56593 57.155233 697.653327 64.42787 710.249203c4.857635 8.429996 13.696953 13.156648 22.795169 13.156648 4.471848 0 8.995885-1.12973 13.158694-3.545756l53.609988-30.969339c12.591783-7.270591 16.909112-23.360034 9.636474-35.953863C156.355559 640.344087 140.268162 636.028804 127.675356 643.323954z" p-id="6757"></path><path d="M932.954913 63.947428 90.024851 63.947428c-14.520715 0-26.315342 11.796674-26.315342 26.317389 0 14.546298 11.794627 26.316366 26.315342 26.316366L425.177074 116.581182l0.051165 107.373473c-96.322789 35.773761-162.089655 128.910998-162.089655 232.943376 0 136.955208 111.409392 248.364601 248.337995 248.364601 136.955208 0 248.363577-111.408369 248.363577-248.364601 0-49.085952-14.314007-96.578616-41.377386-137.288806-8.017604-12.07706-24.364921-15.366989-36.494169-7.349385-12.103666 8.070816-15.393595 24.388457-7.350408 36.494169 21.330818 32.04688 32.587186 69.440535 32.587186 108.144022 0 107.939361-87.789439 195.729823-195.7288 195.729823-107.913778 0-195.704241-87.790462-195.704241-195.729823 0-87.123266 58.492182-164.5548 142.247748-188.328249 10.743692-3.03308 18.401092-12.516058 19.069311-23.64349l0.61603-128.34511 455.248462 0c14.546298 0 26.316366-11.770068 26.316366-26.316366C959.271278 75.744101 947.50121 63.947428 932.954913 63.947428z" p-id="6758"></path></svg>\n</div>`;
  635. document.body.insertAdjacentHTML("beforeend", html);
  636. const btn = document.body.querySelector(`#${info.keyBase}btn`);
  637. (info.modeBtn = btn),
  638. btn.addEventListener("click", () => {
  639. btn.classList.toggle("dark");
  640. setDarkMode(btn.classList.contains("dark"));
  641. });
  642. }
  643. const baseCfg = {
  644. state: "",
  645. isEditing: !1,
  646. hasSelectedPage: !1,
  647. param: {
  648. id: "ll_edit_wrap",
  649. box: document.body,
  650. classBase: "ll_edit_",
  651. w: "500px",
  652. h: "",
  653. contentH: "450px",
  654. bg: "rgba(0, 0, 0, 0.15)",
  655. color: "#333",
  656. fontSize: "15px",
  657. fontFamily:
  658. "PingFang SC, HarmonyOS_Regular, Helvetica Neue, Microsoft YaHei, sans-serif",
  659. zIndex: 11e3,
  660. resetTt: "重置所有设置为默认值",
  661. isShowMenu: !1,
  662. isScrollStyle: !0,
  663. isResetBtn: !0,
  664. isOnlyResetCurPage: !1,
  665. showPage: void 0,
  666. isIntervalRun: !1,
  667. interval: 1e3,
  668. page: [],
  669. callback: {
  670. resetBefore: null,
  671. reset: null,
  672. confirmBefore: null,
  673. finished: null,
  674. interval: null,
  675. cancelBefore: null,
  676. cancelled: null,
  677. },
  678. },
  679. },
  680. cfg = {
  681. version: "v1.2.2",
  682. isEditing: baseCfg.isEditing,
  683. hasSelectedPage: baseCfg.hasSelectedPage,
  684. timer: null,
  685. interval: 1e3,
  686. param: {},
  687. tempParam: {},
  688. allData: {},
  689. oldData: {},
  690. lastData: {},
  691. baseData: {},
  692. controls: {},
  693. doms: { page: [] },
  694. editText: {},
  695. };
  696. const css = function getCss() {
  697. const param = cfg.param,
  698. cBase = (param.page, param.classBase),
  699. baseStart = `#${param.id} .${cBase}`,
  700. fSize = param.fontSize ? param.fontSize : "14px",
  701. css = `#${
  702. param.id
  703. } {\n position: fixed;\n left: 0;\n top: 0;\n width: 100%;\n height: 100%;\n z-index: ${
  704. param.zIndex || 11e3
  705. };\n background: ${
  706. param.bg || "rgba(0, 0, 0, 0.12)"
  707. };\n display: none;\n}\n${baseStart}box {\n text-align: initial;\n letter-spacing: 1px;\n position: relative;\n width: ${
  708. param.w || "450px"
  709. };\n ${
  710. param.h ? "max-height:" + param.h : ""
  711. };\n margin: auto;\n color: ${
  712. param.color || "#333"
  713. };\n background: #fff;\n font-size: ${fSize};\n line-height: normal;\n font-family: ${
  714. param.fontFamily ||
  715. "PingFang SC, HarmonyOS_Regular, Helvetica Neue, Microsoft YaHei, sans-serif"
  716. };\n border: 3px solid #dfedfe;\n border-radius: 10px;\n box-sizing: border-box;\n padding: 14px 8px 10px 15px;\n overflow: hidden;\n overflow-y: auto;\n}\n${baseStart}menu {\n font-weight: bold;\n font-size: ${
  717. parseInt(fSize) + 1
  718. }px;\n display: flex;\n flex-wrap: wrap;\n gap: 0 8px;\n}\n${baseStart}menu-item {\n margin-bottom: 8px;\n border: 1px solid #dfedfe;\n color: #9ecaff;\n background: #eef6ff;\n border-radius: 6px;\n padding: 6px 10px;\n cursor: pointer;\n}\n${baseStart}menu-item:hover {\n color: #65aaff;\n background: #dfedfe;\n border: 1px solid #dfedfe;\n}\n${baseStart}menu-item.active {\n color: #65aaff;\n background: #dfedfe;\n border: 1px solid #dfedfe;\n}\n${baseStart}page-box {\n max-height: ${
  719. param.contentH || ""
  720. };\n padding-right: 7px;\n margin-bottom: 8px;\n overflow: hidden;\n overflow-y: auto;\n}\n${baseStart}page {\n display: none;\n}\n${baseStart}page.curPage {\n display: block;\n}\n${baseStart}comp {\n margin-bottom: 8px;\n}\n${baseStart}comp:last-child {\n margin-bottom: 2px;\n}\n${baseStart}tt {\n font-weight: bold;\n font-size: ${
  721. parseInt(fSize) + 6
  722. }px;\n margin-top: 4px;\n}\n${baseStart}tt2 {\n font-weight: bold;\n font-size: ${
  723. parseInt(fSize) + 4
  724. }px;\n margin-top: 3px;\n margin-bottom: 7px;\n}\n${baseStart}tt3 {\n font-weight: bold;\n font-size: ${
  725. parseInt(fSize) + 2
  726. }px;\n margin-top: 2px;\n margin-bottom: 6px;\n}\n${baseStart}desc {\n line-height: 1.5;\n}\n${baseStart}comp-tt {\n font-weight: bold;\n font-size: ${
  727. parseInt(fSize) + 1
  728. }px;\n line-height: 1.5;\n}\n${baseStart}comp-desc {\n line-height: 1.5;\n}\n${baseStart}rd-arr {\n line-height: 22px;\n}\n${baseStart}rd-arr label {\n margin-right: 6px;\n cursor: pointer;\n}\n${baseStart}rd-arr input {\n vertical-align: -2px;\n cursor: pointer;\n}\n${baseStart}rd-arr span {\n color: #666;\n margin-left: 2px;\n}\n#${
  729. param.id
  730. } textarea {\n width: 100%;\n max-width: 100%;\n max-height: 300px;\n border-radius: 6px;\n line-height: normal;\n padding: 5px 7px;\n outline-color: #cee4ff;\n border: 1px solid #aaa;\n box-sizing: border-box;\n font-size: ${
  731. parseInt(fSize) - 2
  732. }px;\n font-family: PingFang SC, HarmonyOS_Regular, Helvetica Neue, Microsoft YaHei, sans-serif;\n /* 保留空格 */\n white-space: pre-wrap;\n /* 允许词内换行 */\n word-break: break-all;\n letter-spacing: 1px;\n overflow: hidden;\n overflow-y: auto;\n}\n#${
  733. param.id
  734. } textarea::placeholder {\n color: #bbb;\n}\n${baseStart}ta-desc {\n margin-bottom: 3px;\n}\n${baseStart}btn-box {\n display: flex;\n justify-content: flex-end;\n}\n${baseStart}btn-box button {\n font-size: 16px;\n line-height: normal;\n color: #65aaff;\n background: #dfedfe;\n outline: none;\n border: none;\n border-radius: 6px;\n padding: 8px 16px;\n box-sizing: border-box;\n cursor: pointer;\n}\n${baseStart}btn-box .${cBase}reset-btn {\n position: absolute;\n left: 15px;\n bottom: 10px;\n color: #888;\n background: #f4f4f4;\n margin-right: 15px;\n}\n${baseStart}btn-box .${cBase}reset-btn:hover {\n color: #666;\n background: #eee;\n}\n${baseStart}btn-box .${cBase}cancel-btn {\n color: #888;\n background: #f4f4f4;\n margin-right: 15px;\n}\n${baseStart}btn-box .${cBase}cancel-btn:hover {\n color: #666;\n background: #eee;\n}\n${baseStart}btn-box .${cBase}confirm-btn {\n margin-right: 7px;\n}\n${baseStart}btn-box .${cBase}confirm-btn:hover {\n background: #cee4ff;\n}\n`;
  735. return param.isScrollStyle
  736. ? css +
  737. "\n.ll-scroll-style-1::-webkit-scrollbar,\n.ll-scroll-style-1 ::-webkit-scrollbar {\n width: 8px;\n}\n.ll-scroll-style-1-size-2::-webkit-scrollbar,\n.ll-scroll-style-1 .ll-scroll-style-1-size-2::-webkit-scrollbar {\n width: 10px;\n}\n.ll-scroll-style-1-size-3::-webkit-scrollbar,\n.ll-scroll-style-1 .ll-scroll-style-1-size-3::-webkit-scrollbar {\n width: 12px;\n}\n.ll-scroll-style-1::-webkit-scrollbar-thumb,\n.ll-scroll-style-1 ::-webkit-scrollbar-thumb {\n border-radius: 10px;\n -webkit-box-shadow: inset 0 0 8px rgba(0, 0, 0, 0.05);\n opacity: 0.2;\n background: #daedff;\n}\n.ll-scroll-style-1::-webkit-scrollbar-track,\n.ll-scroll-style-1 ::-webkit-scrollbar-track {\n -webkit-box-shadow: inset 0 0 8px rgba(0, 0, 0, 0.08);\n border-radius: 0;\n background: #fff;\n border-radius: 5px;\n}"
  738. : css;
  739. };
  740. const editArea_html = function getHTML() {
  741. function getCompHTML({ info, active = "", id }) {
  742. let type = info.type;
  743. if (
  744. ((type = {
  745. menuTitle: "mtt",
  746. title: "tt",
  747. title2: "tt2",
  748. title3: "tt3",
  749. desc: "ds",
  750. radio: "rd",
  751. checkbox: "cb",
  752. textarea: "ta",
  753. mtt: "mtt",
  754. tt: "tt",
  755. tt2: "tt2",
  756. tt3: "tt3",
  757. ds: "ds",
  758. rd: "rd",
  759. cb: "cb",
  760. ta: "ta",
  761. }[type]),
  762. (id = 0 === id ? "0" : id || ""),
  763. 0 === info.value && (info.value = "0"),
  764. !type)
  765. )
  766. return console.log("不存在的组件类型"), !1;
  767. let title = "",
  768. desc = "",
  769. ctrlTt = "";
  770. switch (
  771. (["tt", "tt2", "tt3", "ds", "mtt"].includes(type) ||
  772. ((title = info.title
  773. ? `<div class="${cBase}comp-tt ${cBase}${type}-tt" title="${
  774. info.tt || ""
  775. }">${info.title}</div>`
  776. : ""),
  777. (desc = info.desc
  778. ? `<div class="${cBase}comp-desc ${cBase}${type}-desc">${info.desc}</div>`
  779. : "")),
  780. type)
  781. ) {
  782. case "mtt":
  783. return (
  784. (info.value = info.value || ""),
  785. info.value
  786. ? `<div class="${cBase}menu-item ${active || ""}" title="${
  787. info.tt || ""
  788. }">${info.value}</div>`
  789. : ""
  790. );
  791. case "tt":
  792. case "tt2":
  793. case "tt3":
  794. return (
  795. (info.value = info.value || ""),
  796. info.value
  797. ? `<div class="${cBase}${type} ${cBase}comp" title="${
  798. info.tt || ""
  799. }">${info.value}</div>`
  800. : ""
  801. );
  802. case "ds":
  803. return (
  804. (info.value = info.value || ""),
  805. info.value
  806. ? `<div class="${cBase}desc ${cBase}comp" title="${
  807. info.descTt || ""
  808. }">${info.value}</div>`
  809. : ""
  810. );
  811. case "rd":
  812. const name = info.name || info.id + new Date().getTime();
  813. (ctrlTt = info.ctrlTt || ""),
  814. ctrlTt && (ctrlTt = `title="${ctrlTt}"`);
  815. let radio = `<div class="${cBase}rd ${cBase}rd-arr" ${ctrlTt}>`;
  816. if (void 0 === info.value && info.radioList[0]) {
  817. const obj = info.radioList[0];
  818. info.value = void 0 === obj.value ? obj.text : obj.value;
  819. }
  820. return (
  821. info.radioList.forEach((item, i) => {
  822. void 0 === item.value && (info.radioList[i].value = item.text),
  823. void 0 === item.text && (info.radioList[i].text = item.value);
  824. const value = item.value;
  825. let tt = item.tt || "";
  826. tt && (tt = `title="${tt}"`);
  827. let selected = "";
  828. info.value + "" == item.value + "" && (selected = "checked"),
  829. (radio += `<label ${tt}><input ${selected} type="radio" name="${name}" data-val="${value}" data-cpid="${id}"><span>${item.text}</span></label>`);
  830. }),
  831. (radio += "</div>"),
  832. `<div class="${cBase}comp ${cBase}ctrl ${cBase}rd-box" data-type="${type}" data-cpid="${id}">${title}${desc}${radio}</div>`
  833. );
  834. case "cb":
  835. const name2 = info.name || new Date().getTime();
  836. if (
  837. ((ctrlTt = info.ctrlTt || ""),
  838. ctrlTt && (ctrlTt = `title="${ctrlTt}"`),
  839. void 0 === info.value && info.radioList[0])
  840. ) {
  841. const obj = info.radioList[0];
  842. info.value = void 0 === obj.value ? obj.text : obj.value;
  843. }
  844. let checkbox = `<div class="${cBase}cb ${cBase}rd-arr" ${ctrlTt}>`;
  845. return (
  846. info.radioList.forEach((item, i) => {
  847. void 0 === item.value && (info.radioList[i].value = item.text),
  848. void 0 === item.text && (info.radioList[i].text = item.value);
  849. const value = item.value;
  850. let tt = item.tt || "";
  851. tt && (tt = `title="${tt}"`);
  852. let selected = "";
  853. info.value.includes(value) && (selected = "checked"),
  854. (checkbox += `<label ${tt}><input ${selected} type="checkbox" name="${name2}" data-val="${value}" data-cpid="${id}"><span>${item.text}</span></label>`);
  855. }),
  856. (checkbox += "</div>"),
  857. `<div class="${cBase}comp ${cBase}ctrl ${cBase}cb-box" data-type="${type}" data-cpid="${id}">${title}${desc}${checkbox}</div>`
  858. );
  859. case "ta":
  860. const taH = `height:${info.height || "30px"};`,
  861. style = `style="${
  862. info.width ? "width:" + info.width + ";" : ""
  863. }${taH}${
  864. info.fontSize ? "font-size:" + info.fontSize + ";" : ""
  865. }${
  866. info.fontFamily ? "font-family:" + info.fontFamily + ";" : ""
  867. }"`,
  868. textarea = `<textarea class="${cBase}ta" ${style} data-cpid="${id}" placeholder="${
  869. info.ph || ""
  870. }" title="${info.ctrlTt || "拖动右下角可调节宽高"}"></textarea>`;
  871. return `<div class="${cBase}comp ${cBase}ctrl ${cBase}ta-box" data-type="${type}" data-cpid="${id}">${title}${desc}${textarea}</div>`;
  872. }
  873. }
  874. const param = cfg.param,
  875. page = param.page,
  876. cBase = param.classBase,
  877. isMenu = 1 !== page.length;
  878. let menu = `<div class="${cBase}menu">`,
  879. pageHTML = `<div class="${cBase}page-box ll-scroll-style-1 ll-scroll-style-1-size-2">`;
  880. page.forEach((curPage, index) => {
  881. let pgid = curPage.id || index;
  882. (pgid += ""), (cfg.allData[pgid] = {}), (cfg.baseData[pgid] = {});
  883. let pageFlag = "";
  884. if (
  885. (cfg.hasSelectedPage ||
  886. ((void 0 === param.showPage || pgid === param.showPage + "") &&
  887. ((pageFlag = "curPage"), (cfg.hasSelectedPage = !0))),
  888. (pageHTML += `<div class="${cBase}page ${pageFlag}" data-pgid="${pgid}">`),
  889. curPage.components)
  890. ) {
  891. let compIndex = 0;
  892. if (isMenu || param.isShowMenu) {
  893. let curMenu = curPage.components.find(
  894. (item) => "menuTitle" === item.type
  895. );
  896. curMenu || (curMenu = { type: "menuTitle", value: pgid }),
  897. (menu += getCompHTML({
  898. info: curMenu,
  899. active: pageFlag ? "active" : "",
  900. }));
  901. }
  902. curPage.components.forEach((item) => {
  903. const cpid = item.id || compIndex;
  904. "menuTitle" !== item.type &&
  905. (pageHTML += getCompHTML({ info: item, id: cpid })),
  906. ["title", "title2", "title3", "desc", "menuTitle"].includes(
  907. item.type
  908. ) ||
  909. ((item.base = void 0 === item.base ? item.value : item.base),
  910. (cfg.allData[pgid][cpid] = item.value),
  911. (cfg.baseData[pgid][cpid] = item.base),
  912. compIndex++);
  913. });
  914. }
  915. pageHTML += "</div>";
  916. }),
  917. (pageHTML += "</div>"),
  918. isMenu || param.isShowMenu ? (menu += "</div>") : (menu = "");
  919. const resetBtn = param.isResetBtn
  920. ? `<button class="${cBase}reset-btn" title="${
  921. param.resetTt || "重置所有设置为默认值"
  922. }">重置</button>`
  923. : "",
  924. btnBox = `<div class="${cBase}btn-box">\n${resetBtn}\n<button class="${cBase}cancel-btn">取 消</button>\n<button class="${cBase}confirm-btn">确 认</button>\n</div>`;
  925. return `<div class="${cBase}box ll-scroll-style-1 ll-scroll-style-1-size-3" data-version="${cfg.version}">\n${menu}\n${pageHTML}\n${btnBox}\n</div>`;
  926. },
  927. baseParam = baseCfg.param,
  928. controls = cfg.controls,
  929. doms = cfg.doms;
  930. function createEditEle({
  931. id = baseParam.id,
  932. box = baseParam.box,
  933. classBase = baseParam.classBase,
  934. w = baseParam.w,
  935. h = baseParam.h,
  936. contentH = baseParam.contentH,
  937. bg = baseParam.bg,
  938. color = baseParam.color,
  939. fontSize = baseParam.fontSize,
  940. fontFamily = baseParam.fontFamily,
  941. zIndex = baseParam.zIndex,
  942. resetTt = baseParam.resetTt,
  943. isShowMenu = baseParam.isShowMenu,
  944. isScrollStyle = baseParam.isScrollStyle,
  945. isResetBtn = baseParam.isResetBtn,
  946. isOnlyResetCurPage = baseParam.isOnlyResetCurPage,
  947. showPage = baseParam.showPage,
  948. isIntervalRun = baseParam.isIntervalRun,
  949. interval = baseParam.interval,
  950. page = [],
  951. callback = baseParam.callback,
  952. } = {}) {
  953. (cfg.state = baseCfg.state),
  954. (cfg.isEditing = baseCfg.isEditing),
  955. (cfg.hasSelectedPage = baseCfg.hasSelectedPage),
  956. (cfg.param = { ...baseParam });
  957. const param = cfg.param;
  958. (box = box || document.body),
  959. (param.id = id),
  960. (param.box = box),
  961. (param.classBase = classBase),
  962. (param.w = w),
  963. (param.h = h),
  964. (param.contentH = contentH),
  965. (param.bg = bg),
  966. (param.color = color),
  967. (param.fontSize = fontSize),
  968. (param.fontFamily = fontFamily),
  969. (param.zIndex = zIndex),
  970. (param.resetTt = resetTt),
  971. (param.isShowMenu = isShowMenu),
  972. (param.isScrollStyle = isScrollStyle),
  973. (param.isResetBtn = isResetBtn),
  974. (param.isOnlyResetCurPage = isOnlyResetCurPage),
  975. (param.showPage = showPage),
  976. (param.isIntervalRun = isIntervalRun),
  977. (param.interval = interval),
  978. (param.page = page),
  979. (param.callback = callback),
  980. (cfg.interval = interval),
  981. (cfg.callback = callback);
  982. const html = editArea_html();
  983. return (
  984. box.querySelector(`#${param.classBase}${param.id}-css`) ||
  985. (function addCss(cssText, box = document.body, id = "") {
  986. const style = document.createElement("style");
  987. return (
  988. id && (style.id = id),
  989. box.appendChild(style),
  990. (style.innerHTML = cssText),
  991. style
  992. );
  993. })(css(), box, param.classBase + param.id + "-css"),
  994. (doms.wrap = (function createEle({
  995. className = "",
  996. id = "",
  997. title = "",
  998. css,
  999. box = document.body,
  1000. type = "div",
  1001. } = {}) {
  1002. const ele = document.createElement(type);
  1003. return (
  1004. id && (ele.id = id),
  1005. className && (ele.className = className),
  1006. title && (ele.title = title),
  1007. css && (ele.style.cssText = css),
  1008. box.appendChild(ele),
  1009. ele
  1010. );
  1011. })({ className: id, id })),
  1012. (doms.wrap.innerHTML = html),
  1013. (function getDoms() {
  1014. const param = cfg.param,
  1015. cBase = param.classBase;
  1016. (doms.box = doms.wrap.querySelector(`.${cBase}box`)),
  1017. (doms.cancel = doms.box.querySelector(`.${cBase}cancel-btn`)),
  1018. (doms.confirm = doms.box.querySelector(`.${cBase}confirm-btn`));
  1019. const isMenu = 1 !== param.page.length;
  1020. (isMenu || param.isShowMenu) &&
  1021. ((doms.menu = doms.box.querySelector(`.${cBase}menu`)),
  1022. (doms.menus = [].slice.call(
  1023. doms.menu.querySelectorAll(`.${cBase}menu-item`)
  1024. )));
  1025. const pages = [].slice.call(doms.box.querySelectorAll(`.${cBase}page`));
  1026. (doms.page = []),
  1027. param.isResetBtn &&
  1028. (doms.reset = doms.box.querySelector(`.${cBase}reset-btn`));
  1029. pages.forEach((curPage, index) => {
  1030. cfg.hasSelectedPage ||
  1031. (curPage.classList.add("curPage"),
  1032. (isMenu || param.isShowMenu) &&
  1033. doms.menus[0].classList.add("active"),
  1034. (cfg.hasSelectedPage = !0));
  1035. const page = {},
  1036. pgid = curPage.dataset.pgid;
  1037. (page.pgid = curPage.pgid = pgid),
  1038. (page.controls = [].slice.call(
  1039. curPage.querySelectorAll(`.${cBase}ctrl`)
  1040. )),
  1041. (page.ele = curPage),
  1042. doms.page.push(page),
  1043. (isMenu || param.isShowMenu) &&
  1044. (doms.menus[index].settingsPage = curPage);
  1045. const ctrls = {};
  1046. (controls[pgid] = ctrls),
  1047. page.controls.forEach((item, i) => {
  1048. const cpid = item.dataset.cpid,
  1049. cType = item.dataset.type;
  1050. let dom;
  1051. (item.cpid = cpid),
  1052. "rd" === cType || "cb" === cType
  1053. ? ((dom = [].slice.call(item.querySelectorAll("input"))),
  1054. (dom.compType = cType))
  1055. : "ta" === cType &&
  1056. ((dom = item.querySelector("textarea")),
  1057. (dom.compType = cType),
  1058. (dom.value = cfg.allData[pgid][cpid])),
  1059. (ctrls[cpid] = dom);
  1060. });
  1061. });
  1062. })(),
  1063. cfg.timer && clearInterval(cfg.timer),
  1064. (function bindEvents() {
  1065. const param = cfg.param;
  1066. function menuHandle(e) {
  1067. const dom = e.target,
  1068. cBase = param.classBase;
  1069. if (dom.classList.contains(`${cBase}menu-item`)) {
  1070. const old = doms.menu.querySelector(".active");
  1071. old.classList.remove("active"),
  1072. old.settingsPage.classList.remove("curPage"),
  1073. dom.classList.add("active"),
  1074. dom.settingsPage.classList.add("curPage");
  1075. }
  1076. }
  1077. function cancelEdit(e) {
  1078. const cBase = param.classBase;
  1079. if (
  1080. (e.stopPropagation(),
  1081. e.target.className !== `${cBase}wrap` &&
  1082. e.target.className !== `${cBase}cancel-btn`)
  1083. )
  1084. return;
  1085. const callback = cfg.callback;
  1086. !1 !== runCallback(callback.cancelBefore) &&
  1087. (showEditArea(!1),
  1088. setCompValue(cfg.oldData),
  1089. param.isIntervalRun &&
  1090. (setCompValue(cfg.oldData), (cfg.allData = cfg.oldData)),
  1091. runCallback(callback.cancelled));
  1092. }
  1093. function confirmEdit() {
  1094. const callback = cfg.callback,
  1095. data = getAllData();
  1096. (cfg.allData = data),
  1097. !1 !== runCallback(callback.confirmBefore, data) &&
  1098. (showEditArea(!1),
  1099. (cfg.state = "finished"),
  1100. runCallback(callback.finished, data),
  1101. (cfg.state = ""));
  1102. }
  1103. function resetEdit() {
  1104. const callback = cfg.callback,
  1105. data = getAllData();
  1106. !1 !== runCallback(callback.resetBefore, data) &&
  1107. (!(function resetEditData(isOnlyPage = !1) {
  1108. const param = cfg.param;
  1109. if (param.isResetBtn)
  1110. if (isOnlyPage) {
  1111. const data = getAllData(),
  1112. curMenu = doms.menu.querySelector(".active");
  1113. (data[curMenu.innerText] = cfg.baseData[curMenu.innerText]),
  1114. setCompValue(data);
  1115. } else setCompValue(cfg.baseData);
  1116. })(param.isOnlyResetCurPage),
  1117. runCallback(callback.reset, data));
  1118. }
  1119. doms.menu && doms.menu.addEventListener("click", menuHandle),
  1120. doms.wrap.addEventListener("click", cancelEdit),
  1121. doms.cancel.addEventListener("click", cancelEdit),
  1122. doms.confirm.addEventListener("click", confirmEdit),
  1123. doms.reset && doms.reset.addEventListener("click", resetEdit);
  1124. })(),
  1125. (cfg.state = "created"),
  1126. cfg
  1127. );
  1128. }
  1129. function getAllData() {
  1130. function getCompItem(pgid, cpid) {
  1131. if (!controls[pgid]) return;
  1132. const ctrl = controls[pgid][cpid];
  1133. if (ctrl) {
  1134. if (!Array.isArray(ctrl)) return ctrl.value;
  1135. if ("rd" === ctrl.compType) {
  1136. const result = ctrl.find((item) => item.checked).dataset.val;
  1137. return "false" !== result && ("true" === result || result);
  1138. }
  1139. if ("cb" === ctrl.compType) {
  1140. return ctrl
  1141. .filter((item) => item.checked)
  1142. .map((item) => {
  1143. const value = item.dataset.val;
  1144. return "false" !== value && ("true" === value || value);
  1145. });
  1146. }
  1147. }
  1148. }
  1149. const data = {};
  1150. if (0 === arguments.length) {
  1151. for (const key in controls) {
  1152. const page = controls[key];
  1153. data[key] = {};
  1154. for (const key2 in page) data[key][key2] = getCompItem(key, key2);
  1155. }
  1156. return data;
  1157. }
  1158. if (1 === arguments.length) {
  1159. const ctrls = arguments[0];
  1160. for (const pgid in ctrls) {
  1161. data[pgid] = {};
  1162. controls[pgid].forEach((cpid) => {
  1163. data[pgid][cpid] = getCompItem(pgid, cpid);
  1164. });
  1165. }
  1166. return cfg.allData;
  1167. }
  1168. return getCompItem(arguments[0], arguments[1]);
  1169. }
  1170. function setCompValue() {
  1171. function setCompItem(pgid, cpid, value) {
  1172. if (!controls[pgid]) return;
  1173. const ctrl = controls[pgid][cpid];
  1174. if (ctrl)
  1175. if (Array.isArray(ctrl)) {
  1176. if ("rd" === ctrl.compType) {
  1177. const selected = ctrl.find((item) => item.checked);
  1178. selected && (selected.checked = !1);
  1179. const select = ctrl.find((item) => item.dataset.val === value + "");
  1180. select && (select.checked = !0);
  1181. } else if ("cb" === ctrl.compType) {
  1182. if (
  1183. (ctrl
  1184. .filter((item) => item.checked)
  1185. .forEach((item) => {
  1186. item.checked = !1;
  1187. }),
  1188. Array.isArray(value))
  1189. )
  1190. value.forEach((val) => {
  1191. const select = ctrl.find(
  1192. (item) => item.dataset.val === val + ""
  1193. );
  1194. select && (select.checked = !0);
  1195. });
  1196. else {
  1197. const select = ctrl.find(
  1198. (item) => item.dataset.val === value + ""
  1199. );
  1200. select && (select.checked = !0);
  1201. }
  1202. }
  1203. } else ctrl.value = value;
  1204. }
  1205. if (1 === arguments.length) {
  1206. const data = arguments[0];
  1207. for (const key in data) {
  1208. const pageData = data[key];
  1209. for (const key2 in pageData) {
  1210. setCompItem(key, key2, pageData[key2]);
  1211. }
  1212. }
  1213. } else {
  1214. setCompItem(arguments[0], arguments[1], arguments[2]);
  1215. }
  1216. }
  1217. function showEditArea(isShow = !0, callback = null) {
  1218. if (
  1219. (cfg.param.isIntervalRun &&
  1220. (cfg.timer && clearInterval(cfg.timer),
  1221. (cfg.timer = setInterval(() => {
  1222. const data = getAllData(),
  1223. oldType = cfg.state;
  1224. (cfg.state = "interval"),
  1225. runCallback(cfg.callback.interval, data),
  1226. (cfg.state = oldType),
  1227. (cfg.lastData = data);
  1228. }, cfg.interval))),
  1229. (cfg.state = "created"),
  1230. isShow)
  1231. ) {
  1232. if (((cfg.oldData = getAllData()), "function" == typeof callback)) {
  1233. if (!1 === callback(cfg.oldData, cfg.oldData, cfg.baseData)) return;
  1234. }
  1235. cfg.state = "show";
  1236. }
  1237. (cfg.isEditing = isShow),
  1238. (doms.wrap.style.display = isShow ? "block" : "none"),
  1239. isShow &&
  1240. !doms.box.style.top &&
  1241. (doms.box.style.top =
  1242. window.innerHeight / 2 - doms.box.clientHeight / 2 + "px"),
  1243. callback && (cfg.callback = callback);
  1244. }
  1245. function runCallback(callback, data) {
  1246. let result;
  1247. if (callback) {
  1248. data || (data = getAllData());
  1249. const func = callback;
  1250. Array.isArray(func)
  1251. ? func.curFn
  1252. ? ((result = func[curFn](data, cfg.oldData, cfg.baseData)),
  1253. (func.curFn = null))
  1254. : func.forEach((fn) => {
  1255. result = fn(data, cfg.oldData, cfg.baseData);
  1256. })
  1257. : "function" == typeof callback &&
  1258. (result = func(data, cfg.oldData, cfg.baseData));
  1259. }
  1260. return result;
  1261. }
  1262. function toPageObj({ settings, param = {}, otherPageName = "无分类" } = {}) {
  1263. param = { ...param };
  1264. const pageArr = [],
  1265. menuList = [];
  1266. let isOtherType = !1;
  1267. for (let key in settings) {
  1268. const item = settings[key];
  1269. item.type
  1270. ? menuList.includes(item.type) || menuList.push(item.type)
  1271. : isOtherType || (isOtherType = !0);
  1272. }
  1273. return (
  1274. isOtherType && menuList.push(otherPageName),
  1275. menuList.forEach((menuTt) => {
  1276. const components = [],
  1277. page = { id: menuTt, components },
  1278. arr = [];
  1279. for (let key in settings) {
  1280. const item = settings[key];
  1281. menuTt === otherPageName
  1282. ? item.type || arr.push(item)
  1283. : item.type === menuTt && arr.push(item);
  1284. }
  1285. arr.forEach((item) => {
  1286. let desc = item.desc || item.txt || "";
  1287. desc && (desc = desc.replaceAll("\n", "<br>").trim());
  1288. let comp,
  1289. base = item.base;
  1290. if (
  1291. (Array.isArray(base) && (base = base.join(", ")), item.groupTitle1)
  1292. ) {
  1293. const comp = {
  1294. id: item.key + "-gTt1",
  1295. type: "title",
  1296. value: item.groupTitle1,
  1297. };
  1298. components.push(comp);
  1299. }
  1300. if (item.groupTitle2) {
  1301. const comp = {
  1302. id: item.key + "-gTt2",
  1303. type: "title2",
  1304. value: item.groupTitle2,
  1305. };
  1306. components.push(comp);
  1307. }
  1308. if (item.groupTitle3) {
  1309. const comp = {
  1310. id: item.key + "-gTt3",
  1311. type: "title3",
  1312. value: item.groupTitle3,
  1313. };
  1314. components.push(comp);
  1315. }
  1316. if (item.groupDesc) {
  1317. const comp = {
  1318. id: item.key + "-gDesc",
  1319. type: "desc",
  1320. value: item.groupDesc,
  1321. };
  1322. components.push(comp);
  1323. }
  1324. if (
  1325. (["menuTitle", "title", "desc", "title2", "title3"].includes(
  1326. item.compType
  1327. )
  1328. ? ((comp = { ...item }),
  1329. (comp.type = comp.compType),
  1330. (comp.desc = desc))
  1331. : (comp = {
  1332. id: item.key,
  1333. type: item.compType,
  1334. tt: item.tt || "",
  1335. title: item.title || "",
  1336. desc,
  1337. descTt: item.descTt || "",
  1338. name: item.key,
  1339. value: item.value,
  1340. base: item.base,
  1341. }),
  1342. "textarea" === comp.type)
  1343. )
  1344. (comp.ph = base),
  1345. (comp.width = item.compW),
  1346. (comp.height = item.compH),
  1347. (comp.ctrlTt = "默认: " + base);
  1348. else if ("radio" === comp.type || "checkbox" === comp.type) {
  1349. let str = "默认: ";
  1350. if ("checkbox" === comp.type) {
  1351. let arr = item.base;
  1352. Array.isArray(arr) || (arr = arr.split(/,|,/)),
  1353. arr.forEach((val, i) => {
  1354. 0 !== i && (str += ", "), (val = val.trim());
  1355. let valTxt = item.valueText[val];
  1356. void 0 === valTxt && (valTxt = val), (str += valTxt);
  1357. });
  1358. } else {
  1359. let val = item.valueText[item.base];
  1360. void 0 === val && (val = item.base), (str += val);
  1361. }
  1362. comp.ctrlTt = str;
  1363. }
  1364. if (item.valueText) {
  1365. comp.radioList = [];
  1366. for (let key in item.valueText) {
  1367. const rd = { text: item.valueText[key], value: key };
  1368. comp.radioList.push(rd);
  1369. }
  1370. }
  1371. components.push(comp);
  1372. }),
  1373. pageArr.push(page);
  1374. }),
  1375. (param.page = pageArr),
  1376. param
  1377. );
  1378. }
  1379. function saveDatas({
  1380. allData,
  1381. settings,
  1382. keyBase = "",
  1383. verifyFn = {},
  1384. getValue,
  1385. setValue,
  1386. }) {
  1387. for (const pageName in allData) {
  1388. const page = allData[pageName];
  1389. for (const key in page) {
  1390. const value = page[key],
  1391. item = settings[key.replace(keyBase, "")];
  1392. if (!item) return void console.log("设置的数据对应的对象获取失败");
  1393. let verify;
  1394. for (const name in verifyFn)
  1395. if (settings[name].key === key) {
  1396. verify = settings[name].verify || verifyFn[name];
  1397. break;
  1398. }
  1399. setValue_setValue({
  1400. value,
  1401. base: item.base,
  1402. key,
  1403. verification: verify,
  1404. getValue,
  1405. setValue,
  1406. });
  1407. }
  1408. }
  1409. }
  1410. function finishedSettings({
  1411. allData,
  1412. settings,
  1413. keyBase = "",
  1414. verifyFn = {},
  1415. isForcedUpdate = !1,
  1416. isRefreshPage = !1,
  1417. callback = null,
  1418. getValue,
  1419. setValue,
  1420. } = {}) {
  1421. if (!isForcedUpdate) {
  1422. if (
  1423. !(function isValueChange(type = "auto") {
  1424. const param = cfg.param,
  1425. curData = getAllData(),
  1426. curDataStr = JSON.stringify(curData);
  1427. let oldDataStr;
  1428. return (
  1429. "auto" === type &&
  1430. ("interval" === cfg.state &&
  1431. param.isIntervalRun &&
  1432. (type = "interval_current"),
  1433. "finished" === cfg.state && (type = "auto")),
  1434. (oldDataStr =
  1435. "interval_current" === type
  1436. ? JSON.stringify(cfg.lastData)
  1437. : "base_current" === type
  1438. ? JSON.stringify(cfg.baseData)
  1439. : JSON.stringify(cfg.oldData)),
  1440. "{}" !== oldDataStr && curDataStr !== oldDataStr
  1441. );
  1442. })()
  1443. )
  1444. return;
  1445. }
  1446. saveDatas({ allData, settings, keyBase, verifyFn, getValue, setValue }),
  1447. callback && "function" == typeof callback && callback(allData),
  1448. isRefreshPage && history.go(0);
  1449. }
  1450. function showSettings() {
  1451. const settings = info.settings;
  1452. info.settingsArea = (function createEdit({
  1453. settings,
  1454. param = {},
  1455. oldEditCfg,
  1456. updateDataFn,
  1457. isNewEdit = !0,
  1458. isSyncOtherPage = !0,
  1459. otherPageName = "无分类",
  1460. } = {}) {
  1461. let oldSettings, curSettings;
  1462. updateDataFn &&
  1463. isSyncOtherPage &&
  1464. ((oldSettings = JSON.stringify(settings)),
  1465. (settings = updateDataFn() || settings),
  1466. (curSettings = JSON.stringify(settings)));
  1467. const editInfo = { settings, param, otherPageName };
  1468. if (oldEditCfg) {
  1469. if (isNewEdit)
  1470. return (
  1471. oldEditCfg.doms.wrap.remove(), createEditEle(toPageObj(editInfo))
  1472. );
  1473. isSyncOtherPage &&
  1474. updateDataFn &&
  1475. oldSettings !== curSettings &&
  1476. (oldEditCfg.doms.wrap.remove(),
  1477. (oldEditCfg = createEditEle(toPageObj(editInfo)))),
  1478. isSyncOtherPage &&
  1479. !updateDataFn &&
  1480. (oldEditCfg.doms.wrap.remove(),
  1481. (oldEditCfg = createEditEle(toPageObj(editInfo))));
  1482. } else oldEditCfg = createEditEle(toPageObj(editInfo));
  1483. return oldEditCfg;
  1484. })({
  1485. settings,
  1486. param: {
  1487. bg: "rgba(0, 0, 0, 0)",
  1488. resetTt: "重置当前页的所有设置为默认值",
  1489. isOnlyResetCurPage: !0,
  1490. isIntervalRun: !0,
  1491. },
  1492. oldEditCfg: info.settingsArea,
  1493. updateDataFn: () => getData(settings, GM_getValue, GM_setValue),
  1494. });
  1495. showEditArea(!0, {
  1496. resetBefore: () => confirm("是否重置当前页的所有设置为默认值?"),
  1497. confirmBefore: () => {},
  1498. finished: (data, oldData) => {
  1499. console.log(data),
  1500. finishedSettings({
  1501. allData: data,
  1502. settings,
  1503. keyBase: info.keyBase,
  1504. verifyFn: settings,
  1505. callback: updateShow,
  1506. getValue: GM_getValue,
  1507. setValue: GM_setValue,
  1508. });
  1509. const f1 = verifyWebsite(
  1510. oldData[settings.website.type][
  1511. settings.website.key.repeat(info.keyBase, "")
  1512. ]
  1513. ),
  1514. f2 = verifyWebsite(settings.website.value);
  1515. if (f1 && !f2)
  1516. return (
  1517. setDarkMode(!1),
  1518. info.modeBtn && info.modeBtn.remove(),
  1519. void (info.timer && clearInterval(info.timer))
  1520. );
  1521. !f1 && f2 && (setStartStopTimer(), createDarkModeBtn());
  1522. },
  1523. interval: (data, oldData) => {
  1524. finishedSettings({
  1525. allData: data,
  1526. settings,
  1527. keyBase: info.keyBase,
  1528. verifyFn: settings,
  1529. callback: updateShow,
  1530. getValue: GM_getValue,
  1531. setValue: GM_setValue,
  1532. });
  1533. },
  1534. cancelled: (data, oldData) => {
  1535. saveDatas({
  1536. allData: oldData,
  1537. settings,
  1538. keyBase: info.keyBase,
  1539. verifyFn: settings,
  1540. getValue: GM_getValue,
  1541. setValue: GM_setValue,
  1542. }),
  1543. getData(settings, GM_getValue, GM_setValue),
  1544. updateShow();
  1545. },
  1546. });
  1547. }
  1548. const settings = info.settings;
  1549. function page_darkMode_bindEvents() {
  1550. const urlChangeHandle = () => {
  1551. const id = info.keyBase + "css",
  1552. dom = document.body.querySelector("#" + id);
  1553. (dom && dom.isDark) || (info.isDarkMode = !1);
  1554. };
  1555. let _wr = function (type) {
  1556. let orig = history[type];
  1557. return function () {
  1558. let rv = orig.apply(this, arguments),
  1559. e = new Event(type);
  1560. return (e.arguments = arguments), window.dispatchEvent(e), rv;
  1561. };
  1562. };
  1563. (history.pushState = _wr("pushState")),
  1564. (history.replaceState = _wr("replaceState")),
  1565. window.addEventListener("popstate", urlChangeHandle),
  1566. window.addEventListener("replaceState", urlChangeHandle),
  1567. window.addEventListener("pushState", urlChangeHandle);
  1568. }
  1569. !(function main() {
  1570. if (
  1571. (getData(settings, GM_getValue, GM_setValue),
  1572. (function getOtherData() {
  1573. const settings = info.otherSettings,
  1574. autoWay = info.settings.autoDarkModeWay.value;
  1575. for (const valName in settings) {
  1576. const setting = settings[valName];
  1577. setting.value = getValue({
  1578. base: setting.base,
  1579. key: setting.key,
  1580. valType: setting.valType,
  1581. getVal: autoWay ? GM_getValue : null,
  1582. setVal: autoWay ? GM_setValue : null,
  1583. });
  1584. }
  1585. })(),
  1586. (function registerMenu(f) {
  1587. let urlTxt = "*" + location.host + "*";
  1588. verifyWebsite(settings.website.value)
  1589. ? GM_registerMenuCommand("当前网站: 已启用✔️", () => {
  1590. const urlList = settings.website.value.trim().split("\n"),
  1591. i = urlList.indexOf("*");
  1592. -1 !== i && urlList.splice(i, 1);
  1593. let index = urlList.indexOf(urlTxt);
  1594. -1 !== index
  1595. ? urlList.splice(index, 1)
  1596. : urlList
  1597. .filter((item) => matchUrlWithWildcard(location.href, item))
  1598. .map((item) => urlList.indexOf(item))
  1599. .forEach((item) => {
  1600. let i = urlList.indexOf(item);
  1601. urlList.splice(i, 1);
  1602. });
  1603. const urlListTxt = urlList.join("\n");
  1604. GM_setValue(settings.website.key, urlListTxt), history.go(0);
  1605. })
  1606. : GM_registerMenuCommand("当前网站: 已禁用❌", () => {
  1607. const urlList = settings.website.value.trim().split("\n"),
  1608. i = urlList.indexOf("*");
  1609. -1 !== i && urlList.splice(i, 1), urlList.push(urlTxt);
  1610. const urlListTxt = Array.from(new Set(urlList)).join("\n");
  1611. GM_setValue(settings.website.key, urlListTxt), history.go(0);
  1612. }),
  1613. GM_registerMenuCommand("设置", () => {
  1614. showSettings();
  1615. });
  1616. })(),
  1617. verifyWebsite(settings.website.value))
  1618. )
  1619. if (
  1620. (settings.autoDarkMode.value
  1621. ? info.otherSettings.oldDarkMode.value && setDarkMode(!0)
  1622. : (setValue_setValue({
  1623. value: !1,
  1624. base: !1,
  1625. key: info.keyBase + "oldDarkMode",
  1626. }),
  1627. setValue_setValue({
  1628. value: !1,
  1629. base: !1,
  1630. key: info.keyBase + "oldDarkMode",
  1631. getValue: GM_getValue,
  1632. setValue: GM_setValue,
  1633. })),
  1634. setDarkMode(info.isDarkMode),
  1635. document.body)
  1636. )
  1637. setStartStopTimer(), createDarkModeBtn(), page_darkMode_bindEvents();
  1638. else {
  1639. const bodyObserver = new MutationObserver(() => {
  1640. document.body &&
  1641. (bodyObserver.disconnect(),
  1642. setStartStopTimer(),
  1643. createDarkModeBtn(),
  1644. page_darkMode_bindEvents());
  1645. });
  1646. bodyObserver.observe(document, { childList: !0, subtree: !0 });
  1647. }
  1648. })();
  1649. })();

QingJ © 2025

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