GitHub file list beautifier

Adds colors to files by type, displays small images in place of file-type icons in a repository source tree

// ==UserScript==
// @name              GitHub file list beautifier
// @description       Adds colors to files by type, displays small images in place of file-type icons in a repository source tree
// @name:zh-CN        GitHub 文件列表美化器
// @description:zh-CN GitHub 文件列表美化器是一个用户脚本,用于增强 GitHub 仓库中的文件显示效果。它可以为文件和文件夹添加颜色,并将文件类型图标替换为小图像,使得代码库更加易于浏览和管理。按类型为文件添加颜色和图标,在存储库源树中显示小图像以代替文件类型图标
// @name:en           GitHub File list beautifier
// @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
// @name:ja           GitHub ファイルリスト整形ツール
// @description:ja    GitHub File List Beautifier はユーザースクリプトです,強化するために使用される GitHub 倉庫内のファイルの表示効果。ファイルやフォルダーに色を追加できます,ファイルタイプのアイコンを小さな画像に置き換えます,コードベースの参照と管理が容易になります。。種類ごとにファイルに色とアイコンを追加する,リポジトリソースツリーのファイルタイプアイコンの代わりに小さな画像を表示します
// @name:ko           GitHub 파일 목록 미화자
// @description:ko    GitHub File List Beautifier는 사용자 스크립트입니다.,향상시키는 데 사용됨 GitHub 창고 내 파일 표시 효과。파일과 폴더에 색상을 추가할 수 있습니다.,파일 형식 아이콘을 작은 이미지로 바꿉니다.,코드 베이스를 더 쉽게 찾아보고 관리할 수 있습니다.。유형별로 파일에 색상 및 아이콘 추가,저장소 소스 트리의 파일 유형 아이콘 대신 작은 이미지 표시
// @name:ru           GitHub Средство украшения списка файлов
// @description:ru    GitHub File List Beautifier — пользовательский скрипт.,используется для улучшения GitHub Эффект отображения файлов на складе。Он может добавлять цвета к файлам и папкам.,и замените значки типов файлов небольшими изображениями,Упрощает просмотр и управление базой кода.。Добавляйте цвета и значки к файлам по типу,Отображение небольших изображений вместо значков типов файлов в дереве исходного кода репозитория.
// @name:zh-TW        GitHub 文件列表美化器
// @description:zh-TW GitHub 文件清單美化器是一個使用者腳本,用於增強 GitHub 倉庫中的文件顯示效果。它可以為文件和資料夾添加顏色,並將文件類型圖示替換為小圖像,使得程式碼庫更加易於瀏覽和管理。按類型為文件添加顏色和圖標,在儲存庫來源樹中顯示小圖像以取代文件類型圖標
// @name:zh-HK        GitHub 文件列表美化器
// @description:zh-HK GitHub 文件清單美化器是一個使用者腳本,用於增強 GitHub 倉庫中的文件顯示效果。它可以為文件和資料夾添加顏色,並將文件類型圖示替換為小圖像,使得程式碼庫更加易於瀏覽和管理。按類型為文件添加顏色和圖標,在儲存庫來源樹中顯示小圖像以取代文件類型圖標
// @license           MIT
// @version           4.1.0.6
// @match             https://github.com/*
// @grant             none
// @run-at            document-start
// @grant             GM_xmlhttpRequest
// @grant             GM_getValue
// @grant             GM_setValue
// @compatible        chrome
// @compatible        firefox
// @compatible        edge
// @compatible        opera
// @compatible        safari
// @author            yutian81
// @namespace         https://github.com/yutian81/greasyfork-js
// @icon              
// @license           MIT
// @supportURL        https://github.com/yutian81/greasyfork-js/issues
// @homepageURL       https://github.com/yutian81/greasyfork-js

// ==/UserScript==
'use strict'
let customColors = GM_getValue('fileTypesColors', {})
var DEBUG = false
var addIcon = true
if (DEBUG) {
    GM_setValue('fileTypesColors', {})
}
GM_setValue('fileTypesColors', {})
if (Object.keys(customColors).length === 0) {
    GM_xmlhttpRequest({
        method: 'GET',
        url: 'https://raw.githubusercontent.com/yutian81/greasyfork-js/refs/heads/main/colors.json',
        // url: 'https://raw.githubusercontent.com/ChinaGodMan/UserScripts/main/github-file-list-beautifier-plus/colors.json',
        onload: function (response) {
            try {
                customColors = JSON.parse(response.responseText)
                GM_setValue('fileTypesColors', customColors) // 保存到本地存储
                requestAnimationFrame(start)
            } catch (e) {
                console.error('解析颜色配置失败:', e)
            }
        },
        onerror: function () {
            console.error('加载颜色配置失败')
        }
    })
} else {
    requestAnimationFrame(start)
}
let savedConfig = {}
try {
    savedConfig = JSON.parse(localStorage.FileListBeautifier) || {}
} catch (e) { }
const config = Object.assign({},
    ...Object.entries({
        iconSize: 24,
        colorSeed1: 13,
        colorSeed2: 1299721,
        colorSeed3: 179426453
    }).map(([k, v]) => ({ [k]: +savedConfig[k] || v })))
