Copy RSS Feed

Automatically detect and quickly display the RSS subscription information of the current website for easy copying.

スクリプトをインストールするには、Tampermonkey, GreasemonkeyViolentmonkey のような拡張機能のインストールが必要です。

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

スクリプトをインストールするには、TampermonkeyViolentmonkey のような拡張機能のインストールが必要です。

スクリプトをインストールするには、TampermonkeyUserscripts のような拡張機能のインストールが必要です。

このスクリプトをインストールするには、Tampermonkeyなどの拡張機能をインストールする必要があります。

このスクリプトをインストールするには、ユーザースクリプト管理ツールの拡張機能をインストールする必要があります。

(ユーザースクリプト管理ツールは設定済みなのでインストール!)

このスタイルをインストールするには、Stylusなどの拡張機能をインストールする必要があります。

このスタイルをインストールするには、Stylus などの拡張機能をインストールする必要があります。

このスタイルをインストールするには、Stylus tなどの拡張機能をインストールする必要があります。

このスタイルをインストールするには、ユーザースタイル管理用の拡張機能をインストールする必要があります。

このスタイルをインストールするには、ユーザースタイル管理用の拡張機能をインストールする必要があります。

このスタイルをインストールするには、ユーザースタイル管理用の拡張機能をインストールする必要があります。

(ユーザースタイル管理ツールは設定済みなのでインストール!)

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください
// ==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);
})();