RSS+ : 显示当前网站所有的 RSS

显示当前网站的所有 RSS(如果有的话)

  1. // ==UserScript==
  2. // @name RSS+ : Show Site All RSS
  3. // @name:zh RSS+ : 显示当前网站所有的 RSS
  4. // @name:zh-CN RSS+ : 显示当前网站所有的 RSS
  5. // @name:zh-TW RSS+ : 顯示當前網站所有的 RSS
  6. // @name:ja RSS+ : 現在のサイトのRSSを表示
  7. // @name:ko RSS+ : 현재 사이트의 RSS 표시
  8. // @name:pt-PT RSS+ : Mostrar todos os RSS do site
  9. // @name:pt-BR RSS+ : Mostrar todos os RSS do site
  10. // @name:fr RSS+ : Afficher tous les flux RSS du site
  11. // @description Show All RSS Of The Site (If Any)
  12. // @description:zh 显示当前网站的所有 RSS(如果有的话)
  13. // @description:zh-CN 显示当前网站的所有 RSS(如果有的话)
  14. // @description:zh-TW 顯示當前網站的所有 RSS(如果有的话)
  15. // @description:ja サイトのすべてのRSSを表示します (あれば)
  16. // @description:ko 웹 사이트의 모든 RSS 를 표시합니다 (있는 경우)
  17. // @description:pt-PT Mostra todos os feeds RSS do site (se houver)
  18. // @description:pt-BR Mostra todos os feeds RSS do site (se houver)
  19. // @description:fr Montre tous les flux RSS du site (s'il y en a)
  20. // @license GPL3.0
  21. // @version 1.1.3
  22.  
  23. // @icon 
  24. // @author Wizos
  25. // @namespace https://blog.wizos.me
  26. // @supportURL wizos@qq.com
  27. // @match http://*/*
  28. // @match https://*/*
  29. // @require https://cdnjs.cloudflare.com/ajax/libs/qrcode-generator/1.4.4/qrcode.min.js
  30. // @resource RulerJs https://fastly.jsdelivr.net/gh/wizos/rssplus@2.0.3/Ruler.js
  31. // @grant GM_xmlhttpRequest
  32. // @grant GM_getResourceText
  33. // @grant GM_setClipboard
  34. // @grant GM_notification
  35. // @grant GM_addStyle
  36. // @grant GM_getValue
  37. // @grant GM_setValue
  38. // @grant GM_registerMenuCommand
  39. // @grant unsafeWindow
  40. // @connect rssplus.vercel.app
  41. // @noframes
  42. // @run-at document-idle
  43. // ==/UserScript==
  44.  
  45. // 2025.04.05_1.1.3 1. 修复火狐136版本无法查看 RSS 问题。2. 支持 NewsBlur 和 The Old Reader。3. 全屏时隐藏图标。4. 保持订阅源 URL 大小写。
  46. // 2025.01.16_1.1.2 1.使用 TTP 给元素增加 innerHTML。2、增加识别 feed.json。
  47. // 2024.10.20_1.1.1 1.修复未开启远程规则的问题。2.增加更多的网站适配。
  48. // 2024.10.08_1.0.9 1.增加支持葡萄牙语和法语(感谢Filipe Mota (BlackSpirits)提供的翻译)。2.支持 RSSHub 访问密钥。3.重构代码。
  49. // 2022.10.18_1.0.8 1.修复 TinyTinyRSS 订阅地址错误问题。2.修复 Bug。
  50. // 2022.07.04_1.0.7 支持用Miniflux订阅(感谢Sevichecc提供的代码https://gist.github.com/Sevichecc/f5608c4ad52e71d98f6fcf74110369df)
  51. // 2022.07.04_1.0.6 修复火狐上因为GSAP库导致无法使用问题
  52. // 2022.05.20_1.0.5 1.解决 jsdelivr 在中国被墙的问题。2.修复bilibili video页面获取feed标题异常问题。
  53. // 2022.04.18_1.0.4 修复导致网页加载卡顿的问题。
  54. // 2022.04.05_1.0.2 1.受unsafe-eval影响,无法本地执行规则时使用远程规则,改用 rssplus.vercel.app 接口。2.调整 UI,每次获取到新 RSS 都同步到 UI 中。3.精简设置项。4.监听 URL 变化,同步获取新的 RSS。
  55. // 2022.04.02_1.0.0 1.将远程规则放到 GitHub。2.修复订阅 RSS 网址的转义问题。
  56. // 2021.12.17_0.9.2 1.支持设置 FreshRSS 一键订阅。2.支持设置带端口的网址。
  57. // 2021.02.24_0.9.1 1.支持开启/关闭二维码。
  58. // 2021.02.19_0.9.0 1.支持鼠标悬停在订阅链接上时展示其二维码,方便扫码订阅。
  59. // 2021.02.05_0.8.1 1.url参数用base64编码,防止服务端获取url参数时,漏掉query部分的数据。
  60. // 2021.02.03_0.8.0 1.支持小屏幕展示。2.支持设置 TinyTinyRSS 服务的域名。
  61. // 2021.01.05_0.7.3 1.改用 GM_xmlhttpRequest。2.改用 rssfinder.vercel.app 接口。
  62. // 2020.12.16_0.7.2 1.修复 bug。
  63. // 2020.12.06_0.7.1 1.调整搞定。2.优化代码。
  64. // 2020.11.16_0.7.0 1.支持设置 InoReader 服务的域名。2.在打印页面时隐藏。3.修复影响页面样式的问题。
  65. // 2020.09.11_0.6.4 1.支持 RSSHub 服务器为 IP 地址。2.被识别的 RSS 地址不再转换为小写(因为 news.google.com 的小写地址打不开)
  66. // 2020.04.28_0.6.3 修复改了脚本name导致无法更新的bug。
  67. // 2020.04.27_0.6.2 修复rsshub domain默认为undefined的bug。
  68. // 2020.04.26_0.6.1 支持设置 RSSHub 服务的域名。
  69. // 2020.03.01_0.6 1.可设置点击“订阅”时打开的rss服务商(feedly,inoreadly)。2.修复火狐浏览器下无法展示的问题。
  70. // 2019.09.29_0.5 增加hexo站点的rss嗅探规则。
  71. // 2019.04.26_0.4.2 1.修复默认圆圈状态下宽度太宽,导致遮挡下层页面事件触发的问题。2.将icon由字体改为svg形式,修复部分站点无法显示icon的问题。3.优化RSS没有title时的默认名称。
  72. // 2018.11.10_0.4.1 关闭发现RSS后的h5通知
  73. // 2018.10.29_0.4 1.在无法链接服务器时也能展示本地的RSS;2.针对开启 Content-Security-Policy 的网站直接展示本地的RSS;3.发现RSS后,进行h5通知
  74. // 2018.10.23_0.4 1.增加识别为 wordpress 站点时,尝试使用/feed后缀;2.增加多语言支持
  75. // 2018.10.16_0.3 1.改为iframe方式显示,兼容性更好;2.改为post方式传递页面地址;
  76. // 2018.10.14_0.2 第一版 RSS+ 成型;
  77. // 2018.09.16_0.1 在 RSS+Atom Feed Subscribe Button Generator 脚本基础上增加连接后端获取feed的方式;
  78.  
  79. (function() {
  80. 'use strict';
  81.  
  82. // 过滤掉明确不包含 RSS 源的URL
  83. if (location.href.match(/(api\.wizos\.me)|(feedly\.com\/i\/subscription)|(inoreader\.com\/feed\/http)/i)) {
  84. return;
  85. }
  86. // 修复火狐上不支持 trustedTypes 问题
  87. if (typeof window.trustedTypes === 'undefined') {
  88. window.trustedTypes = {
  89. createPolicy: function(name, rules) {
  90. return rules;
  91. }
  92. };
  93. }
  94.  
  95. // 国际化文本
  96. const i18n = {
  97. zh: {
  98. noTitle: "无标题",
  99. copied: "已复制",
  100. copy: "复制",
  101. copySucceeded: "复制成功",
  102. follow: "订阅",
  103. found: "发现 ",
  104. feed: "订阅源数量:",
  105. clickToView: "点击右下角的数字查看",
  106. useFeedly: "使用 Feedly 订阅",
  107. useInoreader: "使用 Inoreader 订阅",
  108. useTinytinyrss: "使用 Tiny Tiny RSS 订阅",
  109. useFreshrss: "使用 FreshRSS 订阅",
  110. useMiniflux: "使用 Miniflux 订阅",
  111. useNewsBlur: "使用 NewsBlur 订阅",
  112. useTheOldReader: "使用 The Old Reader 订阅",
  113. settingRSShubDomain: "设置 RSSHub 的域名",
  114. settingRSShubAccessKey: "设置 RSSHub 的访问密钥",
  115. settingInoreaderDomain: "设置 Inoreader 的域名",
  116. settingTinytinyrssDomain: "设置 Tiny Tiny RSS 的域名",
  117. settingFreshrssDomain: "设置 FreshRSS 的域名",
  118. settingMinifluxDomain: "设置 Miniflux 的域名",
  119. domainIsWrong: "服务器地址格式有误,请检查",
  120. enableQRCode: "启用/禁用二维码",
  121. enabled: "已启用",
  122. disabled: "已禁用",
  123. close: "关闭"
  124. },
  125. zhtw: {
  126. noTitle: "無標題",
  127. copied: "已複製",
  128. copy: "複製",
  129. copySucceeded: "複製成功",
  130. follow: "訂閱",
  131. found: "發現 ",
  132. feed: "訂閱源數量:",
  133. clickToView: "點擊右下角的數字查看",
  134. useFeedly: "使用 Feedly 訂閱",
  135. useInoreader: "使用 Inoreader 訂閱",
  136. useTinytinyrss: "使用 Tiny Tiny RSS 訂閱",
  137. useFreshrss: "使用 FreshRSS 訂閱",
  138. useMiniflux: "使用 Miniflux 訂閱",
  139. useNewsBlur: "使用 NewsBlur 訂閱",
  140. useTheOldReader: "使用 The Old Reader 訂閱",
  141. settingRSShubDomain: "設定 RSSHub 的網域",
  142. settingRSShubAccessKey: "設定 RSSHub 的存取密鑰",
  143. settingInoreaderDomain: "設定 Inoreader 的網域",
  144. settingTinytinyrssDomain: "設定 Tiny Tiny RSS 的網域",
  145. settingFreshrssDomain: "設定 FreshRSS 的網域",
  146. settingMinifluxDomain: "設定 Miniflux 的網域",
  147. domainIsWrong: "伺服器位址格式有誤,請檢查",
  148. enableQRCode: "啟用/停用 QR 碼",
  149. enabled: "已啟用",
  150. disabled: "已停用",
  151. close: "關閉"
  152. },
  153. en: {
  154. noTitle: "Untitled",
  155. copied: "Copied",
  156. copy: "Copy",
  157. copySucceeded: "Copy succeeded",
  158. follow: "Subscribe",
  159. found: "Found ",
  160. feed: "Number of feeds: ",
  161. clickToView: "Click the number in the bottom right to view",
  162. useFeedly: "Use Feedly subscription",
  163. useInoreader: "Use Inoreader subscription",
  164. useTinytinyrss: "Use Tiny Tiny RSS subscription",
  165. useFreshrss: "Use FreshRSS subscription",
  166. useMiniflux: "Use Miniflux subscription",
  167. useNewsBlur: "Use NewsBlur subscription",
  168. useTheOldReader: "Use The Old Reader subscription",
  169. settingRSShubDomain: "Set RSSHub domain",
  170. settingRSShubAccessKey: "Set RSSHub access key",
  171. settingInoreaderDomain: "Set Inoreader domain",
  172. settingTinytinyrssDomain: "Set Tiny Tiny RSS domain",
  173. settingFreshrssDomain: "Set FreshRSS domain",
  174. settingMinifluxDomain: "Set Miniflux domain",
  175. domainIsWrong: "Error in domain name format. Please check",
  176. enableQRCode: "Enable/disable QR code",
  177. enabled: "Enabled",
  178. disabled: "Disabled",
  179. close: "Close"
  180. },
  181. ja: {
  182. noTitle: "無題",
  183. copied: "コピーしました",
  184. copy: "コピー",
  185. copySucceeded: "コピーに成功しました",
  186. follow: "購読",
  187. found: "見つかりました ",
  188. feed: "フィード数:",
  189. clickToView: "右下の数字をクリックして表示",
  190. useFeedly: "Feedly で購読",
  191. useInoreader: "Inoreader で購読",
  192. useTinytinyrss: "Tiny Tiny RSS で購読",
  193. useFreshrss: "FreshRSS で購読",
  194. useMiniflux: "Miniflux で購読",
  195. useNewsBlur: "NewsBlur で購読",
  196. useTheOldReader: "The Old Reader で購読",
  197. settingRSShubDomain: "RSSHub のドメインを設定",
  198. settingRSShubAccessKey: "RSSHub のアクセスキーを設定",
  199. settingInoreaderDomain: "Inoreader のドメインを設定",
  200. settingTinytinyrssDomain: "Tiny Tiny RSS のドメインを設定",
  201. settingFreshrssDomain: "FreshRSS のドメインを設定",
  202. settingMinifluxDomain: "Miniflux のドメインを設定",
  203. domainIsWrong: "サーバーアドレスの形式に問題があります。確認してください",
  204. enableQRCode: "QRコードの有効/無効を切り替え",
  205. enabled: "有効",
  206. disabled: "無効",
  207. close: "閉じる"
  208. },
  209. ko: {
  210. noTitle: "제목 없음",
  211. copied: "복사됨",
  212. copy: "복사",
  213. copySucceeded: "복사 성공",
  214. follow: "구독",
  215. found: "발견 ",
  216. feed: "피드 수: ",
  217. clickToView: "오른쪽 하단의 숫자를 클릭하여 보기",
  218. useFeedly: "Feedly로 구독",
  219. useInoreader: "Inoreader로 구독",
  220. useTinytinyrss: "Tiny Tiny RSS로 구독",
  221. useFreshrss: "FreshRSS로 구독",
  222. useMiniflux: "Miniflux로 구독",
  223. useNewsBlur: "NewsBlur로 구독",
  224. useTheOldReader: "The Old Reader로 구독",
  225. settingRSShubDomain: "RSSHub 도메인 설정",
  226. settingRSShubAccessKey: "RSSHub 액세스 키 설정",
  227. settingInoreaderDomain: "Inoreader 도메인 설정",
  228. settingTinytinyrssDomain: "Tiny Tiny RSS 도메인 설정",
  229. settingFreshrssDomain: "FreshRSS 도메인 설정",
  230. settingMinifluxDomain: "Miniflux 도메인 설정",
  231. domainIsWrong: "서버 주소 형식에 문제가 있습니다. 확인해 주세요",
  232. enableQRCode: "QR 코드 활성화/비활성화",
  233. enabled: "활성화됨",
  234. disabled: "비활성화됨",
  235. close: "닫기"
  236. },
  237. ptpt: {
  238. noTitle: "Sem título",
  239. copied: "Copiado",
  240. copy: "Copiar",
  241. copySucceeded: "Cópia bem-sucedida",
  242. follow: "Seguir",
  243. found: "Encontrado ",
  244. feed: "Número de feeds: ",
  245. clickToView: "Clique no número no canto inferior direito para visualizar",
  246. useFeedly: "Utilizar a subscrição do Feedly",
  247. useInoreader: "Utilizar a subscrição do InoReader",
  248. useTinytinyrss: "Utilizar a subscrição do TinyTinyRSS",
  249. useFreshrss: "Utilizar a subscrição do FreshRSS",
  250. useMiniflux: "Utilizar a subscrição do Miniflux",
  251. useNewsBlur: "Utilizar a subscrição do NewsBlur",
  252. useTheOldReader: "Utilizar a subscrição do The Old Reader",
  253. settingRSShubDomain: "Definir o domínio do RSSHub",
  254. settingRSShubAccessKey: "Definir a chave de acesso do RSSHub",
  255. settingInoreaderDomain: "Definir o domínio do InoReader",
  256. settingTinytinyrssDomain: "Definir o domínio do TinyTinyRSS",
  257. settingFreshrssDomain: "Definir o domínio do FreshRSS",
  258. settingMinifluxDomain: "Definir o domínio do Miniflux",
  259. domainIsWrong: "Erro no formato do nome do domínio. Por favor, verifica-o",
  260. enableQRCode: "Ativar/desativar o código QR",
  261. enabled: "Ativado",
  262. disabled: "Desativado",
  263. close: "Fechar"
  264. },
  265. ptbr: {
  266. noTitle: "Sem título",
  267. copied: "Copiado",
  268. copy: "Copiar",
  269. copySucceeded: "Copiado com sucesso",
  270. follow: "Seguir",
  271. found: "Encontrado ",
  272. feed: "Número de feeds: ",
  273. clickToView: "Clique no número no canto inferior direito para visualizá-lo",
  274. useFeedly: "Usar assinatura do Feedly",
  275. useInoreader: "Usar assinatura do Inoreader",
  276. useTinytinyrss: "Usar assinatura do Tiny Tiny RSS",
  277. useFreshrss: "Usar assinatura do FreshRSS",
  278. useMiniflux: "Usar assinatura do Miniflux",
  279. useNewsBlur: "Usar assinatura do NewsBlur",
  280. useTheOldReader: "Usar assinatura do The Old Reader",
  281. settingRSShubDomain: "Configurar domínio do RSSHub",
  282. settingRSShubAccessKey: "Configurar chave de acesso do RSSHub",
  283. settingInoreaderDomain: "Configurar domínio do Inoreader",
  284. settingTinytinyrssDomain: "Configurar domínio do Tiny Tiny RSS",
  285. settingFreshrssDomain: "Configurar domínio do FreshRSS",
  286. settingMinifluxDomain: "Configurar domínio do Miniflux",
  287. domainIsWrong: "Erro no formato do nome de domínio. Por favor, verifique",
  288. enableQRCode: "Ativar/desativar código QR",
  289. enabled: "Ativado",
  290. disabled: "Desativado",
  291. close: "Fechar"
  292. },
  293. fr: {
  294. noTitle: "Sans titre",
  295. copied: "Copié",
  296. copy: "Copier",
  297. copySucceeded: "Copie réussie",
  298. follow: "S'abonner",
  299. found: "Trouvé ",
  300. feed: "Nombre de flux : ",
  301. clickToView: "Cliquer sur le numéro en bas à droite pour le visualiser",
  302. useFeedly: "Utiliser l'abonnement Feedly",
  303. useInoreader: "Utiliser l'abonnement Inoreader",
  304. useTinytinyrss: "Utiliser l'abonnement Tiny Tiny RSS",
  305. useFreshrss: "Utiliser l'abonnement FreshRSS",
  306. useMiniflux: "Utiliser l'abonnement Miniflux",
  307. useNewsBlur: "Utiliser l'abonnement NewsBlur",
  308. useTheOldReader: "Utiliser l'abonnement The Old Reader",
  309. settingRSShubDomain: "Configuration du domaine RSSHub",
  310. settingRSShubAccessKey: "Configuration la clé d'accès de RSSHub",
  311. settingInoreaderDomain: "Configuration du domaine Inoreader",
  312. settingTinytinyrssDomain: "Configuration du domaine TinyTinyRSS",
  313. settingFreshrssDomain: "Configuration du domaine FreshRSS",
  314. settingMinifluxDomain: "Configuration du domaine Miniflux",
  315. domainIsWrong: "Erreur dans le format du nom de domaine. Veuillez vérifier",
  316. enableQRCode: "Activer/désactiver le code QR",
  317. enabled: "Activé",
  318. disabled: "Désactivé",
  319. close: "Fermer"
  320. }
  321. };
  322.  
  323. const ICON_CLOSE = '<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" viewBox="0 0 256 256"><path d="M205.66,194.34a8,8,0,0,1-11.32,11.32L128,139.31,61.66,205.66a8,8,0,0,1-11.32-11.32L116.69,128,50.34,61.66A8,8,0,0,1,61.66,50.34L128,116.69l66.34-66.35a8,8,0,0,1,11.32,11.32L139.31,128Z"></path></svg>'
  324. i18n.zhcn = i18n.zh;
  325. // 获取当前语言
  326. const lang = (navigator.language || 'en').replace('-', "").toLowerCase();
  327. const t = i18n[lang] || i18n.en;
  328.  
  329. GM_registerMenuCommand(t.useFeedly, function() {
  330. GM_setValue("rss_service", "feedly");
  331. });
  332. GM_registerMenuCommand(t.useInoreader, function() {
  333. const input = window.prompt(t.settingInoreaderDomain, GM_getValue("inoreader_domain", "https://www.inoreader.com"));
  334. if (input !== null) {
  335. if (isValidUrl(input)) {
  336. GM_setValue("inoreader_domain", input);
  337. GM_setValue("rss_service", "inoreader");
  338. } else {
  339. alert(t.domainIsWrong);
  340. }
  341. }
  342. });
  343. GM_registerMenuCommand(t.useTinytinyrss, function() {
  344. const input = window.prompt(t.settingTinytinyrssDomain, GM_getValue("tinytinyrss_domain", "https://www.example.com"));
  345. if (input !== null) {
  346. if (isValidUrl(input)) {
  347. GM_setValue("tinytinyrss_domain", input);
  348. GM_setValue("rss_service", "tinytinyrss");
  349. } else {
  350. alert(t.domainIsWrong);
  351. }
  352. }
  353. });
  354. GM_registerMenuCommand(t.useFreshrss, function() {
  355. const input = window.prompt(t.settingFreshrssDomain, GM_getValue("freshrss_domain", "https://www.example.com"));
  356. if (input !== null) {
  357. if (isValidUrl(input)) {
  358. GM_setValue("freshrss_domain", input);
  359. GM_setValue("rss_service", "freshrss");
  360. } else {
  361. alert(t.domainIsWrong);
  362. }
  363. }
  364. });
  365. GM_registerMenuCommand(t.useMiniflux, function() {
  366. const input = window.prompt(t.settingMinifluxDomain, GM_getValue("miniflux_domain", "https://www.example.com"));
  367. if (input !== null) {
  368. if (isValidUrl(input)) {
  369. GM_setValue("miniflux_domain", input);
  370. GM_setValue("rss_service", "miniflux");
  371. } else {
  372. alert(t.domainIsWrong);
  373. }
  374. }
  375. });
  376. GM_registerMenuCommand(t.useNewsBlur, function() {
  377. GM_setValue("rss_service", "newsblur");
  378. });
  379. GM_registerMenuCommand(t.useTheOldReader, function() {
  380. GM_setValue("rss_service", "theoldreader");
  381. });
  382. GM_registerMenuCommand(t.enableQRCode, function() {
  383. if (!GM_getValue("enable_qr_code", true)) {
  384. GM_setValue("enable_qr_code", true);
  385. GM_notification({
  386. text: t.enabled,
  387. title: t.enableQRCode,
  388. timeout: 2000
  389. });
  390. } else {
  391. GM_setValue("enable_qr_code", false);
  392. GM_notification({
  393. text: t.disabled,
  394. title: t.enableQRCode,
  395. timeout: 2000
  396. });
  397. }
  398. });
  399.  
  400. GM_registerMenuCommand(t.settingRSShubDomain, function() {
  401. const input = window.prompt(t.settingRSShubDomain, GM_getValue("rsshub_domain", "https://rsshub.app"));
  402. if (input !== null) {
  403. if (isValidUrl(input)) {
  404. GM_setValue("rsshub_domain", input);
  405. } else {
  406. alert(t.domainIsWrong);
  407. }
  408. }
  409. });
  410.  
  411. GM_registerMenuCommand(t.settingRSShubAccessKey, function() {
  412. const input = window.prompt(t.settingRSShubAccessKey, GM_getValue("rsshub_access_key"));
  413. if (input !== null) {
  414. GM_setValue("rsshub_access_key", input);
  415. }
  416. });
  417.  
  418. // 使用 forceInner 可能会报错 forceInner 被禁用
  419. // 使用 default ,这将禁用 TrustedHTML 分配(CSP)保护
  420. // window.trustedTypes.createPolicy('default', {createHTML: (string, sink) => string});
  421. // WORKAROUND: TypeError: Failed to set the 'innerHTML' property on 'Element': This document requires 'TrustedHTML' assignment.
  422. const TTP = window.trustedTypes.createPolicy("forceInner", {
  423. createHTML: (x) => x
  424. });
  425.  
  426. // 主要功能类
  427. const feedsSet = new Set();
  428. const suffixes = [
  429. '/feed', '/rss', '/rss.xml', '/atom.xml', '/feed.xml', '/rss.json', '/atom.json', '/feed.json', //'/index.xml',
  430. '/?feed=rss', '/?feed=rss2', '/blog/feed', '/blog/rss', '/latest/rss',
  431. '/news/atom', '/feed/index.xml'
  432. ];
  433. let rpDiv;
  434. let rpStyle;
  435. let rpIframe;
  436. let rpDocument;
  437. let rpBadge;
  438. let rpDialog;
  439. let rpDialogTitle;
  440. let rpDialogFeeds;
  441.  
  442. // 检测Feed
  443. function findFeeds() {
  444. const rpDiv = document.getElementById('rss-plus');
  445. if(rpDiv){
  446. return;
  447. }
  448. findKnownFeeds();
  449. findUnknownFeeds();
  450. findCloudFeeds();
  451. }
  452.  
  453. // 检测已知的Feed
  454. // 获取在<head>的<link>元素中,已经声明为RSS的链接
  455. function findKnownFeeds() {
  456. const links = document.getElementsByTagName("link");
  457. for (const link of links) {
  458. const { href, type, title = document.title } = link;
  459. if (type && (type.match(/.+\/(rss|rdf|atom|feed\+json)/i) || type.match(/^text\/xml$/i))) {
  460. addFeed(title, href);
  461. }
  462. }
  463. }
  464.  
  465. // 寻找未明确标示的RSS源
  466. function findUnknownFeeds() {
  467. // links 属性返回一个文档中所有具有 href 属性值的 <area> 元素与 <a> 元素的集合
  468. const links = document.links || document.getElementsByTagName("a");
  469. for (const link of links) {
  470. const href = link.href;
  471. if (
  472. href.match(/^(https|http|ftp|feed).*([.\/]rss([.\/]xml|\.aspx|\.jsp|\/)?$|\/node\/feed$|\/feed(\.xml|\/$|$)|\/rss\/[a-z0-9]+$|[?&;](rss|xml)=|[?&;]feed=rss[0-9.]*$|[?&;]action=rss_rc$|feeds\.feedburner\.com\/[\w\W]+$)/i)
  473. || href.match(/^(https|http|ftp|feed).*\/atom(\.xml|\.aspx|\.jsp|\/)?$|[?&;]feed=atom[0-9.]*$/i)
  474. || href.match(/^(https|http|ftp|feed).*(\/feeds?\/[^.\/]*\.xml$|.*\/index\.xml$|feed\/msgs\.xml(\?num=\d+)?$)/i)
  475. || href.match(/^(https|http|ftp|feed).*\.rdf$/i)
  476. || href.match(/^(rss|feed):\/\//i)
  477. || href.match(/^(https|http):\/\/feed\./i)
  478. ) {
  479. addFeed(link.title || link.textContent || link.innerText || document.title, href);
  480. }
  481. }
  482.  
  483. checkFeedForPlatform("html > head > link", /(wp-content)/i); // WordPress
  484. checkFeedForPlatform("html > body footer a", /(bitcron\.com|typecho\.org|hexo\.io)/i); // Blog platforms
  485. }
  486.  
  487. function findCloudFeeds() {
  488. const url = optimizeLink(location.href);
  489. const res = document.documentElement.outerHTML;
  490. try {
  491. findCloudFeedsByEval(url, res);
  492. } catch (e) {
  493. console.error("报错:", e);
  494. findCloudFeedsByAPI(url);
  495. }
  496. }
  497. function findCloudFeedsByEval(url, res) {
  498. return function(jsStr, url, res) {
  499. if (isEmpty(jsStr)) {
  500. throw new Error("未获取到可执行脚本");
  501. }
  502. const Ruler = Function(`"use strict";return (${jsStr})`)();
  503.  
  504. const list = Ruler.find(url, res);
  505. if (!list) {
  506. return;
  507. }
  508. list.forEach(element => {
  509. addFeed(element.title || document.title, element.link);
  510. });
  511. }.call(window, GM_getResourceText('RulerJs'), url, res);
  512. }
  513.  
  514. function findCloudFeedsByAPI(url) {
  515. console.log("请求远程:" + url);
  516. try {
  517. fetch(`https://rssplus.vercel.app/api/find.js?url=${encodeURIComponent(url)}`, {method: 'get', timeout: 5000}) // 设置超时
  518. .then(async response => {
  519. if (response.status === 200) {
  520. const obj = JSON.parse(await response.text());
  521. if (!obj) {
  522. return;
  523. }
  524. obj.forEach(element => {
  525. addFeed(element.title || document.title, element.link);
  526. });
  527. }
  528. })
  529. .catch(err => console.error('检测 feed 异常:', err))
  530. } catch (error) {
  531. console.error('检测 feed 异常:', error);
  532. }
  533. }
  534. function checkFeedForPlatform(selector, regex) {
  535. const links = document.querySelectorAll(selector);
  536. for (const link of links) {
  537. if (link.href.match(regex)) {
  538. checkFeed(`${document.location.protocol}//${document.domain}`);
  539. break; // 找到匹配项后立即退出
  540. }
  541. }
  542. }
  543. async function checkFeed(url) {
  544. try {
  545. const requests = suffixes.map(suffix =>
  546. fetch(url + suffix, {method: 'HEAD', timeout: 5000}) // 设置超时
  547. .then(response => {
  548. if (response.status === 200) {
  549. addFeed(document.title, url + suffix);
  550. }
  551. })
  552. .catch(err => console.error('检测 feed 异常:', err))
  553. );
  554. await Promise.all(requests); // 并行处理所有请求
  555. } catch (error) {
  556. console.error('检测 feed 异常:', error);
  557. }
  558. }
  559.  
  560. // 添加Feed
  561. function addFeed(title, url) {
  562. console.log("添加RSS:" + title + " => " + url);
  563. if (url.match(/(api\.wizos\.me)|(feedly\.com\/i\/subscription)|(inoreader\.com\/feed\/http)/i)) {
  564. return;
  565. }
  566.  
  567. if (feedsSet.size === 0){
  568. initUI();
  569. }
  570. let absoluteUrl = new URL(url, document.location.href).href.toLowerCase();
  571.  
  572. if (!feedsSet.has(absoluteUrl)) {
  573. feedsSet.add(absoluteUrl);
  574. if(url.match(/^https*:\/\/rsshub.app/i)){
  575. const rsshubDomain = GM_getValue("rsshub_domain");
  576. if(rsshubDomain != null && rsshubDomain !== ""){
  577. absoluteUrl = absoluteUrl.replace(/^https*:\/\/rsshub.app/i, rsshubDomain);
  578. }
  579.  
  580. const rsshubAccessKey = GM_getValue("rsshub_access_key");
  581. if(rsshubAccessKey != null && rsshubAccessKey !== ""){
  582. const uri = new URL(absoluteUrl);
  583. uri.searchParams.set('key', rsshubAccessKey);
  584. absoluteUrl = uri.href;
  585. }
  586. }
  587.  
  588. updateBadge();
  589. updateDialog(title || t.noTitle, url);
  590. }
  591. }
  592.  
  593. function observeUrlChange() {
  594. const _historyWrap = function(type) {
  595. const orig = history[type];
  596. const e = new Event(type);
  597. return function() {
  598. const rv = orig.apply(this, arguments);
  599. e.arguments = arguments;
  600. window.dispatchEvent(e);
  601. return rv;
  602. };
  603. };
  604. history.pushState = _historyWrap('pushState');
  605. history.replaceState = _historyWrap('replaceState');
  606.  
  607. window.addEventListener('pushState', () => {
  608. window.dispatchEvent(new Event('locationchange'));
  609. });
  610. window.addEventListener('replaceState', () => {
  611. window.dispatchEvent(new Event('locationchange'));
  612. });
  613. window.addEventListener('popstate', () => {
  614. window.dispatchEvent(new Event('locationchange'));
  615. });
  616. window.addEventListener('hashchange', function() {
  617. window.dispatchEvent(new Event('locationchange'));
  618. }, false);
  619. window.addEventListener('locationchange', ()=> {
  620. reset();
  621. findFeeds();
  622. });
  623. }
  624.  
  625. // 初始化UI
  626. function initUI() {
  627. rpStyle = document.createElement('style');
  628. rpStyle.innerHTML = TTP.createHTML(
  629. `
  630. @media print {
  631. #rss-plus {
  632. display: none;
  633. }
  634. }
  635.  
  636. #rss-plus {
  637. position: fixed;
  638. bottom: 60px;
  639. right: 5px;
  640. z-index: 9999;
  641. }
  642.  
  643. #rss-plus > iframe {
  644. display: block !important;
  645. max-width: 100% !important;
  646. border: 0px !important;
  647. }
  648. `
  649. );
  650. document.head.appendChild(rpStyle);
  651.  
  652. rpDiv = document.createElement('div');
  653. rpDiv.id = 'rss-plus';
  654. document.body.appendChild(rpDiv);
  655.  
  656. rpIframe = document.createElement("iframe");
  657. rpIframe.id = "rss-plus-iframe";
  658. rpIframe.allowTransparency = "true";
  659. if (navigator.userAgent.indexOf("Firefox") !== -1) {
  660. rpIframe.src = "javascript:";
  661. }
  662. rpDiv.appendChild(rpIframe);
  663. rpDocument = rpIframe.contentDocument || rpIframe.contentWindow.document;// || rssPlusEnvironment.window.document;
  664.  
  665. const rpBoxStyle = rpDocument.createElement("style");
  666.  
  667. rpBoxStyle.textContent = TTP.createHTML(`
  668. .hover-reveal {
  669. position: fixed;
  670. width: 80px;
  671. height: 80px;
  672. top: 0;
  673. left: 0;
  674. pointer-events: none;
  675. opacity: 0
  676. }
  677.  
  678. .hover-reveal__inner,.hover-reveal__img {
  679. width: 100%;
  680. height: 100%;
  681. position: relative
  682. }
  683.  
  684. .hover-reveal__deco {
  685. width: 100%;
  686. height: 100%;
  687. position: absolute;
  688. top: 0;
  689. left: 0;
  690. background-color: #181314
  691. }
  692.  
  693. .hover-reveal__img {
  694. background-size: cover;
  695. background-position: 50% 50%
  696. }
  697.  
  698. body {
  699. margin: 0px;
  700. }
  701.  
  702. #rp-box, #rp-badge, #rp-dialog {
  703. width: 100%;
  704. position: fixed;
  705. z-index: 99999;
  706. bottom: 0px;
  707. right: 0px;
  708. }
  709.  
  710. #rp-badge {
  711. background: #4b5979;
  712. color: white;
  713. border-radius: 50%;
  714. width: 28px;
  715. height: 28px;
  716. text-align: center;
  717. line-height: 30px;
  718. cursor: pointer;
  719. float: right;
  720. min-width: 20px;
  721. border: 1px solid #fff;
  722. white-space: nowrap;
  723. }
  724.  
  725. #rp-badge:hover {
  726. border-color: #e9eaec;
  727. }
  728.  
  729. #rp-dialog {
  730. display: none;
  731. height: 100%;
  732. }
  733.  
  734. #rp-dialog-container {
  735. height: 100%;
  736. display: flex;
  737. flex-direction: column;
  738. font-size: 14px;
  739. background: white;
  740. border-radius: 5px;
  741. border: 2px dashed rgb(209, 217, 224);
  742. }
  743.  
  744. #rp-dialog-title {
  745. display: flex;
  746. justify-content: space-between;
  747. align-items: center;
  748. gap: 10px;
  749. border-radius: 5px 5px 0 0;
  750. padding: 5px 10px 5px 10px;
  751. background-color: #f8f8f9;
  752. }
  753.  
  754. #rp-dialog-feeds-container {
  755. display: flex;
  756. flex-direction: column;
  757. align-items: stretch;
  758. gap: 5px;
  759. padding: 0px;
  760. overflow-y: auto;
  761. }
  762.  
  763. #rp-dialog-feeds {
  764. list-style-type: none;
  765. padding: 0px 10px 5px 10px;
  766. margin: 0;
  767. }
  768.  
  769. .rss-title {
  770. font-weight: bold;
  771. margin: 5px 0 5px 0;
  772. }
  773.  
  774. .rss-link {
  775. color: #666;
  776. font-size: 0.9em;
  777. word-break: break-all;
  778. margin: 5 0 5 0;
  779. display: block;
  780. white-space: nowrap;
  781. overflow: hidden;
  782. text-overflow: ellipsis;
  783. max-width: 70vw;
  784. }
  785.  
  786. .rss-actions {
  787. display: flex;
  788. justify-content: space-between;
  789. }
  790.  
  791. .rss-btn {
  792. background: #4b5979;
  793. color: white;
  794. border: none;
  795. padding: 3px 8px;
  796. margin-right: 5px;
  797. cursor: pointer;
  798. border-radius: 3px;
  799. }
  800.  
  801. .rp-dialog-count {
  802. font-weight: bold;
  803. color: #ed3f14;
  804. }
  805.  
  806. .rp-badge-count {
  807. position: relative;
  808. display: inline-block;
  809. width: 26px;
  810. height: 26px;
  811. color: #fff;
  812. text-align: center;
  813. font-size: 12px;
  814. line-height: 26px;
  815. }
  816.  
  817. .rp-badge-count a,.rp-badge-count a:hover {
  818. color: #fff
  819. }
  820.  
  821. .rp-btn {
  822. cursor: pointer;
  823. display: inline-block;
  824. border: 1px solid transparent;
  825. -ms-touch-action: manipulation;
  826. touch-action: manipulation;
  827. font-weight: 400;
  828. }
  829.  
  830. .rp-btn>.rp-icon {
  831. line-height: 1
  832. }
  833.  
  834. .rp-btn:hover {
  835. color: #6d7380;
  836. background-color: #f9f9f9;
  837. border-color: #e4e5e7
  838. }
  839.  
  840. .rp-btn>.rp-icon+span,.rp-btn>span+.rp-icon {
  841. margin-left: 4px
  842. }
  843.  
  844. .rp-btn-primary {
  845. color: #fff;
  846. background-color: #2d8cf0;
  847. border-color: #2d8cf0;
  848. }
  849.  
  850. .rp-btn-primary:hover {
  851. color: #fff;
  852. background-color: #57a3f3;
  853. border-color: #57a3f3
  854. }
  855.  
  856. .rp-btn-small {
  857. padding: 2px 7px;
  858. font-size: 12px;
  859. border-radius: 3px;
  860. margin: 5px;
  861. color: #495060;
  862. background-color: #f7f7f7;
  863. }
  864.  
  865. .rss-item {
  866. display: flex;
  867. justify-content: space-between;
  868. align-items: center;
  869. padding: 5px 0;
  870. }
  871. `);
  872. rpDocument.head.appendChild(rpBoxStyle);
  873.  
  874. const rpBadgeDiv = rpDocument.createElement("div");
  875. rpBadgeDiv.id = "rp-badge";
  876. // 方法 1
  877. //const span = rpDocument.createElement("span");
  878. //span.classList.add("rp-badge-count", "rp-count");
  879. //rpBadgeDiv.appendChild(span);
  880. // 方法 2
  881. rpBadgeDiv.innerHTML = TTP.createHTML(`<span class="rp-badge-count rp-count"></span>`);
  882. rpDocument.body.appendChild(rpBadgeDiv);
  883.  
  884. rpDialog = rpDocument.createElement("div");
  885. rpDialog.id = "rp-dialog";
  886. rpDialog.innerHTML = TTP.createHTML( `
  887. <div id="rp-dialog-container">
  888. <div id="rp-dialog-title">
  889. <div>${t.feed}<span class="rp-dialog-count rp-count"></span></div>
  890. <button type="button" id="rp-close-btn" class="rp-btn rp-btn-small" title="${t.close}"><span>${ICON_CLOSE}</span></button>
  891. </div>
  892. <div id="rp-dialog-feeds-container">
  893. <ul id="rp-dialog-feeds"></ul>
  894. </div>
  895. </div>
  896. `);
  897. rpDocument.body.appendChild(rpDialog);
  898.  
  899. rpBadge = rpDocument.getElementById("rp-badge");
  900. rpDialog = rpDocument.getElementById("rp-dialog");
  901. rpDialogTitle = rpDocument.getElementById("rp-dialog-title");
  902. rpDialogFeeds = rpDocument.getElementById("rp-dialog-feeds");
  903.  
  904. rpIframe.style.width = rpBadge.getBoundingClientRect().width + "px";
  905. rpIframe.style.height = rpBadge.getBoundingClientRect().height + "px";
  906.  
  907. addEventHandler(rpDocument.getElementById("rp-close-btn"), "click", function () {
  908. rpDialog.style.display = "none";
  909. rpBadge.style.display = "block";
  910. rpIframe.style.width = rpBadge.getBoundingClientRect().width + "px";
  911. rpIframe.style.height = rpBadge.getBoundingClientRect().height + "px";
  912. });
  913.  
  914. rpDocument.getElementById("rp-badge").addEventListener("click", () => {
  915. if (feedsSet.size === 0) {
  916. return
  917. }
  918. rpDialog.style.display = "block";
  919. rpBadge.style.display = "none";
  920.  
  921. resizeIframe();
  922. });
  923.  
  924. window.addEventListener('resize', resizeIframe);
  925. window.addEventListener('load', resizeIframe);
  926. }
  927.  
  928. function resizeIframe() {
  929. if (rpDialog.style.display === "block"){
  930. if (window.innerWidth < 400){
  931. rpIframe.style.width = `${window.innerWidth}px`;
  932. } else {
  933. rpIframe.style.width = "400px";
  934. }
  935. const dialogHeight = rpDialogFeeds.getBoundingClientRect().height + rpDialogTitle.getBoundingClientRect().height;
  936. const availableHeight = rpIframe.getBoundingClientRect().bottom;
  937. if (dialogHeight < availableHeight) {
  938. rpIframe.style.height = `${dialogHeight}px`;
  939. } else {
  940. rpIframe.style.height = `${availableHeight}px`;
  941. }
  942. }
  943. }
  944.  
  945. // 更新徽章
  946. function updateBadge() {
  947. Array.from(rpDocument.getElementsByClassName("rp-count")).forEach(el => {el.textContent = feedsSet.size});
  948. }
  949.  
  950. // 更新对话框内容
  951. function updateDialog(title, url) {
  952. const li = rpDocument.createElement('li');
  953. li.className = 'rss-item';
  954. li.innerHTML = TTP.createHTML(`
  955. <div class="rss-info">
  956. <h5 class="rss-title">${title}</h5>
  957. <a class="rss-link" href="${url}" target="_blank">${url}</a>
  958. </div>
  959. <div class="rss-actions">
  960. <button class="rss-btn rss-copy rp-btn-primary">${t.copy}</button>
  961. <button class="rss-btn rss-follow rp-btn-primary">${t.follow}</button>
  962. </div>
  963. `);
  964. li.querySelector('.rss-copy').addEventListener('click', () => copyToClipboard(url));
  965. li.querySelector('.rss-follow').addEventListener('click', () => followFeed(url));
  966. rpDialogFeeds.appendChild(li);
  967. new HoverImgQR(li.querySelector('.rss-link'));
  968. }
  969.  
  970. // 复制到剪贴板
  971. function copyToClipboard(url) {
  972. GM_setClipboard(url);
  973. GM_notification({
  974. text: t.copied,
  975. title: t.copySucceeded,
  976. timeout: 2000
  977. });
  978. }
  979.  
  980. // 关注Feed
  981. function followFeed(url) {
  982. // 这里可以根据用户设置的RSS阅读器来打开相应的订阅链接
  983. const rssService = GM_getValue("rss_service", "feedly");
  984. if (rssService === "feedly") {
  985. window.open(`https://feedly.com/i/subscription/feed/${encodeURIComponent(url)}`, '_blank');
  986. } else if (rssService === "inoreader") {
  987. window.open(GM_getValue("inoreader_domain", "https://www.inoreader.com") + `/?add_feed=${encodeURIComponent(url)}`, "_blank");
  988. } else if (rssService === "tinytinyrss") {
  989. window.open(GM_getValue("tinytinyrss_domain") + `/public.php?op=bookmarklets--subscribe&feed_url=${url}`, "_blank");
  990. } else if (rssService === "freshrss") {
  991. window.open(GM_getValue("freshrss_domain") + `/i/?c=feed&a=add&url_rss=${encodeURIComponent(url)}`, "_blank");
  992. } else if (rssService === "miniflux") {
  993. window.open(GM_getValue("miniflux_domain") + `/bookmarklet?uri=${encodeURIComponent(url)}`, "_blank");
  994. } else if (rssService === "newsblur") {
  995. window.open(`http://www.newsblur.com/?url=${encodeURIComponent(url)}`, "_blank");
  996. } else if (rssService === "theoldreader") {
  997. window.open(`https://theoldreader.com/feeds/subscribe?url=${encodeURIComponent(url)}`, "_blank");
  998. }
  999. }
  1000.  
  1001. function reset() {
  1002. if (rpDiv){
  1003. rpDiv.remove();
  1004. }
  1005. if (rpStyle){
  1006. rpStyle.remove();
  1007. }
  1008. feedsSet.clear();
  1009. }
  1010.  
  1011.  
  1012. function addEventHandler(target, eventName, eventHandler, scope) {
  1013. let f = scope ? function() {
  1014. eventHandler.apply(scope, arguments);
  1015. } : eventHandler;
  1016. if (target.addEventListener) {
  1017. target.addEventListener(eventName, f, true);
  1018. } else if (target.attachEvent) {
  1019. target.attachEvent("on" + eventName, f);
  1020. }
  1021. return f;
  1022. }
  1023.  
  1024. function optimizeLink(link) {
  1025. if (link.match(/douban\.com\/people/i)) {
  1026. const src = document.querySelector("#profile > div > div.bd > div.basic-info > div.uhead-wrap > img.userface").src;
  1027. const m = src.match(/ul(\d+)-/i);
  1028. link = "https://www.douban.com/people/" + m[1];
  1029. }
  1030. return link;
  1031. }
  1032. function isEmpty(str) {
  1033. return str.trim().length === 0;
  1034. }
  1035. function isValidUrl(url) {
  1036. return url && (url.match(/^https*:\/\/.*?\.\w+(:\d+)?(\/|$)/i) || url.match(/^https*:\/\/\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}(:\d+)*(:\d+)?(\/|$)/i));
  1037. }
  1038. function fetch(url, options = {}) {
  1039. return new Promise((resolve, reject) => {
  1040. GM_xmlhttpRequest({
  1041. url: url,
  1042. method: options.method || 'GET',
  1043. headers: options.headers,
  1044. data: options.body,
  1045. responseType: options.responseType || 'json',
  1046. onload: function(response) {
  1047. resolve({
  1048. ok: response.status >= 200 && response.status < 300,
  1049. status: response.status,
  1050. statusText: response.statusText,
  1051. json: () => {
  1052. if (options.responseType === 'json') {
  1053. return Promise.resolve(response.response); // 直接返回解析后的 JSON
  1054. } else {
  1055. try {
  1056. return Promise.resolve(JSON.parse(response.responseText)); // 手动解析 JSON
  1057. } catch (e) {
  1058. return Promise.reject(new Error('Failed to parse JSON'));
  1059. }
  1060. }
  1061. },
  1062. text: () => Promise.resolve(response.responseText),
  1063. blob: () => Promise.resolve(new Blob([response.response])),
  1064. headers: response.responseHeaders,
  1065. });
  1066. },
  1067. onerror: function(error) {
  1068. reject(error);
  1069. },
  1070. ontimeout: function() {
  1071. reject(new Error('Request timed out'));
  1072. },
  1073. timeout: options.timeout || 0,
  1074. });
  1075. });
  1076. }
  1077.  
  1078. class HoverImgQR {
  1079. constructor(el) {
  1080. this.DOM = {el: el};
  1081. this.DOM.reveal = document.createElement('div');
  1082. this.DOM.reveal.className = 'hover-reveal';
  1083.  
  1084. try {
  1085. const qr = qrcode(0, 'L');
  1086. qr.addData(this.DOM.el.href);
  1087. qr.make();
  1088. const url = qr.createDataURL();
  1089.  
  1090. const thisDOM = this.DOM;
  1091. thisDOM.reveal.innerHTML = TTP.createHTML(`<div class="hover-reveal__inner"><div class="hover-reveal__img" style="background-image:url(${url})"></div></div>`);
  1092. thisDOM.revealInner = thisDOM.reveal.querySelector('.hover-reveal__inner');
  1093. thisDOM.revealInner.style.overflow = 'hidden';
  1094. thisDOM.revealImg = thisDOM.revealInner.querySelector('.hover-reveal__img');
  1095. thisDOM.el.appendChild(thisDOM.reveal);
  1096.  
  1097. this.initEvents();
  1098. } catch (e) {
  1099. console.error("报错:", e);
  1100. }
  1101. }
  1102. getMousePos (e) {
  1103. let posX = 0;
  1104. let posY = 0;
  1105. if (!e) e = window.event;
  1106. if (e.pageX || e.pageY) {
  1107. posX = e.pageX;
  1108. posY = e.pageY;
  1109. } else if (e.clientX || e.clientY) {
  1110. posX = e.clientX + document.body.scrollLeft + document.documentElement.scrollLeft;
  1111. posY = e.clientY + document.body.scrollTop + document.documentElement.scrollTop;
  1112. }
  1113. return {
  1114. x: posX,
  1115. y: posY
  1116. }
  1117. }
  1118. initEvents() {
  1119. this.positionElement = (ev) => {
  1120. const mousePos = this.getMousePos(ev);
  1121. const docScrolls = {
  1122. left: rpDocument.body.scrollLeft + rpDocument.documentElement.scrollLeft,
  1123. top: rpDocument.body.scrollTop + rpDocument.documentElement.scrollTop
  1124. };
  1125. this.DOM.reveal.style.top = `${mousePos.y-70-docScrolls.top}px`;
  1126. this.DOM.reveal.style.left = `${mousePos.x+10-docScrolls.left}px`;
  1127. };
  1128. this.mouseenterFn = (ev) => {
  1129. if (!GM_getValue("enable_qr_code", true)) return;
  1130. this.positionElement(ev);
  1131. this.DOM.revealInner.style.overflow = 'visible';
  1132. this.DOM.reveal.style.opacity = '1';
  1133. };
  1134. this.mousemoveFn = ev => requestAnimationFrame(() => {
  1135. if (!GM_getValue("enable_qr_code", true)) return;
  1136. this.positionElement(ev);
  1137. });
  1138. this.mouseleaveFn = () => {
  1139. if (!GM_getValue("enable_qr_code", true)) return;
  1140. this.DOM.revealInner.style.overflow = 'hidden';
  1141. this.DOM.reveal.style.opacity = '0';
  1142. };
  1143. this.DOM.el.addEventListener('mouseenter', this.mouseenterFn);
  1144. this.DOM.el.addEventListener('mousemove', this.mousemoveFn);
  1145. this.DOM.el.addEventListener('mouseleave', this.mouseleaveFn);
  1146. }
  1147. }
  1148.  
  1149. // 监听全屏变化
  1150. document.addEventListener('fullscreenchange', handleFullscreenChange);
  1151. document.addEventListener('webkitfullscreenchange', handleFullscreenChange);
  1152. document.addEventListener('mozfullscreenchange', handleFullscreenChange);
  1153. function handleFullscreenChange() {
  1154. const isFullscreen = document.fullscreenElement ||
  1155. document.webkitFullscreenElement ||
  1156. document.mozFullScreenElement;
  1157. rpDiv.style.display = isFullscreen ? 'none' : 'block';
  1158. }
  1159.  
  1160.  
  1161. // 检查环境
  1162. if (window.top !== window.self || !window.location.protocol.startsWith('http')) {
  1163. return;
  1164. }
  1165. // 当页面加载完成后初始化
  1166. observeUrlChange();
  1167. findFeeds();
  1168. })();

QingJ © 2025

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