Greasy Fork is available in English.

Github Copy Raw File URL and Download File

Add buttons at the end of each file line to copy the raw file URL and download the file

// ==UserScript==
// @name              Github Copy Raw File URL and Download File
// @name:zh-CN        Github 复制原始文件 URL 与下载文件
// @description:zh-CN 在每个文件行的末尾添加按钮,以复制原始文件 URL 和下载文件
// @name:ar           Github نسخ الملف الأصلي URL مع تنزيل الملفات
// @description:ar    زر إضافة في نهاية كل سطر ملف,لنسخ الملف الأصلي URL وتحميل الملفات
// @name:bg           Github Копирайте оригиналния файл URL с файлове за изтегляне
// @description:bg    Бутон за добавяне в края на всеки файлов ред,за да копирате оригиналния файл URL и изтегляне на файлове
// @name:cs           Github Zkopírujte původní soubor URL se staženými soubory
// @description:cs    Tlačítko Přidat na konci každého řádku souboru,pro zkopírování původního souboru URL a stahovat soubory
// @name:da           Github Kopiér den originale fil URL med download filer
// @description:da    Tilføj knap i slutningen af hver fillinje,for at kopiere den originale fil URL og download filer
// @name:de           Github Originaldatei kopieren URL mit Download-Dateien
// @description:de    Schaltfläche „Hinzufügen“ am Ende jeder Dateizeile,um die Originaldatei zu kopieren URL und Dateien herunterladen
// @name:el           Github Αντιγραφή αρχικού αρχείου URL με λήψη αρχείων
// @description:el    Κουμπί προσθήκης στο τέλος κάθε γραμμής αρχείου,για να αντιγράψετε το αρχικό αρχείο URL και λήψη αρχείων
// @name:en           Github Copy original file URL with download files
// @description:en    Add button at the end of each file line,to copy the original file URL and download files
// @name:eo           Github Kopiu originalan dosieron URL kun elŝutaj dosieroj
// @description:eo    Aldoni butonon ĉe la fino de ĉiu dosierlinio,por kopii la originalan dosieron URL kaj elŝuti dosierojn
// @name:es           Github Copiar archivo original URL con archivos de descarga
// @description:es    Botón Agregar al final de cada línea de archivo,para copiar el archivo original URL y descargar archivos
// @name:fi           Github Kopioi alkuperäinen tiedosto URL lataustiedostojen kanssa
// @description:fi    Lisää-painike jokaisen tiedostorivin lopussa,kopioidaksesi alkuperäisen tiedoston URL ja ladata tiedostoja
// @name:fr           Github Copier le fichier original URL avec des fichiers à télécharger
// @description:fr    Ajouter un bouton à la fin de chaque ligne de fichier,pour copier le fichier original URL et télécharger des fichiers
// @name:he           Github העתק את הקובץ המקורי URL עם קבצים להורדה
// @description:he    כפתור הוסף בסוף כל שורת קובץ,כדי להעתיק את הקובץ המקורי URL ולהוריד קבצים
// @name:hr           Github Kopiraj izvornu datoteku URL s datotekama za preuzimanje
// @description:hr    Dodaj gumb na kraju svakog retka datoteke,za kopiranje izvorne datoteke URL i preuzimanje datoteka
// @name:hu           Github Eredeti fájl másolása URL letölthető fájlokkal
// @description:hu    Hozzáadás gomb minden fájlsor végén,az eredeti fájl másolásához URL és töltsön le fájlokat
// @name:id           Github Salin file asli URL dengan file unduhan
// @description:id    Tombol Tambah di akhir setiap baris file,untuk menyalin file asli URL dan mengunduh file
// @name:it           Github Copia il file originale URL con file scaricabili
// @description:it    Pulsante Aggiungi alla fine di ogni riga del file,per copiare il file originale URL e scaricare file
// @name:ja           Github 元のファイルをコピーする URL ダウンロードファイル付き
// @description:ja    各ファイル行の末尾にある「追加」ボタン,元のファイルをコピーするには URL そしてファイルをダウンロードする
// @name:ka           Github დააკოპირეთ ორიგინალი ფაილი URL ჩამოტვირთვის ფაილებით
// @description:ka    დაამატეთ ღილაკი თითოეული ფაილის ხაზის ბოლოს,ორიგინალური ფაილის კოპირება URL და ჩამოტვირთეთ ფაილები
// @name:ko           Github 원본 파일 복사 URL 다운로드 파일 포함
// @description:ko    각 파일 줄 끝에 추가 버튼,원본 파일을 복사하려면 URL 그리고 파일 다운로드
// @name:nl           Github Kopieer origineel bestand URL met downloadbestanden
// @description:nl    Knop Toevoegen aan het einde van elke bestandsregel,om het originele bestand te kopiëren URL en bestanden downloaden
// @name:nb           Github Kopier originalfilen URL med nedlastingsfiler
// @description:nb    Legg til-knapp på slutten av hver fillinje,for å kopiere originalfilen URL og last ned filer
// @name:pl           Github Skopiuj oryginalny plik URL z plikami do pobrania
// @description:pl    Dodaj przycisk na końcu każdej linii pliku,aby skopiować oryginalny plik URL i pobierz pliki
// @name:pt-BR        Github Copiar arquivo original URL com download de arquivos
// @description:pt-BR Botão Adicionar no final de cada linha do arquivo,para copiar o arquivo original URL e baixar arquivos
// @name:ro           Github Copiați fișierul original URL cu fișiere descărcate
// @description:ro    butonul Adaugă la sfârșitul fiecărei linii de fișier,pentru a copia fișierul original URL și descărcați fișiere
// @name:ru           Github Скопировать исходный файл URL с загрузкой файлов
// @description:ru    Добавить кнопку в конце каждой строки файла,скопировать исходный файл URL и скачать файлы
// @name:sk           Github Skopírujte pôvodný súbor URL so stiahnutými súbormi
// @description:sk    Tlačidlo Pridať na konci každého riadku súboru,na skopírovanie pôvodného súboru URL a sťahovať súbory
// @name:sr           Github Копирајте оригиналну датотеку URL са датотекама за преузимање
// @description:sr    Дугме Додај на крају сваке линије датотеке,да копирате оригиналну датотеку URL и преузимање датотека
// @name:sv           Github Kopiera originalfilen URL med nedladdningsfiler
// @description:sv    Lägg till-knappen i slutet av varje filrad,för att kopiera originalfilen URL och ladda ner filer
// @name:th           Github คัดลอกไฟล์ต้นฉบับ URL พร้อมดาวน์โหลดไฟล์
// @description:th    เพิ่มปุ่มที่ท้ายแต่ละบรรทัดไฟล์,เพื่อคัดลอกไฟล์ต้นฉบับ URL และดาวน์โหลดไฟล์
// @name:tr           Github Orijinal dosyayı kopyala URL indirme dosyalarıyla
// @description:tr    Her dosya satırının sonuna ekle düğmesi,orijinal dosyayı kopyalamak için URL ve dosyaları indir
// @name:ug           Github ئەسلى ھۆججەتنى كۆچۈرۈڭ URL چۈشۈرۈش ھۆججىتى بىلەن
// @description:ug    ھەر بىر ھۆججەت قۇرنىڭ ئاخىرىغا كۇنۇپكا قوشۇڭ,ئەسلى ھۆججەتنى كۆچۈرۈش URL ھۆججەتلەرنى چۈشۈرۈڭ
// @name:uk           Github Скопіюйте вихідний файл URL з файлами для завантаження
// @description:uk    Кнопка «Додати» в кінці кожного рядка файлу,щоб скопіювати вихідний файл URL і завантажити файли
// @name:vi           Github Sao chép tập tin gốc URL với các tập tin tải về
// @description:vi    Nút thêm vào cuối mỗi dòng tập tin,để sao chép tập tin gốc URL và tải về các tập tin
// @name:zh-TW        Github 複製原始文件 URL 與下載文件
// @description:zh-TW 在每個文件行的末尾新增按鈕,以複製原始文件 URL 和下載文件
// @name:zh-HK        Github 複製原始文件 URL 與下載文件
// @description:zh-HK 在每個文件行的末尾新增按鈕,以複製原始文件 URL 和下載文件
// @name:fr-CA        Github Copier le fichier original URL avec des fichiers à télécharger
// @description:fr-CA Ajouter un bouton à la fin de chaque ligne de fichier,pour copier le fichier original URL et télécharger des fichiers
// @description       Add buttons at the end of each file line to copy the raw file URL and download the file
// @namespace         https://github.com/ChinaGodMan/UserScripts
// @version           2.2.0.22
// @author            Kamikaze (https://github.com/Kamiikaze) ,人民的勤务员 <[email protected]>
// @match             https://github.com/*
// @icon              
// @iconbak           https://github.githubassets.com/pinned-octocat.svg
// @run-at            document-ready
// @license           MIT
// @supportURL        https://github.com/ChinaGodMan/UserScripts/issues
// @homepageURL       https://github.com/ChinaGodMan/UserScripts
// ==/UserScript==


