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

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

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         RSS+ : Show Site All RSS
// @name:zh      RSS+ : 显示当前网站所有的 RSS
// @name:zh-CN   RSS+ : 显示当前网站所有的 RSS
// @name:zh-TW   RSS+ : 顯示當前網站所有的 RSS
// @name:ja      RSS+ : 現在のサイトのRSSを表示
// @name:ko      RSS+ : 현재 사이트의 RSS 표시
// @name:pt-PT   RSS+ : Mostrar todos os RSS do site
// @name:pt-BR   RSS+ : Mostrar todos os RSS do site
// @name:fr      RSS+ : Afficher tous les flux RSS du site
// @description         Show All RSS Of The Site (If Any)
// @description:zh      显示当前网站的所有 RSS(如果有的话)
// @description:zh-CN   显示当前网站的所有 RSS(如果有的话)
// @description:zh-TW   顯示當前網站的所有 RSS(如果有的话)
// @description:ja      サイトのすべてのRSSを表示します (あれば)
// @description:ko      웹 사이트의 모든 RSS 를 표시합니다 (있는 경우)
// @description:pt-PT   Mostra todos os feeds RSS do site (se houver)
// @description:pt-BR   Mostra todos os feeds RSS do site (se houver)
// @description:fr      Montre tous les flux RSS du site (s'il y en a)
// @license      GPL3.0
// @version      1.1.4

// @icon         
// @author       Wizos
// @namespace    https://blog.wizos.me
// @supportURL   [email protected]
// @match        http://*/*
// @match        https://*/*
// @require      https://cdnjs.cloudflare.com/ajax/libs/qrcode-generator/1.4.4/qrcode.min.js
// @resource     RulerJs  https://fastly.jsdelivr.net/gh/wizos/rssplus/Ruler.js
// @grant        GM_xmlhttpRequest
// @grant        GM_getResourceText
// @grant        GM_setClipboard
// @grant        GM_notification
// @grant        GM_addStyle
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_registerMenuCommand
// @grant        unsafeWindow
// @connect      rssplus.vercel.app
// @connect      rssplus.pages.dev
// @connect      rssfind.val.run
// @connect      rssplus.deno.dev
// @connect      rssplus.mdict.workers.dev
// @noframes
// @run-at     document-idle
// ==/UserScript==

// 2025.06.29_1.1.4  1.由于流量过大,增加 rssplus.pages.dev, rssfind.val.run, rssplus.deno.dev, rssplus.mdict.workers.dev 域名。2.修复无法使用自定义 RSSHub 域名。3.增加 Folo 订阅。
// 2025.04.05_1.1.3  1.修复火狐136版本无法查看 RSS 问题。2. 支持 NewsBlur 和 The Old Reader。3. 全屏时隐藏图标。4. 保持订阅源 URL 大小写。
// 2025.01.16_1.1.2  1.使用 TTP 给元素增加 innerHTML。2、增加识别 feed.json。
// 2024.10.20_1.1.1  1.修复未开启远程规则的问题。2.增加更多的网站适配。
// 2024.10.08_1.0.9  1.增加支持葡萄牙语和法语(感谢Filipe Mota (BlackSpirits)提供的翻译)。2.支持 RSSHub 访问密钥。3.重构代码。
// 2022.10.18_1.0.8  1.修复 TinyTinyRSS 订阅地址错误问题。2.修复 Bug。
// 2022.07.04_1.0.7  支持用Miniflux订阅(感谢Sevichecc提供的代码https://gist.github.com/Sevichecc/f5608c4ad52e71d98f6fcf74110369df)
// 2022.07.04_1.0.6  修复火狐上因为GSAP库导致无法使用问题
// 2022.05.20_1.0.5  1.解决 jsdelivr 在中国被墙的问题。2.修复bilibili video页面获取feed标题异常问题。
// 2022.04.18_1.0.4  修复导致网页加载卡顿的问题。
// 2022.04.05_1.0.2  1.受unsafe-eval影响,无法本地执行规则时使用远程规则,改用 rssplus.vercel.app 接口。2.调整 UI,每次获取到新 RSS 都同步到 UI 中。3.精简设置项。4.监听 URL 变化,同步获取新的 RSS。
// 2022.04.02_1.0.0  1.将远程规则放到 GitHub。2.修复订阅 RSS 网址的转义问题。
// 2021.12.17_0.9.2  1.支持设置 FreshRSS 一键订阅。2.支持设置带端口的网址。
// 2021.02.24_0.9.1  1.支持开启/关闭二维码。
// 2021.02.19_0.9.0  1.支持鼠标悬停在订阅链接上时展示其二维码,方便扫码订阅。
// 2021.02.05_0.8.1  1.url参数用base64编码,防止服务端获取url参数时,漏掉query部分的数据。
// 2021.02.03_0.8.0  1.支持小屏幕展示。2.支持设置 TinyTinyRSS 服务的域名。
// 2021.01.05_0.7.3  1.改用 GM_xmlhttpRequest。2.改用 rssfinder.vercel.app 接口。
// 2020.12.16_0.7.2  1.修复 bug。
// 2020.12.06_0.7.1  1.调整搞定。2.优化代码。
// 2020.11.16_0.7.0  1.支持设置 InoReader 服务的域名。2.在打印页面时隐藏。3.修复影响页面样式的问题。
// 2020.09.11_0.6.4  1.支持 RSSHub 服务器为 IP 地址。2.被识别的 RSS 地址不再转换为小写(因为 news.google.com 的小写地址打不开)
// 2020.04.28_0.6.3  修复改了脚本name导致无法更新的bug。
// 2020.04.27_0.6.2  修复rsshub domain默认为undefined的bug。
// 2020.04.26_0.6.1  支持设置 RSSHub 服务的域名。
// 2020.03.01_0.6    1.可设置点击“订阅”时打开的rss服务商(feedly,inoreadly)。2.修复火狐浏览器下无法展示的问题。
// 2019.09.29_0.5    增加hexo站点的rss嗅探规则。
// 2019.04.26_0.4.2  1.修复默认圆圈状态下宽度太宽,导致遮挡下层页面事件触发的问题。2.将icon由字体改为svg形式,修复部分站点无法显示icon的问题。3.优化RSS没有title时的默认名称。
// 2018.11.10_0.4.1  关闭发现RSS后的h5通知
// 2018.10.29_0.4    1.在无法链接服务器时也能展示本地的RSS;2.针对开启 Content-Security-Policy 的网站直接展示本地的RSS;3.发现RSS后,进行h5通知
// 2018.10.23_0.4    1.增加识别为 wordpress 站点时,尝试使用/feed后缀;2.增加多语言支持
// 2018.10.16_0.3    1.改为iframe方式显示,兼容性更好;2.改为post方式传递页面地址;
// 2018.10.14_0.2    第一版 RSS+ 成型;
// 2018.09.16_0.1    在 RSS+Atom Feed Subscribe Button Generator 脚本基础上增加连接后端获取feed的方式;

