// ==UserScript==
// @name GitHub Gist Copier & Downloader
// @name:ar Github Gist Code Snippet Copy and Downloader
// @name:bg Github Gist код Shippet Copy and Downloader
// @name:cs Github Gist Code Snippet Copy and Downloader
// @name:da GitHub Gist Code Strappet Copy and Downloader
// @name:de Github Gist Code Snippet Copy und Downloader
// @name:el GitHub Gist Κωδικός απόσπασμα και λήψη
// @name:en GitHub Gist code snippet copy and downloader
// @name:eo Github Gist Code Snippet Copy and Downloader
// @name:es Copia y descargador de fragmentos de código Github Gist
// @name:fi Github GIST -koodin katkelma kopio ja lataaja
// @name:fr GitHub Gist Code Snippet Copy and Downloader
// @name:fr-CA GitHub Gist Code Snippet Copy and Downloader
// @name:he Github GiSt Code Snippet Copy and Downloader
// @name:hr GitHub Gist Code Snippet Kopiranje i preuzimanje
// @name:hu GitHub GIST kódrészlet másolat és letöltő
// @name:id Salinan dan download cuplikan kode GIST GITHUB
// @name:it Copia di snippet di codice GIST GIST e downloader
// @name:ja GitHub Gistコードスニペットコピーとダウンローダー
// @name:ka Github gist კოდის snippet ასლი და ჩამოტვირთვა
// @name:ko github gist 코드 스 니펫 복사 및 다운로더
// @name:nb GitHub Gist Code Snippet Copy and Downloader
// @name:nl GitHub Gist Code Snippet Copy and Downloader
// @name:pl Github gist kod kodu fragmentu i downloader
// @name:pt-BR Cópia do snippet e downloader de código de código do Github
// @name:ro Github Gist Code Snipt Copy and Downloader
// @name:ru Github Gist Code Specy Copy and Downloader
// @name:sk Kópia a sťahovanie kódu github gist
// @name:sr ГитХуб Гист Цоде Сниппет Копирање и преузимање
// @name:sv Github gist code kodavsnitt kopia och nedladdare
// @name:th GitHub Gist Code Snippet Copy and Downloader
// @name:tr Github Gist Kodu Snippet Copy ve Downloader
// @name:ug GitHub GIST CLIPTE CLIPPET ۋە چۈشۈرگۈچى
// @name:uk Копія та завантажувач Github Github Gist Code
// @name:vi Github Gist Code Snippet Sao chép và tải xuống
// @name:zh GitHub Gist 代码片段复制与下载器
// @name:zh-CN GitHub Gist 代码片段复制与下载器
// @name:zh-HK GitHub Gist 代碼片段複製與下載器
// @name:zh-SG GitHub Gist 代码片段复制与下载器
// @name:zh-TW GitHub Gist 代碼片段複製與下載器
// @description Adds copy button to Gist files for easy code copying.| Adds download button to Gist files for easy code downloading.
// @description:ar أضف زر نسخ إلى ملف GIST لجعل الكود يتم نسخه بسهولة. |
// @description:bg Добавете бутон за копиране към файла на GIST, за да направите лесно копирания код. |.
// @description:cs Přidejte tlačítko Kopírovat do souboru GIST, aby se kód snadno zkopíroval. |.
// @description:da Tilføj en kopi -knap til GIST -filen for at gøre koden let kopieret. |.
// @description:de Fügen Sie der GIST -Datei eine Kopierschaltfläche hinzu, um den Code leicht zu kopieren. |.
// @description:el Προσθέστε ένα κουμπί αντιγραφής στο αρχείο GIST για να αντιγράψετε εύκολα τον κώδικα. | Προσθέστε ένα κουμπί λήψης στο αρχείο GIST για εύκολη λήψη του κώδικα.
// @description:en Add a copy button to the Gist file to make the code easily copied. | Add a download button to the Gist file to easily download the code.
// @description:eo Aldonu kopian butonon al la GIST -dosiero por fari la kodon facile kopiita. | Aldoni elŝutan butonon al la GIST -dosiero por facile elŝuti la kodon.
// @description:es Agregue un botón de copia al archivo GIST para que el código se copie fácilmente. | Agregar un botón de descarga al archivo GIST para descargar fácilmente el código.
// @description:fi Lisää kopio -painike GIST -tiedostoon, jotta koodi kopioidaan helposti. | Lisää latauspainike GIST -tiedostoon koodin helposti.
// @description:fr Ajoutez un bouton de copie au fichier GIST pour que le code soit facilement copié. | Ajouter un bouton de téléchargement au fichier GIST pour télécharger facilement le code.
// @description:fr-CA Ajoutez un bouton de copie au fichier GIST pour que le code soit facilement copié. | Ajouter un bouton de téléchargement au fichier GIST pour télécharger facilement le code.
// @description:he הוסף כפתור העתק לקובץ ה- GIST כדי להפוך את הקוד להעתיק בקלות. |.
// @description:hr Dodajte gumb za kopiranje u datoteku Gist da biste se kôd lako kopirali. | Dodajte gumb za preuzimanje u datoteku Gist da biste lako preuzeli kôd.
// @description:hu Adjon hozzá egy másolat gombot a GIST fájlhoz, hogy a kód könnyen másoljon. | Adjon hozzá egy letöltési gombot a GIST fájlhoz, hogy könnyen letöltse a kódot.
// @description:id Tambahkan tombol Salin ke file GIST untuk membuat kode dengan mudah disalin. Tambahkan tombol unduh ke file GIST untuk dengan mudah mengunduh kode.
// @description:it Aggiungi un pulsante Copia al file GIST per rendere il codice facilmente copiato. | Aggiungi un pulsante di download al file GIST per scaricare facilmente il codice.
// @description:ja GISTファイルにコピーボタンを追加して、コードを簡単にコピーすることができます。 |ダウンロードボタンをGISTファイルに追加して、コードを簡単にダウンロードします。
// @description:ka დაამატეთ კოპირების ღილაკი GIST ფაილში, რომ კოდი ადვილად გადაწეროთ. |
// @description:ko 코드를 쉽게 복사하기 위해 GIST 파일에 복사 버튼을 추가하십시오. |. 코드를 쉽게 다운로드하려면 GIST 파일에 다운로드 버튼을 추가하십시오.
// @description:nb Legg til en kopiknapp i GIST -filen for å gjøre koden enkelt kopiert. |.
// @description:nl Voeg een kopie -knop toe aan het GIST -bestand om de code gemakkelijk te laten kopiëren. | Voeg een downloadknop toe aan het GIST -bestand om de code eenvoudig te downloaden.
// @description:pl Dodaj przycisk kopii do pliku GIST, aby kod był łatwy do kopiowania. |
// @description:pt-BR Adicione um botão de cópia ao arquivo GIST para tornar o código facilmente copiado. |.
// @description:ro Adăugați un buton de copiere în fișierul GIST pentru a face codul ușor copiat. |
// @description:ru Добавьте кнопку копирования в файл GIST, чтобы сделать код легко скопированный. |.
// @description:sk Pridajte tlačidlo kopírovania do súboru GIST, aby sa kód ľahko skopíroval. | Dodajte tlačidlo sťahovania do súboru GIST, aby ste si ľahko stiahnuť kód.
// @description:sr Додајте дугме за копирање у датотеку Гист да бисте кодекс лако копирали. | Додајте дугме за преузимање у датотеку ГИСТ да бисте лако преузели код.
// @description:sv Lägg till en kopieringsknapp i GIST -filen så att koden enkelt kopieras. | Lägg till en nedladdningsknapp i GIST -filen för att enkelt ladda ner koden.
// @description:th เพิ่มปุ่มคัดลอกไปยังไฟล์ GIST เพื่อให้รหัสคัดลอกได้อย่างง่ายดาย | เพิ่มปุ่มดาวน์โหลดไปยังไฟล์ GIST เพื่อดาวน์โหลดรหัสได้อย่างง่ายดาย
// @description:tr Kodu kolayca kopyalamak için GIST dosyasına bir kopya düğmesi ekleyin. Kodu kolayca indirmek için bir indir düğmesi ekleyin.
// @description:ug كود ھۆججىتىگە بىر نۇسخا قوشۇش كۇنۇپكىسىنى قوشۇڭ. | بىز GIST ھۆججىتىگە چۈشۈرۈش كۇنۇپكىسىنى قوشۇڭ, كودنى ئاسانلا چۈشۈرۈش.
// @description:uk Додайте кнопку копіювання до файлу GIST, щоб зробити код легко скопіювати. | Додайте кнопку завантаження до файлу GIST, щоб легко завантажити код.
// @description:vi Thêm một nút sao chép vào tệp GIST để dễ dàng sao chép mã. |
// @description:zh 向 Gist 文件添加复制按钮,以便轻松复制代码。| 向 Gist 文件添加下载按钮,以便轻松下载代码。
// @description:zh-CN 向 Gist 文件添加复制按钮,以便轻松复制代码。| 向 Gist 文件添加下载按钮,以便轻松下载代码。
// @description:zh-HK 向 Gist 文件添加複制按鈕,以便輕鬆複製代碼。 | 向 Gist 文件添加下載按鈕,以便輕鬆下載代碼。
// @description:zh-SG 向 Gist 文件添加复制按钮,以便轻松复制代码。| 向 Gist 文件添加下载按钮,以便轻松下载代码。
// @description:zh-TW 向 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()
})()