暗黑模式

最简单的全网通用暗黑模式

目前為 2021-05-13 提交的版本,檢視 最新版本

  1. // ==UserScript==
  2. // @name 暗黑模式
  3. // @version 1.0.0
  4. // @author X.I.U
  5. // @description 最简单的全网通用暗黑模式
  6. // @match *://*/*
  7. // @icon https://i.loli.net/2021/03/07/rdijeYm83pznxWq.png
  8. // @grant GM_registerMenuCommand
  9. // @grant GM_unregisterMenuCommand
  10. // @grant GM_getValue
  11. // @grant GM_setValue
  12. // @noframes
  13. // @license GPL-3.0 License
  14. // @run-at document-start
  15. // @namespace https://github.com/XIU2/UserScript
  16. // ==/UserScript==
  17.  
  18. (function() {
  19. var menu_ALL = [
  20. ['menu_runDuringTheDay', '白天保持开启 (比晚上更亮一点)', '白天保持开启', true],
  21. ['menu_darkModeType', '点击切换模式', '点击切换模式', 1]
  22. ], menu_ID = [];
  23. for (let i=0;i<menu_ALL.length;i++){ // 如果读取到的值为 null 就写入默认值
  24. if (GM_getValue(menu_ALL[i][0]) == null){GM_setValue(menu_ALL[i][0], menu_ALL[i][3])};
  25. }
  26. registerMenuCommand();
  27. addStyle();
  28.  
  29. // 注册脚本菜单
  30. function registerMenuCommand() {
  31. if (menu_ID.length > menu_ALL.length){ // 如果菜单ID数组多于菜单数组,说明不是首次添加菜单,需要卸载所有脚本菜单
  32. for (let i=0;i<menu_ID.length;i++){
  33. GM_unregisterMenuCommand(menu_ID[i]);
  34. }
  35. }
  36. for (let i=0;i<menu_ALL.length;i++){ // 循环注册脚本菜单
  37. menu_ALL[i][3] = GM_getValue(menu_ALL[i][0]);
  38. if (menu_ALL[i][0] === 'menu_darkModeType') {
  39. if (menu_ALL[i][3] > 3){ // 避免在减少 raw 数组后,用户储存的数据大于数组而报错
  40. menu_ALL[i][3] = 1;
  41. GM_setValue('menu_darkModeType', menu_ALL[i][3]);
  42. }
  43. menu_ID[i] = GM_registerMenuCommand(`🔄 [ ${menu_ALL[i][3]} ] ${menu_ALL[i][1]}`, function(){menu_toggle(`${menu_ALL[i][3]}`,`${menu_ALL[i][0]}`)});
  44. } else {
  45. menu_ID[i] = GM_registerMenuCommand(`🌝 [ ${menu_ALL[i][3]?'√':'×'} ] ${menu_ALL[i][1]}`, function(){menu_switch(`${menu_ALL[i][3]}`,`${menu_ALL[i][0]}`,`${menu_ALL[i][2]}`)});
  46. }
  47. }
  48. menu_ID[menu_ID.length] = GM_registerMenuCommand('💬 反馈 & 建议', function () {window.GM_openInTab('https://github.com/XIU2/UserScript#xiu2userscript', {active: true,insert: true,setParent: true});window.GM_openInTab('https://greasyfork.org/zh-CN/scripts/412212/feedback', {active: true,insert: true,setParent: true});});
  49. }
  50.  
  51.  
  52. // 切换暗黑模式
  53. function menu_toggle(menu_status, Name) {
  54. menu_status = parseInt(menu_status)
  55. if (menu_status >= 3){
  56. menu_status = 1;
  57. } else {
  58. menu_status += 1;
  59. }
  60. GM_setValue(`${Name}`, menu_status);
  61. location.reload(); // 刷新网页
  62. };
  63.  
  64.  
  65. // 菜单开关
  66. function menu_switch(menu_status, Name, Tips) {
  67. if (menu_status == 'true'){
  68. GM_setValue(`${Name}`, false);
  69. GM_notification({text: `已关闭 [${Tips}] 功能\n(刷新网页后生效)`, timeout: 3500});
  70. }else{
  71. GM_setValue(`${Name}`, true);
  72. GM_notification({text: `已开启 [${Tips}] 功能\n(刷新网页后生效)`, timeout: 3500});
  73. }
  74. registerMenuCommand(); // 重新注册脚本菜单
  75. };
  76.  
  77.  
  78. // 返回菜单值
  79. function menu_value(menuName) {
  80. for (let menu of menu_ALL) {
  81. if (menu[0] == menuName) {
  82. return menu[3]
  83. }
  84. }
  85. }
  86.  
  87.  
  88. // 添加样式
  89. function addStyle() {
  90. let grayLevel,rgbValueArry,
  91. style_Add = document.createElement('style'),
  92. hours = new Date().getHours(),
  93. style = ``,
  94. style_00 = `body {background-color: #ffffff !important;}`,
  95. style_11 = `html {filter: brightness(80%) !important;}`,
  96. style_11_firefox = `html {filter: brightness(80%) !important; background-image: url();}`,
  97. style_12 = `html {filter: brightness(70%) !important;}`,
  98. style_12_firefox = `html {filter: brightness(70%) !important; background-image: url();}`,
  99. style_21 = `html {filter: brightness(85%) sepia(20%) !important;}`,
  100. style_21_firefox = `html {filter: brightness(85%) sepia(20%) !important; background-image: url();}`,
  101. style_22 = `html {filter: brightness(70%) sepia(30%) !important;}`,
  102. style_22_firefox = `html {filter: brightness(70%) sepia(30%) !important; background-image: url();}`,
  103. style_31 = `html {filter: invert(80%) !important;} img, video {filter: invert(1) !important;}`,
  104. style_31_firefox = `html {filter: invert(80%) !important;} img, video {filter: invert(1) !important; background-image: url();}`;
  105.  
  106. // 判断网页是否没有设置背景颜色(没有背景颜色会导致滤镜对背景颜色无效)
  107. if (document.body) {
  108. rgbValueArry = window.getComputedStyle(document.body).backgroundColor.replace ('rgb(', '').replace ('rgba(', '').replace (')', '').split (', ');
  109. grayLevel = rgbValueArry [0] + rgbValueArry [1] + rgbValueArry [2];
  110. if (grayLevel === "000") style += style_00
  111. }
  112.  
  113. // Firefox 浏览器需要特殊对待
  114. if (navigator.userAgent.toLowerCase().indexOf('firefox') > -1) {
  115. style_11 = style_11_firefox
  116. style_12 = style_12_firefox
  117. style_21 = style_21_firefox
  118. style_22 = style_22_firefox
  119. style_31 = style_31_firefox
  120. }
  121.  
  122. // 白天(7点到19点)
  123. if (hours > 6 || hours < 19) {
  124. if (menu_value('menu_runDuringTheDay')) {
  125. style_12 = style_11
  126. style_22 = style_21
  127. } else {
  128. style_12 = style_22 = ''
  129. }
  130. }
  131.  
  132. switch(menu_value('menu_darkModeType')) {
  133. case 1:
  134. style += style_12;
  135. break;
  136. case 2:
  137. style += style_22;
  138. break;
  139. case 3:
  140. style += style_31;
  141. break;
  142. }
  143. style_Add.innerHTML = style;
  144. if (document.head) {
  145. document.head.appendChild(style_Add);
  146. } else { // 为了避免脚本运行的时候 head 还没加载导致报错
  147. let timer = setInterval(function(){
  148. if (document.head) {
  149. document.head.appendChild(style_Add);
  150. clearInterval(timer);
  151. }
  152. }, 1);
  153. }
  154.  
  155. // 为了避免 body 还没加载导致无法检查是否设置背景颜色的备用措施
  156. if (!grayLevel) {
  157. let timer2 = setInterval(function(){
  158. if (document.body) {
  159. let rgbValueArry = window.getComputedStyle(document.body).backgroundColor.replace ('rgb(', '').replace ('rgba(', '').replace (')', '').split (', '),
  160. style_Add1 = document.createElement('style');
  161. if (rgbValueArry [0] + rgbValueArry [1] + rgbValueArry [2] === "000") {
  162. style_Add1.innerHTML = 'body {background-color: #ffffff !important;}';
  163. document.head.appendChild(style_Add1);
  164. }
  165. clearInterval(timer2);
  166. }
  167. }, 1);
  168. }
  169. }
  170. })();