const IMG_CLS = 'wOxxOm-image-icon'
const rxImages = /^(png|jpe?g|bmp|gif|cur|ico|svg)$/i
const styleQueue = []
const { sheet } = document.documentElement.appendChild($create('style', {
    textContent: /*language=CSS*/ `
    .${IMG_CLS} {
      width: ${config.iconSize}px;
      height: ${config.iconSize}px;
      object-fit: scale-down;
      margin: 0 -4px;
    }
     .qinwuyuan-file-icon {
      width: 16px; 
      height: 16px; 
      object-fit: scale-down;
      margin: 0 -4px;
    }
    a[file-type=":folder"] {
      font-weight: bold;
    }
    `.replace(/;/g, '!important;')
}))
const filetypes = {}
const ME = Symbol(GM_info.script.name)
const ob = new MutationObserver(start)
let lumaBias, lumaFix, lumaAmp
function start() {
    beautify()
    ob.observe(document, { subtree: true, childList: true })
}
function beautify() {
    for (const el of document.querySelectorAll('.react-directory-truncate, .js-navigation-open')) {
        if (ME in el)
            continue
        el[ME] = true
        const isOld = el.tagName === 'A'
        const a = isOld ? el : el.getElementsByTagName('a')[0]
        const url = a && a.href
        if (!url)
            continue
        const icon = el.closest(isOld ? '.js-navigation-item' : 'td').querySelector('svg')
        if (icon.classList.contains(isOld ? 'octicon-file-directory-fill' : 'icon-directory')) {
            a.setAttribute('file-type', ':folder')
            continue
        }
        let filename = url.split('/').pop().toLowerCase()
        let ext = (url.match(/\.(\w+)$|$/)[1] || filename).toLowerCase()
        if (customColors[filename]) {
            ext = filename
        }
        a.setAttribute('file-type', ext)
        const customIcon = customColors[filename] && customColors[filename].icon
            ? customColors[filename].icon
            : (customColors[ext] && customColors[ext].icon) || null
        if (!filetypes[ext])
            addFileTypeStyle(ext)
        
        if (customIcon && addIcon) {
            let iconUrl = customIcon
            iconUrl = iconUrl.replace(/\.svgl$/i, '.svg')
            if (!iconUrl.startsWith('https://') && 
                !iconUrl.startsWith('data:image') && 
                !iconUrl.endsWith('.svg')) {
                iconUrl += '.svg'
            }
            if (!iconUrl.startsWith('https://') && !iconUrl.startsWith('data:image')) {
                iconUrl = `https://raw.githubusercontent.com/PKief/vscode-material-icon-theme/main/icons/${iconUrl}`
            }
            const img = $create('img', {
                className: 'qinwuyuan-file-icon',
                src: iconUrl,
                alt: ext
            })
            img.onerror = function() {
                this.style.display = 'none'
                setTimeout(() => {
                    this.src = iconUrl.replace('PKief', 'the-userr').replace('vscode-material-icon-theme', 'GitHub-Icons')
                    this.style.display = ''
                }, 500)
            }
            icon.replaceWith(img)
        }
    }
}

function addFileTypeStyle(type) {
    filetypes[type] = true
    if (!styleQueue.length)
        requestAnimationFrame(commitStyleQueue)
    styleQueue.push(type)
}
function commitStyleQueue() {
    if (!lumaAmp) initLumaScale()
    const seed2 = config.colorSeed2
    const seed3 = config.colorSeed3
    for (const type of styleQueue) {
        const colorConfig = customColors[type]
        if (colorConfig) {
            const color = colorConfig.color
            if (color) {
                sheet.insertRule(/*language=CSS*/ `
                a[file-type="${type}"]:not(#foo) {
                  color: ${color} !important;
                }
              `)
            }
        } else {
            const hash = calcSimpleHash(type)
            const H = hash % 360
            const Hq = H / 60
            const S = hash * seed2 % 50 + 50 | 0
            const redFix = (Hq < 1 ? 1 - Hq : Hq > 4 ? (Hq - 4) / 2 : 0)
            const blueFix = (Hq < 3 || Hq > 5 ? 0 : Hq < 4 ? Hq - 3 : 5 - Hq) * 3
            const L = hash * seed3 % lumaAmp + lumaBias + (redFix + blueFix) * lumaFix * S / 100 | 0
            sheet.insertRule(/*language=CSS*/ `
            a[file-type="${type}"]:not(#foo) {
              color: hsl(${H},${S}%,${L}%) !important;
            }
          `)
        }
    }
    styleQueue.length = 0
}
function calcSimpleHash(text) {
    let hash = 0
    for (let i = 0, len = text.length; i < len; i++)
        hash = ((hash << 5) - hash) + text.charCodeAt(i)
    return Math.abs(hash * config.colorSeed1 | 0)
}
function initLumaScale() {
    const [, r, g, b] = getComputedStyle(document.body).backgroundColor.split(/[^\d.]+/).map(parseFloat)
    const isDark = (r * .2126 + g * .7152 + b * .0722) < 128;
    [lumaBias, lumaAmp, lumaFix] = isDark ? [30, 50, 12] : [25, 15, 0]
}
function $create(tag, props) {
    return Object.assign(document.createElement(tag), props)
}