// Need an Interval to detect path changes on github tree one-pager
// Define the number of seconds
const scanInterval = 2


const waitForFilelist = setInterval(() => {
    let fileListContainer = document.querySelector('div.Box > div.js-details-container.Details div') || document.querySelector('table')
    let fileList = []
    let isTable = false

    if (fileListContainer.tBodies) {
        fileList = fileListContainer.tBodies[0].children
        isTable = true
    } else {
        fileList = fileListContainer.children
    }

    if (fileList < 1) return

    appendButtons(fileList, isTable)

}, scanInterval * 1000)

function appendButtons(fileList, isTable = false) {
    let fileUrl = ''
    let rawFileUrl = ''
    for (let i = 0; i < fileList.length; i++) {
        let file = fileList[i]

        if (file.classList.contains('cp-btn-rdy')) continue

        file.classList.add('cp-btn-rdy')

        if (!isTable) {
            if (
                file.classList.contains('sr-only') ||
                file.childElementCount !== 4
            ) continue

            fileUrl = file.querySelector('div:nth-child(2) .js-navigation-open')
                .href
        } else {
            if (i === 0) continue

            if (
                file.classList.contains('sr-only')
            ) continue


            fileUrl = file.querySelector('a')
                .href
            file = file.querySelector('td:nth-child(4) > div')
        }
        //alert(fileUrl)
        // Dont add button if its a folder
        if (!fileUrl.includes('/blob/')) continue

        rawFileUrl = fileUrl.replace('/blob/', '/raw/')
        file.style = 'display: flex; justify-content: flex-end;'
        file.append(creatyCopyButton(rawFileUrl))
        file.append(creatyDownButton(rawFileUrl))
    }
}

