GitHub 文件列表美化器

GitHub 文件列表美化器是一个用户脚本,用于增强 GitHub 仓库中的文件显示效果。它可以为文件和文件夹添加颜色,并将文件类型图标替换为小图像,使得代码库更加易于浏览和管理。按类型为文件添加颜色和图标,在存储库源树中显示小图像以代替文件类型图标

  1. // ==UserScript==
  2. // @name GitHub file list beautifier
  3. // @description Adds colors to files by type, displays small images in place of file-type icons in a repository source tree
  4. // @name:zh-CN GitHub 文件列表美化器
  5. // @description:zh-CN GitHub 文件列表美化器是一个用户脚本,用于增强 GitHub 仓库中的文件显示效果。它可以为文件和文件夹添加颜色,并将文件类型图标替换为小图像,使得代码库更加易于浏览和管理。按类型为文件添加颜色和图标,在存储库源树中显示小图像以代替文件类型图标
  6. // @name:en GitHub File list beautifier
  7. // @description:en GitHub File List Beautifier is a user script,used to enhance GitHub Display effect of files in warehouse。It can add colors to files and folders,and replace the file type icons with small images,Makes the code base easier to browse and manage。Add colors and icons to files by type,Display small images in place of file type icons in repository source tree
  8. // @name:ja GitHub ファイルリスト整形ツール
  9. // @description:ja GitHub File List Beautifier はユーザースクリプトです,強化するために使用される GitHub 倉庫内のファイルの表示効果。ファイルやフォルダーに色を追加できます,ファイルタイプのアイコンを小さな画像に置き換えます,コードベースの参照と管理が容易になります。。種類ごとにファイルに色とアイコンを追加する,リポジトリソースツリーのファイルタイプアイコンの代わりに小さな画像を表示します
  10. // @name:ko GitHub 파일 목록 미화자
  11. // @description:ko GitHub File List Beautifier는 사용자 스크립트입니다.,향상시키는 데 사용됨 GitHub 창고 내 파일 표시 효과。파일과 폴더에 색상을 추가할 수 있습니다.,파일 형식 아이콘을 작은 이미지로 바꿉니다.,코드 베이스를 더 쉽게 찾아보고 관리할 수 있습니다.。유형별로 파일에 색상 및 아이콘 추가,저장소 소스 트리의 파일 유형 아이콘 대신 작은 이미지 표시
  12. // @name:ru GitHub Средство украшения списка файлов
  13. // @description:ru GitHub File List Beautifier — пользовательский скрипт.,используется для улучшения GitHub Эффект отображения файлов на складе。Он может добавлять цвета к файлам и папкам.,и замените значки типов файлов небольшими изображениями,Упрощает просмотр и управление базой кода.。Добавляйте цвета и значки к файлам по типу,Отображение небольших изображений вместо значков типов файлов в дереве исходного кода репозитория.
  14. // @name:zh-TW GitHub 文件列表美化器
  15. // @description:zh-TW GitHub 文件清單美化器是一個使用者腳本,用於增強 GitHub 倉庫中的文件顯示效果。它可以為文件和資料夾添加顏色,並將文件類型圖示替換為小圖像,使得程式碼庫更加易於瀏覽和管理。按類型為文件添加顏色和圖標,在儲存庫來源樹中顯示小圖像以取代文件類型圖標
  16. // @name:zh-HK GitHub 文件列表美化器
  17. // @description:zh-HK GitHub 文件清單美化器是一個使用者腳本,用於增強 GitHub 倉庫中的文件顯示效果。它可以為文件和資料夾添加顏色,並將文件類型圖示替換為小圖像,使得程式碼庫更加易於瀏覽和管理。按類型為文件添加顏色和圖標,在儲存庫來源樹中顯示小圖像以取代文件類型圖標
  18. // @license MIT
  19. // @version 4.1.0.6
  20. // @match https://github.com/*
  21. // @grant none
  22. // @run-at document-start
  23. // @grant GM_xmlhttpRequest
  24. // @grant GM_getValue
  25. // @grant GM_setValue
  26. // @compatible chrome
  27. // @compatible firefox
  28. // @compatible edge
  29. // @compatible opera
  30. // @compatible safari
  31. // @author yutian81
  32. // @namespace https://github.com/yutian81/greasyfork-js
  33. // @icon 
  34. // @license MIT
  35. // @supportURL https://github.com/yutian81/greasyfork-js/issues
  36. // @homepageURL https://github.com/yutian81/greasyfork-js
  37.  
  38. // ==/UserScript==
  39. 'use strict'
  40. let customColors = GM_getValue('fileTypesColors', {})
  41. var DEBUG = false
  42. var addIcon = true
  43. if (DEBUG) {
  44. GM_setValue('fileTypesColors', {})
  45. }
  46. GM_setValue('fileTypesColors', {})
  47. if (Object.keys(customColors).length === 0) {
  48. GM_xmlhttpRequest({
  49. method: 'GET',
  50. url: 'https://raw.githubusercontent.com/yutian81/greasyfork-js/refs/heads/main/colors.json',
  51. // url: 'https://raw.githubusercontent.com/ChinaGodMan/UserScripts/main/github-file-list-beautifier-plus/colors.json',
  52. onload: function (response) {
  53. try {
  54. customColors = JSON.parse(response.responseText)
  55. GM_setValue('fileTypesColors', customColors) // 保存到本地存储
  56. requestAnimationFrame(start)
  57. } catch (e) {
  58. console.error('解析颜色配置失败:', e)
  59. }
  60. },
  61. onerror: function () {
  62. console.error('加载颜色配置失败')
  63. }
  64. })
  65. } else {
  66. requestAnimationFrame(start)
  67. }
  68. let savedConfig = {}
  69. try {
  70. savedConfig = JSON.parse(localStorage.FileListBeautifier) || {}
  71. } catch (e) { }
  72. const config = Object.assign({},
  73. ...Object.entries({
  74. iconSize: 24,
  75. colorSeed1: 13,
  76. colorSeed2: 1299721,
  77. colorSeed3: 179426453
  78. }).map(([k, v]) => ({ [k]: +savedConfig[k] || v })))
  79. const IMG_CLS = 'wOxxOm-image-icon'
  80. const rxImages = /^(png|jpe?g|bmp|gif|cur|ico|svg)$/i
  81. const styleQueue = []
  82. const { sheet } = document.documentElement.appendChild($create('style', {
  83. textContent: /*language=CSS*/ `
  84. .${IMG_CLS} {
  85. width: ${config.iconSize}px;
  86. height: ${config.iconSize}px;
  87. object-fit: scale-down;
  88. margin: 0 -4px;
  89. }
  90. .qinwuyuan-file-icon {
  91. width: 16px;
  92. height: 16px;
  93. object-fit: scale-down;
  94. margin: 0 -4px;
  95. }
  96. a[file-type=":folder"] {
  97. font-weight: bold;
  98. }
  99. `.replace(/;/g, '!important;')
  100. }))
  101. const filetypes = {}
  102. const ME = Symbol(GM_info.script.name)
  103. const ob = new MutationObserver(start)
  104. let lumaBias, lumaFix, lumaAmp
  105. function start() {
  106. beautify()
  107. ob.observe(document, { subtree: true, childList: true })
  108. }
  109. function beautify() {
  110. for (const el of document.querySelectorAll('.react-directory-truncate, .js-navigation-open')) {
  111. if (ME in el)
  112. continue
  113. el[ME] = true
  114. const isOld = el.tagName === 'A'
  115. const a = isOld ? el : el.getElementsByTagName('a')[0]
  116. const url = a && a.href
  117. if (!url)
  118. continue
  119. const icon = el.closest(isOld ? '.js-navigation-item' : 'td').querySelector('svg')
  120. if (icon.classList.contains(isOld ? 'octicon-file-directory-fill' : 'icon-directory')) {
  121. a.setAttribute('file-type', ':folder')
  122. continue
  123. }
  124. let filename = url.split('/').pop().toLowerCase()
  125. let ext = (url.match(/\.(\w+)$|$/)[1] || filename).toLowerCase()
  126. if (customColors[filename]) {
  127. ext = filename
  128. }
  129. a.setAttribute('file-type', ext)
  130. const customIcon = customColors[filename] && customColors[filename].icon
  131. ? customColors[filename].icon
  132. : (customColors[ext] && customColors[ext].icon) || null
  133. if (!filetypes[ext])
  134. addFileTypeStyle(ext)
  135. if (customIcon && addIcon) {
  136. let iconUrl = customIcon
  137. iconUrl = iconUrl.replace(/\.svgl$/i, '.svg')
  138. if (!iconUrl.startsWith('https://') &&
  139. !iconUrl.startsWith('data:image') &&
  140. !iconUrl.endsWith('.svg')) {
  141. iconUrl += '.svg'
  142. }
  143. if (!iconUrl.startsWith('https://') && !iconUrl.startsWith('data:image')) {
  144. iconUrl = `https://raw.githubusercontent.com/PKief/vscode-material-icon-theme/main/icons/${iconUrl}`
  145. }
  146. const img = $create('img', {
  147. className: 'qinwuyuan-file-icon',
  148. src: iconUrl,
  149. alt: ext
  150. })
  151. img.onerror = function() {
  152. this.style.display = 'none'
  153. setTimeout(() => {
  154. this.src = iconUrl.replace('PKief', 'the-userr').replace('vscode-material-icon-theme', 'GitHub-Icons')
  155. this.style.display = ''
  156. }, 500)
  157. }
  158. icon.replaceWith(img)
  159. }
  160. }
  161. }
  162.  
  163. function addFileTypeStyle(type) {
  164. filetypes[type] = true
  165. if (!styleQueue.length)
  166. requestAnimationFrame(commitStyleQueue)
  167. styleQueue.push(type)
  168. }
  169. function commitStyleQueue() {
  170. if (!lumaAmp) initLumaScale()
  171. const seed2 = config.colorSeed2
  172. const seed3 = config.colorSeed3
  173. for (const type of styleQueue) {
  174. const colorConfig = customColors[type]
  175. if (colorConfig) {
  176. const color = colorConfig.color
  177. if (color) {
  178. sheet.insertRule(/*language=CSS*/ `
  179. a[file-type="${type}"]:not(#foo) {
  180. color: ${color} !important;
  181. }
  182. `)
  183. }
  184. } else {
  185. const hash = calcSimpleHash(type)
  186. const H = hash % 360
  187. const Hq = H / 60
  188. const S = hash * seed2 % 50 + 50 | 0
  189. const redFix = (Hq < 1 ? 1 - Hq : Hq > 4 ? (Hq - 4) / 2 : 0)
  190. const blueFix = (Hq < 3 || Hq > 5 ? 0 : Hq < 4 ? Hq - 3 : 5 - Hq) * 3
  191. const L = hash * seed3 % lumaAmp + lumaBias + (redFix + blueFix) * lumaFix * S / 100 | 0
  192. sheet.insertRule(/*language=CSS*/ `
  193. a[file-type="${type}"]:not(#foo) {
  194. color: hsl(${H},${S}%,${L}%) !important;
  195. }
  196. `)
  197. }
  198. }
  199. styleQueue.length = 0
  200. }
  201. function calcSimpleHash(text) {
  202. let hash = 0
  203. for (let i = 0, len = text.length; i < len; i++)
  204. hash = ((hash << 5) - hash) + text.charCodeAt(i)
  205. return Math.abs(hash * config.colorSeed1 | 0)
  206. }
  207. function initLumaScale() {
  208. const [, r, g, b] = getComputedStyle(document.body).backgroundColor.split(/[^\d.]+/).map(parseFloat)
  209. const isDark = (r * .2126 + g * .7152 + b * .0722) < 128;
  210. [lumaBias, lumaAmp, lumaFix] = isDark ? [30, 50, 12] : [25, 15, 0]
  211. }
  212. function $create(tag, props) {
  213. return Object.assign(document.createElement(tag), props)
  214. }

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址