(function() {
    'use strict';

    // 过滤掉明确不包含 RSS 源的URL
    if (location.href.match(/(api\.wizos\.me)|(feedly\.com\/i\/subscription)|(inoreader\.com\/feed\/http)/i)) {
        return;
    }
    // 修复火狐上不支持 trustedTypes 问题
    if (typeof window.trustedTypes === 'undefined') {
        window.trustedTypes = {
            createPolicy: function(name, rules) {
                return rules;
            }
        };
    }

    // 国际化文本
    const i18n = {
        zh: {
            noTitle: "无标题",
            copied: "已复制",
            copy: "复制",
            copySucceeded: "复制成功",
            follow: "订阅",
            found: "发现 ",
            feed: "订阅源数量:",
            clickToView: "点击右下角的数字查看",
            useFeedly: "使用 Feedly 订阅",
            useInoreader: "使用 Inoreader 订阅",
            useTinytinyrss: "使用 Tiny Tiny RSS 订阅",
            useFreshrss: "使用 FreshRSS 订阅",
            useMiniflux: "使用 Miniflux 订阅",
            useFoloReader: "使用 Folo 订阅",
            useNewsBlur: "使用 NewsBlur 订阅",
            useTheOldReader: "使用 The Old Reader 订阅",
            settingRSShubDomain: "设置 RSSHub 的域名",
            settingRSShubAccessKey: "设置 RSSHub 的访问密钥",
            settingInoreaderDomain: "设置 Inoreader 的域名",
            settingTinytinyrssDomain: "设置 Tiny Tiny RSS 的域名",
            settingFreshrssDomain: "设置 FreshRSS 的域名",
            settingMinifluxDomain: "设置 Miniflux 的域名",
            domainIsWrong: "服务器地址格式有误,请检查",
            enableQRCode: "启用/禁用二维码",
            enabled: "已启用",
            disabled: "已禁用",
            close: "关闭"
        },
        zhtw: {
            noTitle: "無標題",
            copied: "已複製",
            copy: "複製",
            copySucceeded: "複製成功",
            follow: "訂閱",
            found: "發現 ",
            feed: "訂閱源數量:",
            clickToView: "點擊右下角的數字查看",
            useFeedly: "使用 Feedly 訂閱",
            useInoreader: "使用 Inoreader 訂閱",
            useTinytinyrss: "使用 Tiny Tiny RSS 訂閱",
            useFreshrss: "使用 FreshRSS 訂閱",
            useMiniflux: "使用 Miniflux 訂閱",
            useFoloReader: "使用 Folo 訂閱",
            useNewsBlur: "使用 NewsBlur 訂閱",
            useTheOldReader: "使用 The Old Reader 訂閱",
            settingRSShubDomain: "設定 RSSHub 的網域",
            settingRSShubAccessKey: "設定 RSSHub 的存取密鑰",
            settingInoreaderDomain: "設定 Inoreader 的網域",
            settingTinytinyrssDomain: "設定 Tiny Tiny RSS 的網域",
            settingFreshrssDomain: "設定 FreshRSS 的網域",
            settingMinifluxDomain: "設定 Miniflux 的網域",
            domainIsWrong: "伺服器位址格式有誤,請檢查",
            enableQRCode: "啟用/停用 QR 碼",
            enabled: "已啟用",
            disabled: "已停用",
            close: "關閉"
        },
        en: {
            noTitle: "Untitled",
            copied: "Copied",
            copy: "Copy",
            copySucceeded: "Copy succeeded",
            follow: "Subscribe",
            found: "Found ",
            feed: "Number of feeds: ",
            clickToView: "Click the number in the bottom right to view",
            useFeedly: "Use Feedly subscription",
            useInoreader: "Use Inoreader subscription",
            useTinytinyrss: "Use Tiny Tiny RSS subscription",
            useFreshrss: "Use FreshRSS subscription",
            useMiniflux: "Use Miniflux subscription",
            useFoloReader: "Use Folo subscription",
            useNewsBlur: "Use NewsBlur subscription",
            useTheOldReader: "Use The Old Reader subscription",
            settingRSShubDomain: "Set RSSHub domain",
            settingRSShubAccessKey: "Set RSSHub access key",
            settingInoreaderDomain: "Set Inoreader domain",
            settingTinytinyrssDomain: "Set Tiny Tiny RSS domain",
            settingFreshrssDomain: "Set FreshRSS domain",
            settingMinifluxDomain: "Set Miniflux domain",
            domainIsWrong: "Error in domain name format. Please check",
            enableQRCode: "Enable/disable QR code",
            enabled: "Enabled",
            disabled: "Disabled",
            close: "Close"
        },
        ja: {
            noTitle: "無題",
            copied: "コピーしました",
            copy: "コピー",
            copySucceeded: "コピーに成功しました",
            follow: "購読",
            found: "見つかりました ",
            feed: "フィード数:",
            clickToView: "右下の数字をクリックして表示",
            useFeedly: "Feedly で購読",
            useInoreader: "Inoreader で購読",
            useTinytinyrss: "Tiny Tiny RSS で購読",
            useFreshrss: "FreshRSS で購読",
            useMiniflux: "Miniflux で購読",
            useFoloReader: "Folo で購読",
            useNewsBlur: "NewsBlur で購読",
            useTheOldReader: "The Old Reader で購読",
            settingRSShubDomain: "RSSHub のドメインを設定",
            settingRSShubAccessKey: "RSSHub のアクセスキーを設定",
            settingInoreaderDomain: "Inoreader のドメインを設定",
            settingTinytinyrssDomain: "Tiny Tiny RSS のドメインを設定",
            settingFreshrssDomain: "FreshRSS のドメインを設定",
            settingMinifluxDomain: "Miniflux のドメインを設定",
            domainIsWrong: "サーバーアドレスの形式に問題があります。確認してください",
            enableQRCode: "QRコードの有効/無効を切り替え",
            enabled: "有効",
            disabled: "無効",
            close: "閉じる"
        },
        ko: {
            noTitle: "제목 없음",
            copied: "복사됨",
            copy: "복사",
            copySucceeded: "복사 성공",
            follow: "구독",
            found: "발견 ",
            feed: "피드 수: ",
            clickToView: "오른쪽 하단의 숫자를 클릭하여 보기",
            useFeedly: "Feedly로 구독",
            useInoreader: "Inoreader로 구독",
            useTinytinyrss: "Tiny Tiny RSS로 구독",
            useFreshrss: "FreshRSS로 구독",
            useMiniflux: "Miniflux로 구독",
            useFoloReader: "Folo로 구독",
            useNewsBlur: "NewsBlur로 구독",
            useTheOldReader: "The Old Reader로 구독",
            settingRSShubDomain: "RSSHub 도메인 설정",
            settingRSShubAccessKey: "RSSHub 액세스 키 설정",
            settingInoreaderDomain: "Inoreader 도메인 설정",
            settingTinytinyrssDomain: "Tiny Tiny RSS 도메인 설정",
            settingFreshrssDomain: "FreshRSS 도메인 설정",
            settingMinifluxDomain: "Miniflux 도메인 설정",
            domainIsWrong: "서버 주소 형식에 문제가 있습니다. 확인해 주세요",
            enableQRCode: "QR 코드 활성화/비활성화",
            enabled: "활성화됨",
            disabled: "비활성화됨",
            close: "닫기"
        },
        ptpt: {
            noTitle: "Sem título",
            copied: "Copiado",
            copy: "Copiar",
            copySucceeded: "Cópia bem-sucedida",
            follow: "Seguir",
            found: "Encontrado ",
            feed: "Número de feeds: ",
            clickToView: "Clique no número no canto inferior direito para visualizar",
            useFeedly: "Utilizar a subscrição do Feedly",
            useInoreader: "Utilizar a subscrição do InoReader",
            useTinytinyrss: "Utilizar a subscrição do TinyTinyRSS",
            useFreshrss: "Utilizar a subscrição do FreshRSS",
            useMiniflux: "Utilizar a subscrição do Miniflux",
            useFoloReader: "Utilizar a subscrição do Folo",
            useNewsBlur: "Utilizar a subscrição do NewsBlur",
            useTheOldReader: "Utilizar a subscrição do The Old Reader",
            settingRSShubDomain: "Definir o domínio do RSSHub",
            settingRSShubAccessKey: "Definir a chave de acesso do RSSHub",
            settingInoreaderDomain: "Definir o domínio do InoReader",
            settingTinytinyrssDomain: "Definir o domínio do TinyTinyRSS",
            settingFreshrssDomain: "Definir o domínio do FreshRSS",
            settingMinifluxDomain: "Definir o domínio do Miniflux",
            domainIsWrong: "Erro no formato do nome do domínio. Por favor, verifica-o",
            enableQRCode: "Ativar/desativar o código QR",
            enabled: "Ativado",
            disabled: "Desativado",
            close: "Fechar"
        },
        ptbr: {
            noTitle: "Sem título",
            copied: "Copiado",
            copy: "Copiar",
            copySucceeded: "Copiado com sucesso",
            follow: "Seguir",
            found: "Encontrado ",
            feed: "Número de feeds: ",
            clickToView: "Clique no número no canto inferior direito para visualizá-lo",
            useFeedly: "Usar assinatura do Feedly",
            useInoreader: "Usar assinatura do Inoreader",
            useTinytinyrss: "Usar assinatura do Tiny Tiny RSS",
            useFreshrss: "Usar assinatura do FreshRSS",
            useMiniflux: "Usar assinatura do Miniflux",
            useFoloReader: "Usar assinatura do Folo",
            useNewsBlur: "Usar assinatura do NewsBlur",
            useTheOldReader: "Usar assinatura do The Old Reader",
            settingRSShubDomain: "Configurar domínio do RSSHub",
            settingRSShubAccessKey: "Configurar chave de acesso do RSSHub",
            settingInoreaderDomain: "Configurar domínio do Inoreader",
            settingTinytinyrssDomain: "Configurar domínio do Tiny Tiny RSS",
            settingFreshrssDomain: "Configurar domínio do FreshRSS",
            settingMinifluxDomain: "Configurar domínio do Miniflux",
            domainIsWrong: "Erro no formato do nome de domínio. Por favor, verifique",
            enableQRCode: "Ativar/desativar código QR",
            enabled: "Ativado",
            disabled: "Desativado",
            close: "Fechar"
        },
        fr: {
            noTitle: "Sans titre",
            copied: "Copié",
            copy: "Copier",
            copySucceeded: "Copie réussie",
            follow: "S'abonner",
            found: "Trouvé ",
            feed: "Nombre de flux : ",
            clickToView: "Cliquer sur le numéro en bas à droite pour le visualiser",
            useFeedly: "Utiliser l'abonnement Feedly",
            useInoreader: "Utiliser l'abonnement Inoreader",
            useTinytinyrss: "Utiliser l'abonnement Tiny Tiny RSS",
            useFreshrss: "Utiliser l'abonnement FreshRSS",
            useMiniflux: "Utiliser l'abonnement Miniflux",
            useFoloReader: "Utiliser l'abonnement Folo",
            useNewsBlur: "Utiliser l'abonnement NewsBlur",
            useTheOldReader: "Utiliser l'abonnement The Old Reader",
            settingRSShubDomain: "Configuration du domaine RSSHub",
            settingRSShubAccessKey: "Configuration la clé d'accès de RSSHub",
            settingInoreaderDomain: "Configuration du domaine Inoreader",
            settingTinytinyrssDomain: "Configuration du domaine TinyTinyRSS",
            settingFreshrssDomain: "Configuration du domaine FreshRSS",
            settingMinifluxDomain: "Configuration du domaine Miniflux",
            domainIsWrong: "Erreur dans le format du nom de domaine. Veuillez vérifier",
            enableQRCode: "Activer/désactiver le code QR",
            enabled: "Activé",
            disabled: "Désactivé",
            close: "Fermer"
        }
    };

    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>'
    i18n.zhcn = i18n.zh;
    // 获取当前语言
    const lang = (navigator.language || 'en').replace('-', "").toLowerCase();
    const t = i18n[lang] || i18n.en;

    const proxyList = [
        "https://rssplus.vercel.app/",
        "https://rssplus.pages.dev/",
        "https://rssfind.val.run/",
        "https://rssplus.deno.dev/",
        "https://rssplus.mdict.workers.dev/"
    ];

    GM_registerMenuCommand(t.useFeedly, function() {
        GM_setValue("rss_service", "feedly");
    });
    GM_registerMenuCommand(t.useInoreader, function() {
        const input = window.prompt(t.settingInoreaderDomain, GM_getValue("inoreader_domain", "https://www.inoreader.com"));
        if (input !== null) {
            if (isValidUrl(input)) {
                GM_setValue("inoreader_domain", input);
                GM_setValue("rss_service", "inoreader");
            } else {
                alert(t.domainIsWrong);
            }
        }
    });
    GM_registerMenuCommand(t.useTinytinyrss, function() {
        const input = window.prompt(t.settingTinytinyrssDomain, GM_getValue("tinytinyrss_domain", "https://www.example.com"));
        if (input !== null) {
            if (isValidUrl(input)) {
                GM_setValue("tinytinyrss_domain", input);
                GM_setValue("rss_service", "tinytinyrss");
            } else {
                alert(t.domainIsWrong);
            }
        }
    });
    GM_registerMenuCommand(t.useFreshrss, function() {
        const input = window.prompt(t.settingFreshrssDomain, GM_getValue("freshrss_domain", "https://www.example.com"));
        if (input !== null) {
            if (isValidUrl(input)) {
                GM_setValue("freshrss_domain", input);
                GM_setValue("rss_service", "freshrss");
            } else {
                alert(t.domainIsWrong);
            }
        }
    });
    GM_registerMenuCommand(t.useMiniflux, function() {
        const input = window.prompt(t.settingMinifluxDomain, GM_getValue("miniflux_domain", "https://www.example.com"));
        if (input !== null) {
            if (isValidUrl(input)) {
                GM_setValue("miniflux_domain", input);
                GM_setValue("rss_service", "miniflux");
            } else {
                alert(t.domainIsWrong);
            }
        }
    });
    GM_registerMenuCommand(t.useFoloReader, function() {
        GM_setValue("rss_service", "folo");
    });
    GM_registerMenuCommand(t.useNewsBlur, function() {
        GM_setValue("rss_service", "newsblur");
    });
    GM_registerMenuCommand(t.useTheOldReader, function() {
        GM_setValue("rss_service", "theoldreader");
    });
    GM_registerMenuCommand(t.enableQRCode, function() {
        if (!GM_getValue("enable_qr_code", true)) {
            GM_setValue("enable_qr_code", true);
            GM_notification({
                text: t.enabled,
                title: t.enableQRCode,
                timeout: 2000
            });
        } else {
            GM_setValue("enable_qr_code", false);
            GM_notification({
                text: t.disabled,
                title: t.enableQRCode,
                timeout: 2000
            });
        }
    });

    GM_registerMenuCommand(t.settingRSShubDomain, function() {
        const input = window.prompt(t.settingRSShubDomain, GM_getValue("rsshub_domain", "https://rsshub.app"));
        if (input !== null) {
            if (isValidUrl(input)) {
                GM_setValue("rsshub_domain", input);
            } else {
                alert(t.domainIsWrong);
            }
        }
    });

    GM_registerMenuCommand(t.settingRSShubAccessKey, function() {
        const input = window.prompt(t.settingRSShubAccessKey, GM_getValue("rsshub_access_key"));
        if (input !== null) {
            GM_setValue("rsshub_access_key", input);
        }
    });

    // 使用 forceInner 可能会报错 forceInner 被禁用
    // 使用 default ,这将禁用 TrustedHTML 分配(CSP)保护
    // window.trustedTypes.createPolicy('default', {createHTML: (string, sink) => string});
    // WORKAROUND: TypeError: Failed to set the 'innerHTML' property on 'Element': This document requires 'TrustedHTML' assignment.
    const TTP = window.trustedTypes.createPolicy("forceInner", {
        createHTML: (x) => x
    });

    // 主要功能类
    const feedsSet = new Set();
    const suffixes = [
        '/feed', '/rss', '/rss.xml', '/atom.xml', '/feed.xml', '/rss.json', '/atom.json', '/feed.json', //'/index.xml',
        '/?feed=rss', '/?feed=rss2', '/blog/feed', '/blog/rss', '/latest/rss',
        '/news/atom', '/feed/index.xml'
    ];
    let rpDiv;
    let rpStyle;
    let rpIframe;
    let rpDocument;
    let rpBadge;
    let rpDialog;
    let rpDialogTitle;
    let rpDialogFeeds;

    // 检测Feed
    function findFeeds() {
        const rpDiv = document.getElementById('rss-plus');
        if(rpDiv){
            return;
        }
        findKnownFeeds();
        findUnknownFeeds();
        findCloudFeeds();
    }

    // 检测已知的Feed
    // 获取在<head>的<link>元素中,已经声明为RSS的链接
    function findKnownFeeds() {
        const links = document.getElementsByTagName("link");
        for (const link of links) {
            const { href, type, title = document.title } = link;
            if (type && (type.match(/.+\/(rss|rdf|atom|feed\+json)/i) || type.match(/^text\/xml$/i))) {
                addFeed(title, href);
            }
        }
    }

    // 寻找未明确标示的RSS源
    function findUnknownFeeds() {
        // links 属性返回一个文档中所有具有 href 属性值的 <area> 元素与 <a> 元素的集合
        const links = document.links || document.getElementsByTagName("a");
        for (const link of links) {
            const href = link.href;
            if (
                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)
                || href.match(/^(https|http|ftp|feed).*\/atom(\.xml|\.aspx|\.jsp|\/)?$|[?&;]feed=atom[0-9.]*$/i)
                || href.match(/^(https|http|ftp|feed).*(\/feeds?\/[^.\/]*\.xml$|.*\/index\.xml$|feed\/msgs\.xml(\?num=\d+)?$)/i)
                || href.match(/^(https|http|ftp|feed).*\.rdf$/i)
                || href.match(/^(rss|feed):\/\//i)
                || href.match(/^(https|http):\/\/feed\./i)
            ) {
                addFeed(link.title || link.textContent || link.innerText || document.title, href);
            }
        }

        checkFeedForPlatform("html > head > link", /(wp-content)/i); // WordPress
        checkFeedForPlatform("html > body footer a", /(bitcron\.com|typecho\.org|hexo\.io)/i); // Blog platforms
    }

    function findCloudFeeds() {
        const url = optimizeLink(location.href);
        const res = document.documentElement.outerHTML;
        try {
            findCloudFeedsByEval(url, res);
        } catch (e) {
            console.error("无法通过 Eval 执行:", e);
            findCloudFeedsByAPI(url);
        }
    }
    function findCloudFeedsByEval(url, res) {
        return function(jsStr, url, res) {
            if (isEmpty(jsStr)) {
                throw new Error("未获取到可执行脚本");
            }
            const Ruler = Function(`"use strict";return (${jsStr})`)();

            const list = Ruler.find(url, res);
            if (!list) {
                return;
            }
            list.forEach(element => {
                addFeed(element.title || document.title, element.link);
            });
        }.call(window, GM_getResourceText('RulerJs'), url, res);
    }

    function findCloudFeedsByAPI(url) {
        console.log("请求远程:" + url);
        try {
            fetch(`${proxyList[Math.floor(Math.random() * proxyList.length)]}api/find?url=${encodeURIComponent(url)}`, {method: 'get', timeout: 5000}) // 设置超时
                .then(async response => {
                    if (response.status === 200) {
                        const obj = JSON.parse(await response.text());
                        if (!obj) {
                            return;
                        }
                        obj.forEach(element => {
                            addFeed(element.title || document.title, element.link);
                        });
                    }
                })
                .catch(err => console.error('检测 feed 异常:', err))
        } catch (error) {
            console.error('检测 feed 异常:', error);
        }
    }
    function checkFeedForPlatform(selector, regex) {
        const links = document.querySelectorAll(selector);
        for (const link of links) {
            if (link.href.match(regex)) {
                checkFeed(`${document.location.protocol}//${document.domain}`);
                break; // 找到匹配项后立即退出
            }
        }
    }
    async function checkFeed(url) {
        try {
            const requests = suffixes.map(suffix =>
                fetch(url + suffix, {method: 'HEAD', timeout: 5000}) // 设置超时
                    .then(response => {
                        if (response.status === 200) {
                            addFeed(document.title, url + suffix);
                        }
                    })
                    .catch(err => console.error('检测 feed 异常:', err))
            );
            await Promise.all(requests); // 并行处理所有请求
        } catch (error) {
            console.error('检测 feed 异常:', error);
        }
    }

    // 添加Feed
    function addFeed(title, url) {
        console.log("添加RSS:" + title + " => " + url);
        if (url.match(/(api\.wizos\.me)|(feedly\.com\/i\/subscription)|(inoreader\.com\/feed\/http)/i)) {
            return;
        }

        if (feedsSet.size === 0){
            initUI();
        }
        let absoluteUrl = new URL(url, document.location.href).href;
        let lookupValue = absoluteUrl.toLowerCase();

        if (!feedsSet.has(lookupValue)) {
            feedsSet.add(lookupValue);
            if(url.match(/^https*:\/\/rsshub.app/i)){
                const rsshubDomain = GM_getValue("rsshub_domain");
                if(rsshubDomain != null && rsshubDomain !== ""){
                    absoluteUrl = absoluteUrl.replace(/^https*:\/\/rsshub.app/i, rsshubDomain);
                }

                const rsshubAccessKey = GM_getValue("rsshub_access_key");
                if(rsshubAccessKey != null && rsshubAccessKey !== ""){
                    const uri = new URL(absoluteUrl);
                    uri.searchParams.set('key', rsshubAccessKey);
                    absoluteUrl = uri.href;
                }
            }

            updateBadge();
            updateDialog(title || t.noTitle, absoluteUrl);
        }
    }

    function observeUrlChange() {
        const _historyWrap = function(type) {
            const orig = history[type];
            const e = new Event(type);
            return function() {
                const rv = orig.apply(this, arguments);
                e.arguments = arguments;
                window.dispatchEvent(e);
                return rv;
            };
        };
        history.pushState = _historyWrap('pushState');
        history.replaceState = _historyWrap('replaceState');

        window.addEventListener('pushState', () => {
            window.dispatchEvent(new Event('locationchange'));
        });
        window.addEventListener('replaceState', () => {
            window.dispatchEvent(new Event('locationchange'));
        });
        window.addEventListener('popstate', () => {
            window.dispatchEvent(new Event('locationchange'));
        });
        window.addEventListener('hashchange', function() {
            window.dispatchEvent(new Event('locationchange'));
        }, false);
        window.addEventListener('locationchange', ()=> {
            reset();
            findFeeds();
        });
    }

    // 初始化UI
    function initUI() {
        rpStyle = document.createElement('style');
        rpStyle.innerHTML = TTP.createHTML(
            `
@media print {
    #rss-plus {
        display: none;
    }
}

#rss-plus {
    position: fixed;
    bottom: 60px;
    right: 5px;
    z-index: 9999;
}

#rss-plus > iframe {
    display: block !important;
    max-width: 100% !important;
    border: 0px !important;
}
            `
        );
        document.head.appendChild(rpStyle);

        rpDiv = document.createElement('div');
        rpDiv.id = 'rss-plus';
        document.body.appendChild(rpDiv);

        rpIframe = document.createElement("iframe");
        rpIframe.id = "rss-plus-iframe";
        rpIframe.allowTransparency = "true";
        if (navigator.userAgent.indexOf("Firefox") !== -1) {
            rpIframe.src = "javascript:";
        }
        rpDiv.appendChild(rpIframe);
        rpDocument = rpIframe.contentDocument || rpIframe.contentWindow.document;// || rssPlusEnvironment.window.document;

        const rpBoxStyle = rpDocument.createElement("style");

        rpBoxStyle.textContent = TTP.createHTML(`
.hover-reveal {
    position: fixed;
    width: 80px;
    height: 80px;
    top: 0;
    left: 0;
    pointer-events: none;
    opacity: 0
}

.hover-reveal__inner,.hover-reveal__img {
    width: 100%;
    height: 100%;
    position: relative
}

.hover-reveal__deco {
    width: 100%;
    height: 100%;
    position: absolute;
    top: 0;
    left: 0;
    background-color: #181314
}

.hover-reveal__img {
    background-size: cover;
    background-position: 50% 50%
}

body {
    margin: 0px;
}

#rp-box, #rp-badge, #rp-dialog {
    width: 100%;
    position: fixed;
    z-index: 99999;
    bottom: 0px;
    right: 0px;
}

#rp-badge {
    background: #4b5979;
    color: white;
    border-radius: 50%;
    width: 28px;
    height: 28px;
    text-align: center;
    line-height: 30px;
    cursor: pointer;
    float: right;
    min-width: 20px;
    border: 1px solid #fff;
    white-space: nowrap;
}

#rp-badge:hover {
    border-color: #e9eaec;
}

#rp-dialog {
    display: none;
    height: 100%;
}

#rp-dialog-container {
    height: 100%;
    display: flex;
    flex-direction: column;
    font-size: 14px;
    background: white;
    border-radius: 5px;
    border: 2px dashed rgb(209, 217, 224);
}

#rp-dialog-title {
    display: flex;
    justify-content: space-between;
    align-items: center;
    gap: 10px;
    border-radius: 5px 5px 0 0;
    padding: 5px 10px 5px 10px;
    background-color: #f8f8f9;
}

#rp-dialog-feeds-container {
    display: flex;
    flex-direction: column;
    align-items: stretch;
    gap: 5px;
    padding: 0px;
    overflow-y: auto;
}

#rp-dialog-feeds {
    list-style-type: none;
    padding: 0px 10px 5px 10px;
    margin: 0;
}

.rss-title {
    font-weight: bold;
    margin: 5px 0 5px 0;
}

.rss-link {
    color: #666;
    font-size: 0.9em;
    word-break: break-all;
    margin: 5 0 5 0;
    display: block;
    white-space: nowrap;
    overflow: hidden;
    text-overflow: ellipsis;
    max-width: 70vw;
}

.rss-actions {
    display: flex;
    justify-content: space-between;
}

.rss-btn {
    background: #4b5979;
    color: white;
    border: none;
    padding: 3px 8px;
    margin-right: 5px;
    cursor: pointer;
    border-radius: 3px;
}

.rp-dialog-count {
    font-weight: bold;
    color: #ed3f14;
}

.rp-badge-count {
    position: relative;
    display: inline-block;
    width: 26px;
    height: 26px;
    color: #fff;
    text-align: center;
    font-size: 12px;
    line-height: 26px;
}

.rp-badge-count a,.rp-badge-count a:hover {
    color: #fff
}

.rp-btn {
    cursor: pointer;
    display: inline-block;
    border: 1px solid transparent;
    -ms-touch-action: manipulation;
    touch-action: manipulation;
    font-weight: 400;
}

.rp-btn>.rp-icon {
    line-height: 1
}

.rp-btn:hover {
    color: #6d7380;
    background-color: #f9f9f9;
    border-color: #e4e5e7
}

.rp-btn>.rp-icon+span,.rp-btn>span+.rp-icon {
    margin-left: 4px
}

.rp-btn-primary {
    color: #fff;
    background-color: #2d8cf0;
    border-color: #2d8cf0;
}

.rp-btn-primary:hover {
    color: #fff;
    background-color: #57a3f3;
    border-color: #57a3f3
}

.rp-btn-small {
    padding: 2px 7px;
    font-size: 12px;
    border-radius: 3px;
    margin: 5px;
    color: #495060;
    background-color: #f7f7f7;
}

.rss-item {
    display: flex;
    justify-content: space-between;
    align-items: center;
    padding: 5px 0;
}
        `);
        rpDocument.head.appendChild(rpBoxStyle);

        const rpBadgeDiv = rpDocument.createElement("div");
        rpBadgeDiv.id = "rp-badge";
        // 方法 1
        //const span = rpDocument.createElement("span");
        //span.classList.add("rp-badge-count", "rp-count");
        //rpBadgeDiv.appendChild(span);
        // 方法 2
        rpBadgeDiv.innerHTML = TTP.createHTML(`<span class="rp-badge-count rp-count"></span>`);
        rpDocument.body.appendChild(rpBadgeDiv);

        rpDialog = rpDocument.createElement("div");
        rpDialog.id = "rp-dialog";
        rpDialog.innerHTML = TTP.createHTML( `
<div id="rp-dialog-container">
<div id="rp-dialog-title">
<div>${t.feed}<span class="rp-dialog-count rp-count"></span></div>
<button type="button" id="rp-close-btn" class="rp-btn rp-btn-small" title="${t.close}"><span>${ICON_CLOSE}</span></button>
</div>
<div id="rp-dialog-feeds-container">
<ul id="rp-dialog-feeds"></ul>
</div>
</div>
`);
        rpDocument.body.appendChild(rpDialog);

        rpBadge = rpDocument.getElementById("rp-badge");
        rpDialog = rpDocument.getElementById("rp-dialog");
        rpDialogTitle = rpDocument.getElementById("rp-dialog-title");
        rpDialogFeeds = rpDocument.getElementById("rp-dialog-feeds");

        rpIframe.style.width = rpBadge.getBoundingClientRect().width + "px";
        rpIframe.style.height = rpBadge.getBoundingClientRect().height + "px";

        addEventHandler(rpDocument.getElementById("rp-close-btn"), "click", function () {
            rpDialog.style.display = "none";
            rpBadge.style.display = "block";
            rpIframe.style.width = rpBadge.getBoundingClientRect().width + "px";
            rpIframe.style.height = rpBadge.getBoundingClientRect().height + "px";
        });

        rpDocument.getElementById("rp-badge").addEventListener("click", () => {
            if (feedsSet.size === 0) {
                return
            }
            rpDialog.style.display = "block";
            rpBadge.style.display = "none";

            resizeIframe();
        });

        window.addEventListener('resize', resizeIframe);
        window.addEventListener('load', resizeIframe);
    }

    function resizeIframe() {
        if (rpDialog.style.display === "block"){
            if (window.innerWidth < 400){
                rpIframe.style.width = `${window.innerWidth}px`;
            } else {
                rpIframe.style.width = "400px";
            }
            const dialogHeight = rpDialogFeeds.getBoundingClientRect().height + rpDialogTitle.getBoundingClientRect().height;
            const availableHeight = rpIframe.getBoundingClientRect().bottom;
            if (dialogHeight < availableHeight) {
                rpIframe.style.height = `${dialogHeight}px`;
            } else {
                rpIframe.style.height = `${availableHeight}px`;
            }
        }
    }

    // 更新徽章
    function updateBadge() {
        Array.from(rpDocument.getElementsByClassName("rp-count")).forEach(el => {el.textContent = feedsSet.size});
    }

    // 更新对话框内容
    function updateDialog(title, url) {
        const li = rpDocument.createElement('li');
        li.className = 'rss-item';
        li.innerHTML = TTP.createHTML(`
                    <div class="rss-info">
                        <h5 class="rss-title">${title}</h5>
                        <a class="rss-link" href="${url}" target="_blank">${url}</a>
                    </div>
                    <div class="rss-actions">
                        <button class="rss-btn rss-copy rp-btn-primary">${t.copy}</button>
                        <button class="rss-btn rss-follow rp-btn-primary">${t.follow}</button>
                    </div>
                `);
        li.querySelector('.rss-copy').addEventListener('click', () => copyToClipboard(url));
        li.querySelector('.rss-follow').addEventListener('click', () => followFeed(url));
        rpDialogFeeds.appendChild(li);
        new HoverImgQR(li.querySelector('.rss-link'));
    }

    // 复制到剪贴板
    function copyToClipboard(url) {
        GM_setClipboard(url);
        GM_notification({
            text: t.copied,
            title: t.copySucceeded,
            timeout: 2000
        });
    }

    // 关注Feed
    function followFeed(url) {
        // 这里可以根据用户设置的RSS阅读器来打开相应的订阅链接
        const rssService = GM_getValue("rss_service", "feedly");
        if (rssService === "feedly") {
            window.open(`https://feedly.com/i/subscription/feed/${encodeURIComponent(url)}`, '_blank');
        } else if (rssService === "inoreader") {
            window.open(GM_getValue("inoreader_domain", "https://www.inoreader.com") + `/?add_feed=${encodeURIComponent(url)}`, "_blank");
        } else if (rssService === "tinytinyrss") {
            window.open(GM_getValue("tinytinyrss_domain") + `/public.php?op=bookmarklets--subscribe&feed_url=${url}`, "_blank");
        } else if (rssService === "freshrss") {
            window.open(GM_getValue("freshrss_domain") + `/i/?c=feed&a=add&url_rss=${encodeURIComponent(url)}`, "_blank");
        } else if (rssService === "miniflux") {
            window.open(GM_getValue("miniflux_domain") + `/bookmarklet?uri=${encodeURIComponent(url)}`, "_blank");
        } else if (rssService === "folo") {
            window.open(`follow://add?url=${encodeURIComponent(url)}`, "_blank");
        } else if (rssService === "newsblur") {
            window.open(`http://www.newsblur.com/?url=${encodeURIComponent(url)}`, "_blank");
        } else if (rssService === "theoldreader") {
            window.open(`https://theoldreader.com/feeds/subscribe?url=${encodeURIComponent(url)}`, "_blank");
        }
    }

    function reset() {
        if (rpDiv){
            rpDiv.remove();
        }
        if (rpStyle){
            rpStyle.remove();
        }
        feedsSet.clear();
    }


    function addEventHandler(target, eventName, eventHandler, scope) {
        let f = scope ? function() {
            eventHandler.apply(scope, arguments);
        } : eventHandler;
        if (target.addEventListener) {
            target.addEventListener(eventName, f, true);
        } else if (target.attachEvent) {
            target.attachEvent("on" + eventName, f);
        }
        return f;
    }

    function optimizeLink(link) {
        if (link.match(/douban\.com\/people/i)) {
            const src = document.querySelector("#profile > div > div.bd > div.basic-info > div.uhead-wrap > img.userface").src;
            const m = src.match(/ul(\d+)-/i);
            link = "https://www.douban.com/people/" + m[1];
        }
        return link;
    }
    function isEmpty(str) {
        return str.trim().length === 0;
    }
    function isValidUrl(url) {
        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));
    }
    function fetch(url, options = {}) {
        return new Promise((resolve, reject) => {
            GM_xmlhttpRequest({
                url: url,
                method: options.method || 'GET',
                headers: options.headers,
                data: options.body,
                responseType: options.responseType || 'json',
                onload: function(response) {
                    resolve({
                        ok: response.status >= 200 && response.status < 300,
                        status: response.status,
                        statusText: response.statusText,
                        json: () => {
                            if (options.responseType === 'json') {
                                return Promise.resolve(response.response); // 直接返回解析后的 JSON
                            } else {
                                try {
                                    return Promise.resolve(JSON.parse(response.responseText)); // 手动解析 JSON
                                } catch (e) {
                                    return Promise.reject(new Error('Failed to parse JSON'));
                                }
                            }
                        },
                        text: () => Promise.resolve(response.responseText),
                        blob: () => Promise.resolve(new Blob([response.response])),
                        headers: response.responseHeaders,
                    });
                },
                onerror: function(error) {
                    reject(error);
                },
                ontimeout: function() {
                    reject(new Error('Request timed out'));
                },
                timeout: options.timeout || 0,
            });
        });
    }

    class HoverImgQR {
        constructor(el) {
            this.DOM = {el: el};
            this.DOM.reveal = document.createElement('div');
            this.DOM.reveal.className = 'hover-reveal';

            try {
                const qr = qrcode(0, 'L');
                qr.addData(this.DOM.el.href);
                qr.make();
                const url = qr.createDataURL();

                const thisDOM = this.DOM;
                thisDOM.reveal.innerHTML = TTP.createHTML(`<div class="hover-reveal__inner"><div class="hover-reveal__img" style="background-image:url(${url})"></div></div>`);
                thisDOM.revealInner = thisDOM.reveal.querySelector('.hover-reveal__inner');
                thisDOM.revealInner.style.overflow = 'hidden';
                thisDOM.revealImg = thisDOM.revealInner.querySelector('.hover-reveal__img');
                thisDOM.el.appendChild(thisDOM.reveal);

                this.initEvents();
            } catch (e) {
                console.error("报错:", e);
            }
        }
        getMousePos (e) {
            let posX = 0;
            let posY = 0;
            if (!e) e = window.event;
            if (e.pageX || e.pageY) {
                posX = e.pageX;
                posY = e.pageY;
            } else if (e.clientX || e.clientY) {
                posX = e.clientX + document.body.scrollLeft + document.documentElement.scrollLeft;
                posY = e.clientY + document.body.scrollTop + document.documentElement.scrollTop;
            }
            return {
                x: posX,
                y: posY
            }
        }
        initEvents() {
            this.positionElement = (ev) => {
                const mousePos = this.getMousePos(ev);
                const docScrolls = {
                    left: rpDocument.body.scrollLeft + rpDocument.documentElement.scrollLeft,
                    top: rpDocument.body.scrollTop + rpDocument.documentElement.scrollTop
                };
                this.DOM.reveal.style.top = `${mousePos.y-70-docScrolls.top}px`;
                this.DOM.reveal.style.left = `${mousePos.x+10-docScrolls.left}px`;
            };
            this.mouseenterFn = (ev) => {
                if (!GM_getValue("enable_qr_code", true)) return;
                this.positionElement(ev);
                this.DOM.revealInner.style.overflow = 'visible';
                this.DOM.reveal.style.opacity = '1';
            };
            this.mousemoveFn = ev => requestAnimationFrame(() => {
                if (!GM_getValue("enable_qr_code", true)) return;
                this.positionElement(ev);
            });
            this.mouseleaveFn = () => {
                if (!GM_getValue("enable_qr_code", true)) return;
                this.DOM.revealInner.style.overflow = 'hidden';
                this.DOM.reveal.style.opacity = '0';
            };
            this.DOM.el.addEventListener('mouseenter', this.mouseenterFn);
            this.DOM.el.addEventListener('mousemove', this.mousemoveFn);
            this.DOM.el.addEventListener('mouseleave', this.mouseleaveFn);
        }
    }

    // 监听全屏变化
    document.addEventListener('fullscreenchange', handleFullscreenChange);
    document.addEventListener('webkitfullscreenchange', handleFullscreenChange);
    document.addEventListener('mozfullscreenchange', handleFullscreenChange);
    function handleFullscreenChange() {
        const isFullscreen = document.fullscreenElement ||
                        document.webkitFullscreenElement ||
                        document.mozFullScreenElement;
        rpDiv.style.display = isFullscreen ? 'none' : 'block';
    }


    // 检查环境
    if (window.top !== window.self || !window.location.protocol.startsWith('http')) {
        return;
    }
    // 当页面加载完成后初始化
    observeUrlChange();
    findFeeds();
})();