Github 复制原始文件 URL 与下载文件

在每个文件行的末尾添加按钮,以复制原始文件 URL 和下载文件

当前为 2024-09-11 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Github Copy Raw File URL and Download File
  3. // @name:zh-CN Github 复制原始文件 URL 与下载文件
  4. // @description:zh-CN 在每个文件行的末尾添加按钮,以复制原始文件 URL 和下载文件
  5. // @name:ar Github نسخ الملف الأصلي URL مع تنزيل الملفات
  6. // @description:ar زر إضافة في نهاية كل سطر ملف,لنسخ الملف الأصلي URL وتحميل الملفات
  7. // @name:bg Github Копирайте оригиналния файл URL с файлове за изтегляне
  8. // @description:bg Бутон за добавяне в края на всеки файлов ред,за да копирате оригиналния файл URL и изтегляне на файлове
  9. // @name:cs Github Zkopírujte původní soubor URL se staženými soubory
  10. // @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
  11. // @name:da Github Kopiér den originale fil URL med download filer
  12. // @description:da Tilføj knap i slutningen af ​​hver fillinje,for at kopiere den originale fil URL og download filer
  13. // @name:de Github Originaldatei kopieren URL mit Download-Dateien
  14. // @description:de Schaltfläche „Hinzufügen“ am Ende jeder Dateizeile,um die Originaldatei zu kopieren URL und Dateien herunterladen
  15. // @name:el Github Αντιγραφή αρχικού αρχείου URL με λήψη αρχείων
  16. // @description:el Κουμπί προσθήκης στο τέλος κάθε γραμμής αρχείου,για να αντιγράψετε το αρχικό αρχείο URL και λήψη αρχείων
  17. // @name:en Github Copy original file URL with download files
  18. // @description:en Add button at the end of each file line,to copy the original file URL and download files
  19. // @name:eo Github Kopiu originalan dosieron URL kun elŝutaj dosieroj
  20. // @description:eo Aldoni butonon ĉe la fino de ĉiu dosierlinio,por kopii la originalan dosieron URL kaj elŝuti dosierojn
  21. // @name:es Github Copiar archivo original URL con archivos de descarga
  22. // @description:es Botón Agregar al final de cada línea de archivo,para copiar el archivo original URL y descargar archivos
  23. // @name:fi Github Kopioi alkuperäinen tiedosto URL lataustiedostojen kanssa
  24. // @description:fi Lisää-painike jokaisen tiedostorivin lopussa,kopioidaksesi alkuperäisen tiedoston URL ja ladata tiedostoja
  25. // @name:fr Github Copier le fichier original URL avec des fichiers à télécharger
  26. // @description:fr Ajouter un bouton à la fin de chaque ligne de fichier,pour copier le fichier original URL et télécharger des fichiers
  27. // @name:he Github העתק את הקובץ המקורי URL עם קבצים להורדה
  28. // @description:he כפתור הוסף בסוף כל שורת קובץ,כדי להעתיק את הקובץ המקורי URL ולהוריד קבצים
  29. // @name:hr Github Kopiraj izvornu datoteku URL s datotekama za preuzimanje
  30. // @description:hr Dodaj gumb na kraju svakog retka datoteke,za kopiranje izvorne datoteke URL i preuzimanje datoteka
  31. // @name:hu Github Eredeti fájl másolása URL letölthető fájlokkal
  32. // @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
  33. // @name:id Github Salin file asli URL dengan file unduhan
  34. // @description:id Tombol Tambah di akhir setiap baris file,untuk menyalin file asli URL dan mengunduh file
  35. // @name:it Github Copia il file originale URL con file scaricabili
  36. // @description:it Pulsante Aggiungi alla fine di ogni riga del file,per copiare il file originale URL e scaricare file
  37. // @name:ja Github 元のファイルをコピーする URL ダウンロードファイル付き
  38. // @description:ja 各ファイル行の末尾にある「追加」ボタン,元のファイルをコピーするには URL そしてファイルをダウンロードする
  39. // @name:ka Github დააკოპირეთ ორიგინალი ფაილი URL ჩამოტვირთვის ფაილებით
  40. // @description:ka დაამატეთ ღილაკი თითოეული ფაილის ხაზის ბოლოს,ორიგინალური ფაილის კოპირება URL და ჩამოტვირთეთ ფაილები
  41. // @name:ko Github 원본 파일 복사 URL 다운로드 파일 포함
  42. // @description:ko 각 파일 줄 끝에 추가 버튼,원본 파일을 복사하려면 URL 그리고 파일 다운로드
  43. // @name:nl Github Kopieer origineel bestand URL met downloadbestanden
  44. // @description:nl Knop Toevoegen aan het einde van elke bestandsregel,om het originele bestand te kopiëren URL en bestanden downloaden
  45. // @name:nb Github Kopier originalfilen URL med nedlastingsfiler
  46. // @description:nb Legg til-knapp på slutten av hver fillinje,for å kopiere originalfilen URL og last ned filer
  47. // @name:pl Github Skopiuj oryginalny plik URL z plikami do pobrania
  48. // @description:pl Dodaj przycisk na końcu każdej linii pliku,aby skopiować oryginalny plik URL i pobierz pliki
  49. // @name:pt-BR Github Copiar arquivo original URL com download de arquivos
  50. // @description:pt-BR Botão Adicionar no final de cada linha do arquivo,para copiar o arquivo original URL e baixar arquivos
  51. // @name:ro Github Copiați fișierul original URL cu fișiere descărcate
  52. // @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
  53. // @name:ru Github Скопировать исходный файл URL с загрузкой файлов
  54. // @description:ru Добавить кнопку в конце каждой строки файла,скопировать исходный файл URL и скачать файлы
  55. // @name:sk Github Skopírujte pôvodný súbor URL so stiahnutými súbormi
  56. // @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
  57. // @name:sr Github Копирајте оригиналну датотеку URL са датотекама за преузимање
  58. // @description:sr Дугме Додај на крају сваке линије датотеке,да копирате оригиналну датотеку URL и преузимање датотека
  59. // @name:sv Github Kopiera originalfilen URL med nedladdningsfiler
  60. // @description:sv Lägg till-knappen i slutet av varje filrad,för att kopiera originalfilen URL och ladda ner filer
  61. // @name:th Github คัดลอกไฟล์ต้นฉบับ URL พร้อมดาวน์โหลดไฟล์
  62. // @description:th เพิ่มปุ่มที่ท้ายแต่ละบรรทัดไฟล์,เพื่อคัดลอกไฟล์ต้นฉบับ URL และดาวน์โหลดไฟล์
  63. // @name:tr Github Orijinal dosyayı kopyala URL indirme dosyalarıyla
  64. // @description:tr Her dosya satırının sonuna ekle düğmesi,orijinal dosyayı kopyalamak için URL ve dosyaları indir
  65. // @name:ug Github ئەسلى ھۆججەتنى كۆچۈرۈڭ URL چۈشۈرۈش ھۆججىتى بىلەن
  66. // @description:ug ھەر بىر ھۆججەت قۇرنىڭ ئاخىرىغا كۇنۇپكا قوشۇڭ,ئەسلى ھۆججەتنى كۆچۈرۈش URL ھۆججەتلەرنى چۈشۈرۈڭ
  67. // @name:uk Github Скопіюйте вихідний файл URL з файлами для завантаження
  68. // @description:uk Кнопка «Додати» в кінці кожного рядка файлу,щоб скопіювати вихідний файл URL і завантажити файли
  69. // @name:vi Github Sao chép tập tin gốc URL với các tập tin tải về
  70. // @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
  71. // @name:zh-TW Github 複製原始文件 URL 與下載文件
  72. // @description:zh-TW 在每個文件行的末尾新增按鈕,以複製原始文件 URL 和下載文件
  73. // @name:zh-HK Github 複製原始文件 URL 與下載文件
  74. // @description:zh-HK 在每個文件行的末尾新增按鈕,以複製原始文件 URL 和下載文件
  75. // @name:fr-CA Github Copier le fichier original URL avec des fichiers à télécharger
  76. // @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
  77. // @description Add buttons at the end of each file line to copy the raw file URL and download the file
  78. // @namespace https://github.com/ChinaGodMan/UserScripts
  79. // @version 2.2.0.20
  80. // @author Kamikaze (https://github.com/Kamiikaze) ,人民的勤务员 <toniaiwanowskiskr47@gmail.com>
  81. // @match https://github.com/*
  82. // @icon data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACAAAAAgCAYAAABzenr0AAADY0lEQVR42q1XTUwTURDek3pQEw+7/dmlqbSlaFFBiZgQA0JEDwjIBTwT4pF4RgxyQsSgQOGCBzApBsQTJnjgxxCinrDeeoDYlDSpJEArxLb8jO9LKKnt27K0+yWTwM68mXkz82amgla4XK5TJkm6Z5Qkt1EU5xn5DJL0B4S/8Q08yEBW0AtWUTQyxcMGUQwzIo0UxhmcFbKF3W4/zZR0mQyGHSjNhnAWOqArm1t/gxI9CLo0R0ORpKssnwEc1JOgE7qPvTnHuK5OwIZqzlPDfru8nB63ttIlp/PExgrsdmptaaG71dVp6eDWBIolVcm7sTEC9vb2yOPx0JWiIsL3/IsXqbKighrq6uhhfT3dqawkh81G4BUWFNDbkRGKx+MEfJqe5tVEV1roedW+vLxMydjZ3iafz0cHBwfEw8rKCm1tbVEy/H4/93X8lwq8WV4YvV4v5Yq1tTW1lzF81OHUmszkxATlitnZWdVmBdsCWidPIN9qpUAgQLliY2ODLhcWcp2AbYTfzWMODgyQXhj3eNTS4BYwRHjM1dVV0gvhcJjrAGzDAR8v/HrjRkkJzwGfgHGayrhZWkp6435NTZoDsI0aSHv/NtZodAYuxauBHTjg5+VnfX2d9EI0GiWLovAc+AUHvvMc+DA5SXphfm5OdUwLJlF8w2MiZ3rhUXMz1wHYRiNq5DFBQ2435Yr34+Okph+2BafTeY79E1EJEb3o7qbd3V06KTCwcAGz0ajmQMQlimcTw2gwwXjW0UEfp6YwYo+ErxcX08ueHvqysECbm5ukhkgkQouLi/S6r49ulZVlXk4MhgEhAUUU7cyJGBhOh4NgCKO3uqoq7eCD2lrVcdzY0KB1R4zmSZItdSF5nhCw5+dTMBhEO04bJJ9nZkgNS0tLWh3o5K5krDN5E0JP2toI+B0K0fDQED1tbydsSLFYjFSA9BxrHDZU13Sz2ZzHhIIQRPFglp8UssmUyYEgbAiZIEvStWQnmpuaCNU8NjpKr3p7aX9/nzIgU9UHoVvQAlmWFRaqHzxFWDYzAPseL+w/2c0t2fw060TFJitDX88EJp9cbDFGXVar9YyQLfBc2Jvtxw4nm81/WQQGiegroxCj+CGFDr/158lyBGMW246iKA5BL6BrWSyWC1rSx6J3Xqvef4nmwJHOHrGpAAAAAElFTkSuQmCC
  83. // @iconbak https://github.githubassets.com/pinned-octocat.svg
  84. // @run-at document-ready
  85. // @license MIT
  86. // @supportURL https://github.com/ChinaGodMan/UserScripts/issues
  87. // @homepageURL https://github.com/ChinaGodMan/UserScripts
  88. // ==/UserScript==
  89.  
  90.  
  91. // Need an Interval to detect path changes on github tree one-pager
  92. // Define the number of seconds
  93. const scanInterval = 2
  94.  
  95.  
  96. const waitForFilelist = setInterval(() => {
  97. let fileListContainer = document.querySelector("div.Box > div.js-details-container.Details div") || document.querySelector("table")
  98. let fileList = []
  99. let isTable = false
  100.  
  101. if (fileListContainer.tBodies) {
  102. fileList = fileListContainer.tBodies[0].children
  103. isTable = true
  104. } else {
  105. fileList = fileListContainer.children
  106. }
  107.  
  108. if (fileList < 1) return
  109.  
  110. appendButtons(fileList, isTable)
  111.  
  112. }, scanInterval * 1000)
  113.  
  114. function appendButtons(fileList, isTable = false) {
  115. let fileUrl = ""
  116. let rawFileUrl = ""
  117. for (let i = 0; i < fileList.length; i++) {
  118. let file = fileList[i]
  119.  
  120. if (file.classList.contains("cp-btn-rdy")) continue
  121.  
  122. file.classList.add("cp-btn-rdy")
  123.  
  124. if (!isTable) {
  125. if (
  126. file.classList.contains("sr-only") ||
  127. file.childElementCount !== 4
  128. ) continue
  129.  
  130. fileUrl = file.querySelector('div:nth-child(2) .js-navigation-open')
  131. .href
  132. } else {
  133. if (i === 0) continue
  134.  
  135. if (
  136. file.classList.contains("sr-only")
  137. ) continue
  138.  
  139.  
  140. fileUrl = file.querySelector("a")
  141. .href
  142. file = file.querySelector("td:nth-child(4) > div")
  143. }
  144. //alert(fileUrl)
  145. // Dont add button if its a folder
  146. if (!fileUrl.includes("/blob/")) continue
  147.  
  148. rawFileUrl = fileUrl.replace('/blob/', '/raw/')
  149. file.style = "display: flex; justify-content: flex-end;"
  150. file.append(creatyCopyButton(rawFileUrl))
  151. file.append(creatyDownButton(rawFileUrl))
  152. }
  153. };
  154.  
  155. function creatyCopyButton(copyText) {
  156. const copy2clipboard = `
  157. <clipboard-copy aria-label="Copy" value="test value" data-view-component="true" class="" tabindex="0" role="button" title="Copy raw file url">
  158. <svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-copy">
  159. <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>
  160. </svg>
  161. </clipboard-copy>`
  162.  
  163. const copyButton = document.createElement('div')
  164. copyButton.setAttribute('role', 'gridcell')
  165. copyButton.style = "margin-left: 10px; display: inline;"
  166. copyButton.innerHTML = copy2clipboard
  167. copyButton.children[0].value = copyText
  168. copyButton.children[0].style = "cursor: pointer;"
  169.  
  170. return copyButton
  171. }
  172. function creatyDownButton(copyText) {
  173. const copy2clipboard = `
  174. <clipboard-copy aria-label="Download" value="test value" data-view-component="true" class="" tabindex="0" role="button" title="Download raw file url">
  175. <svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-download">
  176. <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>
  177. </svg>
  178. </clipboard-copy>
  179. `
  180.  
  181. const copyButton = document.createElement('div')
  182. copyButton.setAttribute('role', 'gridcell')
  183. copyButton.style = "margin-left: 10px; display: inline;"
  184. copyButton.innerHTML = copy2clipboard
  185. copyButton.children[0].value = copyText
  186. copyButton.children[0].style = "cursor: pointer;"
  187. copyButton.addEventListener('click', () => {
  188. // window.location.href = copyText;
  189. downloadFile(copyText, getFilenameFromUrl(copyText))
  190. })
  191. return copyButton
  192. }
  193. function downloadFile(url, filename) {
  194. var xhr = new XMLHttpRequest()
  195. xhr.open('GET', url, true)
  196. xhr.responseType = 'blob'
  197. xhr.onload = function () {
  198. if (xhr.status === 200) {
  199. var blob = xhr.response
  200. var objectUrl = window.URL.createObjectURL(blob)
  201. var a = document.createElement('a')
  202. a.href = objectUrl
  203. a.download = filename // 设置下载文件名
  204. document.body.appendChild(a)
  205. a.click()
  206. window.URL.revokeObjectURL(objectUrl) // 清理 object URL
  207. document.body.removeChild(a) // 清理 DOM
  208. }
  209. }
  210. xhr.send()
  211. }
  212. function getFilenameFromUrl(url) {
  213. if (typeof url !== 'string' || url.trim() === '') {
  214. logMessage('getFilenameFromUrl', 'URL无效,默认文件名download', false)
  215. return 'download' // 返回一个默认的文件名
  216. }
  217. var lastSlashIndex = url.lastIndexOf('/')
  218. if (lastSlashIndex === -1 || lastSlashIndex === url.length - 1) {
  219. logMessage('getFilenameFromUrl', 'URL格式无效缺少文件名,默认文件名download', false)
  220. return 'download' // 返回一个默认的文件名
  221. }
  222. var filenameWithExtension = url.substring(lastSlashIndex + 1)
  223. var decodedFilename = decodeURIComponent(filenameWithExtension)
  224. decodedFilename = decodedFilename.replace(/%20/g, '_') // 替换所有的 %20 为下划线
  225. return decodedFilename
  226. }