GitHub Gist Copier & Downloader

Adds copy button to Gist files for easy code copying.| Adds download button to Gist files for easy code downloading.

Version au 11/03/2025. Voir la dernière version.

Vous devrez installer une extension telle que Tampermonkey, Greasemonkey ou Violentmonkey pour installer ce script.

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

Vous devrez installer une extension telle que Tampermonkey ou Violentmonkey pour installer ce script.

Vous devrez installer une extension telle que Tampermonkey ou Userscripts pour installer ce script.

Vous devrez installer une extension telle que Tampermonkey pour installer ce script.

Vous devrez installer une extension de gestionnaire de script utilisateur pour installer ce script.

(J'ai déjà un gestionnaire de scripts utilisateur, laissez-moi l'installer !)

Vous devrez installer une extension telle que Stylus pour installer ce style.

Vous devrez installer une extension telle que Stylus pour installer ce style.

Vous devrez installer une extension telle que Stylus pour installer ce style.

Vous devrez installer une extension du gestionnaire de style pour utilisateur pour installer ce style.

Vous devrez installer une extension du gestionnaire de style pour utilisateur pour installer ce style.

Vous devrez installer une extension du gestionnaire de style pour utilisateur pour installer ce style.

(J'ai déjà un gestionnaire de style utilisateur, laissez-moi l'installer!)

// ==UserScript==
// @name GitHub Gist Copier & Downloader
// @name:zh-CN GitHub Gist 代码片段复制与下载器
// @description  Adds copy button to Gist files for easy code copying.| Adds download button to Gist files for easy code downloading.
// @description:zh-CN 向 Gist 文件添加复制按钮,以便轻松复制代码。| 向 Gist 文件添加下载按钮,以便轻松下载代码。
// @author             afkarxyz,人民的勤务员 <[email protected]>
// @namespace    https://github.com/ChinaGodMan/UserScripts
// @supportURL    https://github.com/ChinaGodMan/UserScripts/issues
// @homepageURL   https://github.com/ChinaGodMan/UserScripts
// @license      MIT
// @icon              
// @run-at       document-end
// @match        https://gist.github.com/*
// @grant        GM_xmlhttpRequest
// @grant        GM_addStyle
// @connect      githubusercontent.com
// @compatible     chrome
// @compatible     firefox
// @compatible     edge
// @compatible     opera
// @compatible     safari
// @compatible     kiwi
// @compatible     qq
// @compatible     via
// @compatible      brave
// @version         2025.03.12.0022
// @created         2025-03-12 00:22:40
// @modified        2025-03-12 00:22:40
// ==/UserScript==
/**
 * File: github-gist-copier.user.js
 * Project: UserScripts
 * File Created: 2025/03/12,Wednesday 00:22:50
 * Author: 人民的勤务员@ChinaGodMan ([email protected])
 * -----
 * Last Modified: 2025/03/12,Wednesday 02:32:41
 * Modified By: 人民的勤务员@ChinaGodMan ([email protected])
 * -----
 * License: MIT License
 * Copyright © 2024 - 2025 ChinaGodMan,Inc
 */
