-
-
Notifications
You must be signed in to change notification settings - Fork 5.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Show matching icons based on the filename in the repo tree view #24147
Changes from all commits
8c8391e
506f83d
e6dcafc
bf64d24
e2bbdf7
d1b3eea
fed8cae
4f43424
15a1715
b2aa31f
4acbbd7
7431263
14746ed
ed4a844
87f897e
eb941c1
b62288f
bcd7bbb
ab4793d
416ae9d
d1e22e5
95c79ec
57d61b5
22adca1
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Large diffs are not rendered by default.
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -2,14 +2,36 @@ | |
import fastGlob from 'fast-glob'; | ||
import {optimize} from 'svgo'; | ||
import {parse} from 'node:path'; | ||
import {readFile, writeFile, mkdir} from 'node:fs/promises'; | ||
import {readFile, writeFile, mkdir, copyFile, rm} from 'node:fs/promises'; | ||
import {fileURLToPath} from 'node:url'; | ||
import {execSync} from 'node:child_process'; | ||
|
||
const glob = (pattern) => fastGlob.sync(pattern, { | ||
cwd: fileURLToPath(new URL('..', import.meta.url)), | ||
absolute: true, | ||
}); | ||
|
||
const removeUnwantedSvgs = () => { | ||
// remove folder from icons as we have a custom, colorful material folder in web_src/svg | ||
const removeFolder = rm('node_modules/material-icon-theme/icons/folder.svg', {force: true}); | ||
|
||
// remove all icons of open folders as we don't use them anywhere | ||
const removeOpenFolders = glob('node_modules/material-icon-theme/icons/folder*-open.svg').map((file) => rm(file, {force: true})); | ||
|
||
return Promise.all([removeFolder, ...removeOpenFolders]); | ||
}; | ||
|
||
// inspired by https://github.com/Claudiohbsantos/github-material-icons-extension/blob/ff97e50980/scripts/build-dependencies.js | ||
const generateIconMap = async () => { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Could there be a comment about how much size does this icon package increase? |
||
// build icon map | ||
execSync('npm run generateJson', {cwd: 'node_modules/material-icon-theme'}); | ||
|
||
// copy icon map to assets | ||
const src = fileURLToPath(new URL('../node_modules/material-icon-theme/dist/material-icons.json', import.meta.url)); | ||
const dest = fileURLToPath(new URL('../assets/material-icons.json', import.meta.url)); | ||
await copyFile(src, dest); | ||
}; | ||
|
||
function exit(err) { | ||
if (err) console.error(err); | ||
process.exit(err ? 1 : 0); | ||
|
@@ -52,11 +74,14 @@ function processFiles(pattern, opts) { | |
} | ||
|
||
async function main() { | ||
await removeUnwantedSvgs(); | ||
try { | ||
await mkdir(fileURLToPath(new URL('../public/img/svg', import.meta.url)), {recursive: true}); | ||
} catch {} | ||
|
||
await Promise.all([ | ||
generateIconMap, | ||
...processFiles('node_modules/material-icon-theme/icons/*.svg', {prefix: 'material'}), | ||
...processFiles('node_modules/@primer/octicons/build/svg/*-16.svg', {prefix: 'octicon'}), | ||
...processFiles('web_src/svg/*.svg'), | ||
...processFiles('public/img/gitea.svg', {fullName: 'gitea-gitea'}), | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,100 @@ | ||
// Copyright 2023 The Gitea Authors. All rights reserved. | ||
// SPDX-License-Identifier: MIT | ||
|
||
// This file was ported from https://github.com/Claudiohbsantos/github-material-icons-extension/blob/ff97e50980/src/lib/replace-icon.js | ||
// to go with small changes. | ||
|
||
package icon | ||
|
||
import ( | ||
"os" | ||
"path" | ||
"strings" | ||
|
||
"code.gitea.io/gitea/modules/git" | ||
"code.gitea.io/gitea/modules/json" | ||
"code.gitea.io/gitea/modules/log" | ||
) | ||
|
||
type Map struct { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This type is exported, but I guess it only used in this package internally? |
||
FileNames map[string]string `json:"fileNames"` | ||
FolderNames map[string]string `json:"folderNames"` | ||
FileExtensions map[string]string `json:"fileExtensions"` | ||
LanguageIds map[string]string `json:"languageIds"` | ||
} | ||
|
||
var iconMap = Map{ | ||
FileNames: make(map[string]string), | ||
FolderNames: make(map[string]string), | ||
FileExtensions: make(map[string]string), | ||
LanguageIds: make(map[string]string), | ||
} | ||
|
||
func Init() error { | ||
data, err := os.ReadFile("assets/material-icons.json") | ||
if err != nil { | ||
return err | ||
} | ||
|
||
err = json.Unmarshal(data, &iconMap) | ||
if err != nil { | ||
return err | ||
} | ||
|
||
return nil | ||
} | ||
|
||
// EntryIcon returns the icon for the given git entry | ||
func EntryIcon(entry *git.TreeEntry) string { | ||
yardenshoham marked this conversation as resolved.
Show resolved
Hide resolved
|
||
if entry.IsLink() { | ||
te, err := entry.FollowLink() | ||
if err != nil { | ||
log.Debug(err.Error()) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It's better to make logs have some context texts or prefixes. Otherwise it's different to know where the log comes from if a user sees the "err.Error()" string. |
||
return "octicon-file-symlink-file" | ||
} | ||
if te.IsDir() { | ||
return "material-folder-symlink" | ||
} | ||
return "octicon-file-symlink-file" | ||
} | ||
return "material-" + lookForMaterialMatch(entry) | ||
} | ||
|
||
func lookForMaterialMatch(entry *git.TreeEntry) string { | ||
if entry.IsSubModule() { | ||
return "folder-git" | ||
} | ||
|
||
fileName := entry.Name() | ||
|
||
if !entry.IsDir() { | ||
if iconMap.FileNames[fileName] != "" { | ||
return iconMap.FileNames[fileName] | ||
} | ||
Comment on lines
+71
to
+73
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Here and below, it doesn't seem a Golang style. if name, ok := theMap[key]; ok {
return name
} |
||
|
||
lowerFileName := strings.ToLower(fileName) | ||
if iconMap.FileNames[lowerFileName] != "" { | ||
return iconMap.FileNames[lowerFileName] | ||
} | ||
|
||
fileExtension := strings.TrimPrefix(path.Ext(fileName), ".") | ||
if iconMap.FileExtensions[fileExtension] != "" { | ||
return iconMap.FileExtensions[fileExtension] | ||
} | ||
|
||
if iconMap.LanguageIds[fileExtension] != "" { | ||
return iconMap.LanguageIds[fileExtension] | ||
} | ||
return "file" | ||
} | ||
|
||
if iconMap.FolderNames[fileName] != "" { | ||
return iconMap.FolderNames[fileName] | ||
} | ||
|
||
lowerFileName := strings.ToLower(fileName) | ||
if iconMap.FolderNames[lowerFileName] != "" { | ||
return iconMap.FolderNames[lowerFileName] | ||
} | ||
return "folder" | ||
} |
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should
material-icons.json
also be processed to remove these entries?And does the
iconDefinitions
key could be removed?