复制RSS地址

自动检测并快捷展示当前网站的 RSS 订阅信息,方便复制(Copy RSS Feed)

// ==UserScript==
// @name         Copy RSS Feed
// @name:zh      复制RSS地址
// @name:zh-CN   复制RSS地址
// @description         Automatically detect and quickly display the RSS subscription information of the current website for easy copying.
// @description:zh      自动检测并快捷展示当前网站的 RSS 订阅信息,方便复制(Copy RSS Feed)
// @description:zh-CN   自动检测并快捷展示当前网站的 RSS 订阅信息,方便复制(Copy RSS Feed)
// @author       Wilsonyiyi
// @namespace    https://github.com/wilsonyiyi
// @icon         
// @match        *://*/*
// @grant        GM_setClipboard
// @grant        GM_addStyle
// @license      GPL3.0
// @version      1.0.3
// ==/UserScript==

(function() {
    'use strict';

    GM_addStyle(`
        #__wilsonyiyi__rss-float-container {
            position: fixed;
            right: -30px;
            top: 30%;
            height: 40px;
            z-index: 9999;
            display: flex;
            align-items: center;
            transition: all 0.3s ease;
            overflow: visible;
        }
        #__wilsonyiyi__rss-float-container:hover {
            right: 0;
        }

        #__wilsonyiyi__rss-float-btn {
            width: 40px;
            height: 40px;
            background: #ff9800;
            border-top-left-radius: 20px;
            border-bottom-left-radius: 20px;
            box-shadow: 0 2px 10px rgba(0, 0, 0, 0.2);
            cursor: pointer;
            display: flex;
            justify-content: center;
            align-items: center;
            user-select: none;
            position: relative;
        }

        #__wilsonyiyi__rss-float-btn:hover {
            background: #ff7d00;
        }

        #__wilsonyiyi__rss-float-btn svg {
            width: 24px;
            height: 24px;
            fill: white;
        }

        #__wilsonyiyi__rss-label {
            height: 40px;
            line-height: 40px;
            background: #ff9800;
            color: white;
            font-family: Arial, sans-serif;
            font-size: 14px;
            padding: 0 15px;
            white-space: nowrap;
            max-width: 0;
            overflow: hidden;
            transition: max-width 0.3s ease;
            cursor: pointer;
        }

        #__wilsonyiyi__rss-float-container:hover #__wilsonyiyi__rss-label {
            max-width: 200px;
        }

        #__wilsonyiyi__rss-close-btn {
            width: 40px;
            height: 40px;
            background: #ff9800;
            display: flex;
            justify-content: center;
            align-items: center;
            cursor: pointer;
            max-width: 0;
            overflow: hidden;
            transition: max-width 0.3s ease;
        }

        #__wilsonyiyi__rss-float-container:hover #__wilsonyiyi__rss-close-btn {
            max-width: 40px;
        }

        #__wilsonyiyi__rss-close-btn svg {
            width: 16px;
            height: 16px;
            fill: white;
            min-width: 16px;
        }

        #__wilsonyiyi__rss-toast {
            position: fixed;
            top: 20px;
            left: 50%;
            transform: translateX(-50%);
            background: rgba(0, 0, 0, 0.7);
            color: white;
            padding: 10px 20px;
            border-radius: 5px;
            z-index: 10000;
            font-family: Arial, sans-serif;
            font-size: 14px;
            opacity: 0;
            transition: opacity 0.3s ease;
        }

        #__wilsonyiyi__rss-toast.show {
            opacity: 1;
        }
    `);

    const findRssLinksByLinkTag = () => {
        const links = document.querySelectorAll('link[type="application/rss+xml"], link[type="application/atom+xml"]');
        return Array.from(links).map(link => link.href);
    };

    const findRssLinksByATag = () => {
      const links = Array.from(document.querySelectorAll('a')).filter(x => x.href.endsWith('.xml'))
      return Array.from(links).map(link => link.href);
    }

    const getRssLinks = () => {
      const r1 = findRssLinksByLinkTag()
      if (r1.length) return r1

      const r2 = findRssLinksByATag()
      if (r2.length) return r2

      return []
    }

    const createFloatButton = (rssLink) => {
        const container = document.createElement('div');
        container.id = '__wilsonyiyi__rss-float-container';

        const button = document.createElement('div');
        button.id = '__wilsonyiyi__rss-float-btn';
        button.innerHTML = '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M6.18,15.64A2.18,2.18 0 0,1 8.36,17.82C8.36,19 7.38,20 6.18,20C5,20 4,19 4,17.82A2.18,2.18 0 0,1 6.18,15.64M4,4.44A15.56,15.56 0 0,1 19.56,20H16.73A12.73,12.73 0 0,0 4,7.27V4.44M4,10.1A9.9,9.9 0 0,1 13.9,20H11.07A7.07,7.07 0 0,0 4,12.93V10.1Z"/></svg>';

        const label = document.createElement('div');
        label.id = '__wilsonyiyi__rss-label';
        label.textContent = 'Copy RSS Feed';

        const closeBtn = document.createElement('div');
        closeBtn.id = '__wilsonyiyi__rss-close-btn';
        closeBtn.innerHTML = '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M19,6.41L17.59,5L12,10.59L6.41,5L5,6.41L10.59,12L5,17.59L6.41,19L12,13.41L17.59,19L19,17.59L13.41,12L19,6.41Z"/></svg>';

        container.appendChild(button);
        container.appendChild(label);
        container.appendChild(closeBtn);

        document.body.appendChild(container);

        const copyRssLink = () => {
            GM_setClipboard(rssLink);
            showToast('🎉 Copied 🎉');
        };

        button.addEventListener('click', copyRssLink);
        label.addEventListener('click', copyRssLink);

        closeBtn.addEventListener('click', (e) => {
            e.stopPropagation();
            container.style.display = 'none';
        });

        return container;
    };

    const createToast = () => {
        const toast = document.createElement('div');
        toast.id = '__wilsonyiyi__rss-toast';
        document.body.appendChild(toast);
        return toast;
    };

    const showToast = (message) => {
        const toast = document.getElementById('__wilsonyiyi__rss-toast') || createToast();
        toast.textContent = message;
        toast.classList.add('show');

        setTimeout(() => {
            toast.classList.remove('show');
        }, 2000);
    };

    const init = () => {
        const rssLinks = getRssLinks();

        if (rssLinks.length > 0) {
            createFloatButton(rssLinks[0]);
            createToast();
        }
    };

    window.addEventListener('load', init);
})();

QingJ © 2025

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