(function () {
    'use strict'
    function noop() { }
    function debounce(f, delay) {
        let timeoutId = null
        return function (...args) {
            if (timeoutId) {
                clearTimeout(timeoutId)
            }
            timeoutId = setTimeout(() => {
                f.apply(this, args)
            }, delay)
        }
    }

    function createCopyButton(fileElement) {
        const fileActionElement = fileElement.querySelector('.file-actions')
        if (!fileActionElement) {
            return noop
        }

        const rawButton = fileActionElement.querySelector('a[href*="/raw/"]')
        if (!rawButton) {
            return noop
        }

        const buttonToDown = document.createElement('button')
        buttonToDown.className = 'btn-octicon gist-down-button'
        buttonToDown.style.marginRight = '5px'
        const button = document.createElement('button')
        button.className = 'btn-octicon gist-copy-button'
        button.style.marginRight = '5px'
        buttonToDown.innerHTML = `
        <svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-down">
            <path d="M2.75 14A1.75 1.75 0 0 1 1 12.25v-2.5a.75.75 0 0 1 1.5 0v2.5c0 .138.112.25.25.25h10.5a.25.25 0 0 0 .25-.25v-2.5a.75.75 0 0 1 1.5 0v2.5A1.75 1.75 0 0 1 13.25 14Z"></path>
            <path d="M7.25 7.689V2a.75.75 0 0 1 1.5 0v5.689l1.97-1.969a.749.749 0 1 1 1.06 1.06l-3.25 3.25a.749.749 0 0 1-1.06 0L4.22 6.78a.749.749 0 1 1 1.06-1.06l1.97 1.969Z"></path>
        </svg>

        <svg style="display: none;" aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-check color-fg-success">
            <path d="M13.78 4.22a.75.75 0 0 1 0 1.06l-7.25 7.25a.75.75 0 0 1-1.06 0L2.22 9.28a.751.751 0 0 1 .018-1.042.751.751 0 0 1 1.042-.018L6 10.94l6.72-6.72a.75.75 0 0 1 1.06 0Z"></path>
        </svg>
        `

        button.innerHTML = `
        <svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-down">
            <path d="M0 6.75C0 5.784.784 5 1.75 5h1.5a.75.75 0 0 1 0 1.5h-1.5a.25.25 0 0 0-.25.25v7.5c0 .138.112.25.25.25h7.5a.25.25 0 0 0 .25-.25v-1.5a.75.75 0 0 1 1.5 0v1.5A1.75 1.75 0 0 1 9.25 16h-7.5A1.75 1.75 0 0 1 0 14.25Z"></path><path d="M5 1.75C5 .784 5.784 0 6.75 0h7.5C15.216 0 16 .784 16 1.75v7.5A1.75 1.75 0 0 1 14.25 11h-7.5A1.75 1.75 0 0 1 5 9.25Zm1.75-.25a.25.25 0 0 0-.25.25v7.5c0 .138.112.25.25.25h7.5a.25.25 0 0 0 .25-.25v-7.5a.25.25 0 0 0-.25-.25Z"></path>
        </svg>

        <svg style="display: none;" aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-check color-fg-success">
            <path d="M13.78 4.22a.75.75 0 0 1 0 1.06l-7.25 7.25a.75.75 0 0 1-1.06 0L2.22 9.28a.751.751 0 0 1 .018-1.042.751.751 0 0 1 1.042-.018L6 10.94l6.72-6.72a.75.75 0 0 1 1.06 0Z"></path>
        </svg>
        `

        const copyIcon = button.querySelector('.octicon-copy')
        const checkIcon = button.querySelector('.octicon-check')
        const downIcon = buttonToDown.querySelector('.octicon-down')
        const downcheckIcon = buttonToDown.querySelector('.octicon-check')
        let timeoutId = null
        const copyHandler = (e) => {
            if (timeoutId) {
                return
            }
            e.preventDefault()
            const rawUrl = rawButton.href
            GM_xmlhttpRequest({
                method: 'GET',
                url: rawUrl,
                onload: function (response) {
                    if (response.status === 200) {
                        navigator.clipboard.writeText(response.responseText).then(() => {
                            copyIcon.style.display = 'none'
                            checkIcon.style.display = 'inline-block'
                            timeoutId = setTimeout(() => {
                                copyIcon.style.display = 'inline-block'
                                checkIcon.style.display = 'none'
                                timeoutId = null
                            }, 500)
                        }).catch(err => {
                            console.error('Failed to copy text: ', err)
                        })
                    }
                },
                onerror: function (error) {
                    timeoutId = null
                }
            })
        }
        const downHandler = (e, element) => {
            if (timeoutId) {
                return
            }
            e.preventDefault()
            const gistName = element.parentElement.querySelector('.gist-blob-name').innerText
            const rawUrl = rawButton.href
            GM_xmlhttpRequest({
                method: 'GET',
                url: rawUrl,
                onload: function (response) {
                    if (response.status === 200) {
                        const blob = new Blob([response.responseText], {
                            type: 'text/plain'
                        })
                        const url = URL.createObjectURL(blob)
                        const a = document.createElement('a')
                        a.href = url
                        a.download = gistName
                        a.click()
                        downIcon.style.display = 'none'
                        downcheckIcon.style.display = 'inline-block'
                        timeoutId = setTimeout(() => {
                            downIcon.style.display = 'inline-block'
                            downcheckIcon.style.display = 'none'
                            timeoutId = null
                        }, 500)
                    } else {
                        console.error('Download Failed')
                    }
                },
                onerror: function (error) {
                    timeoutId = null
                }
            })
        }

        buttonToDown.addEventListener('click', (e) => downHandler(e, fileActionElement))
        button.addEventListener('click', copyHandler)
        fileActionElement.insertBefore(button, fileActionElement.firstChild)
        fileActionElement.insertBefore(buttonToDown, fileActionElement.firstChild)
        return () => {
            button.removeEventListener('click', copyHandler)
            if (timeoutId) {
                clearTimeout(timeoutId)
            }
            button.remove()
        }
    }

    function runGist() {
        let removeAllListeners = noop

        function tryCreateCopyButtons() {
            removeAllListeners()
            const fileElements = [...document.querySelectorAll('.file')]
            const removeListeners = fileElements.map(createCopyButton)
            removeAllListeners = () => {
                removeListeners.map((f) => f());
                [...document.querySelectorAll('.gist-down-button')].forEach((el) => {
                    el.remove()
                });
                [...document.querySelectorAll('.gist-copy-button')].forEach((el) => {
                    el.remove()
                })
            }
        }

        setTimeout(tryCreateCopyButtons, 300)

        const observer = new MutationObserver(debounce(() => {
            if (document.querySelectorAll('.file').length > 0 &&
                document.querySelectorAll('.gist-copy-button').length === 0) {
                tryCreateCopyButtons()
            }
        }, 100))

        observer.observe(document.body, {
            childList: true,
            subtree: true
        })

        if (window.onurlchange === null) {
            window.addEventListener('urlchange', debounce(tryCreateCopyButtons, 16))
        }
    }

    runGist()


})()