GitHub Gist Copier & Downloader

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

Per 11-03-2025. Zie de nieuwste versie.

Voor het installeren van scripts heb je een extensie nodig, zoals Tampermonkey, Greasemonkey of Violentmonkey.

Voor het installeren van scripts heb je een extensie nodig, zoals {tampermonkey_link:Tampermonkey}.

Voor het installeren van scripts heb je een extensie nodig, zoals Tampermonkey of Violentmonkey.

Voor het installeren van scripts heb je een extensie nodig, zoals Tampermonkey of Userscripts.

Voor het installeren van scripts heb je een extensie nodig, zoals {tampermonkey_link:Tampermonkey}.

Voor het installeren van scripts heb je een gebruikersscriptbeheerder nodig.

(Ik heb al een user script manager, laat me het downloaden!)

Voor het installeren van gebruikersstijlen heb je een extensie nodig, zoals {stylus_link:Stylus}.

Voor het installeren van gebruikersstijlen heb je een extensie nodig, zoals {stylus_link:Stylus}.

Voor het installeren van gebruikersstijlen heb je een extensie nodig, zoals {stylus_link:Stylus}.

Voor het installeren van gebruikersstijlen heb je een gebruikersstijlbeheerder nodig.

Voor het installeren van gebruikersstijlen heb je een gebruikersstijlbeheerder nodig.

Voor het installeren van gebruikersstijlen heb je een gebruikersstijlbeheerder nodig.

(Ik heb al een beheerder - laat me doorgaan met de installatie!)

// ==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()


})()