function creatyCopyButton(copyText) {
    const copy2clipboard = `
		<clipboard-copy aria-label="Copy" value="test value" data-view-component="true" class="" tabindex="0" role="button" title="Copy raw file url">
			<svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-copy">
			<path fill-rule="evenodd" d="M0 6.75C0 5.784.784 5 1.75 5h1.5a.75.75 0 010 1.5h-1.5a.25.25 0 00-.25.25v7.5c0 .138.112.25.25.25h7.5a.25.25 0 00.25-.25v-1.5a.75.75 0 011.5 0v1.5A1.75 1.75 0 019.25 16h-7.5A1.75 1.75 0 010 14.25v-7.5z"></path><path fill-rule="evenodd" d="M5 1.75C5 .784 5.784 0 6.75 0h7.5C15.216 0 16 .784 16 1.75v7.5A1.75 1.75 0 0114.25 11h-7.5A1.75 1.75 0 015 9.25v-7.5zm1.75-.25a.25.25 0 00-.25.25v7.5c0 .138.112.25.25.25h7.5a.25.25 0 00.25-.25v-7.5a.25.25 0 00-.25-.25h-7.5z"></path>
			</svg>
		</clipboard-copy>`

    const copyButton = document.createElement('div')
    copyButton.setAttribute('role', 'gridcell')
    copyButton.style = 'margin-left: 10px; display: inline;'
    copyButton.innerHTML = copy2clipboard
    copyButton.children[0].value = copyText
    copyButton.children[0].style = 'cursor: pointer;'

    return copyButton
}
function creatyDownButton(copyText) {
    const copy2clipboard = `
<clipboard-copy aria-label="Download" value="test value" data-view-component="true" class="" tabindex="0" role="button" title="Download raw file url">
	<svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-download">
		<path fill-rule="evenodd" d="M1.75 14.25A1.75 1.75 0 013.5 12.5h9a1.75 1.75 0 011.75 1.75v1.5a.75.75 0 01-.75.75H2.5a.75.75 0 01-.75-.75v-1.5zM10.75 9.25a.25.25 0 01.25.25v2.5a.25.25 0 01-.25.25H5.25a.25.25 0 01-.25-.25v-2.5a.25.25 0 01.25-.25h5.5zM8 1.75a.25.25 0 01.25.25v7.5a.25.25 0 01-.25.25H6.75a.25.25 0 01-.25-.25v-7.5a.25.25 0 01.25-.25h1.5zM10.25 5.25l1.5 1.5a.25.25 0 01.35 0l3-3a.25.25 0 00-.35-.35L11 5.25 9.25 3.5a.25.25 0 00-.35.35z"></path>
	</svg>
</clipboard-copy>
`

    const copyButton = document.createElement('div')
    copyButton.setAttribute('role', 'gridcell')
    copyButton.style = 'margin-left: 10px; display: inline;'
    copyButton.innerHTML = copy2clipboard
    copyButton.children[0].value = copyText
    copyButton.children[0].style = 'cursor: pointer;'
    copyButton.addEventListener('click', () => {
        //    window.location.href = copyText;
        downloadFile(copyText, getFilenameFromUrl(copyText))
    })
    return copyButton
}
function downloadFile(url, filename) {
    var xhr = new XMLHttpRequest()
    xhr.open('GET', url, true)
    xhr.responseType = 'blob'
    xhr.onload = function () {
        if (xhr.status === 200) {
            var blob = xhr.response
            var objectUrl = window.URL.createObjectURL(blob)
            var a = document.createElement('a')
            a.href = objectUrl
            a.download = filename // 设置下载文件名
            document.body.appendChild(a)
            a.click()
            window.URL.revokeObjectURL(objectUrl) // 清理 object URL
            document.body.removeChild(a) // 清理 DOM
        }
    }
    xhr.send()
}
function getFilenameFromUrl(url) {
    if (typeof url !== 'string' || url.trim() === '') {
        logMessage('getFilenameFromUrl', 'URL无效,默认文件名download', false)
        return 'download' // 返回一个默认的文件名
    }
    var lastSlashIndex = url.lastIndexOf('/')
    if (lastSlashIndex === -1 || lastSlashIndex === url.length - 1) {
        logMessage('getFilenameFromUrl', 'URL格式无效缺少文件名,默认文件名download', false)
        return 'download' // 返回一个默认的文件名
    }
    var filenameWithExtension = url.substring(lastSlashIndex + 1)
    var decodedFilename = decodeURIComponent(filenameWithExtension)
    decodedFilename = decodedFilename.replace(/%20/g, '_') // 替换所有的 %20 为下划线
    return decodedFilename
}