Skip to content
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

Allow custom file icons #32331

Draft
wants to merge 5 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions custom/conf/app.example.ini
Original file line number Diff line number Diff line change
Expand Up @@ -1262,6 +1262,9 @@ LEVEL = Info
;; Leave it empty to allow users to select any theme from "{CustomPath}/public/assets/css/theme-*.css"
;THEMES =
;;
;; Icon theme used by the file tree. Icons have to be place in data/icons/<iconpack>/. Default is "".
;FILE_ICON_THEME =
;;
;; All available reactions users can choose on issues/prs and comments.
;; Values can be emoji alias (:smile:) or a unicode emoji.
;; For custom reactions, add a tightly cropped square image to public/assets/img/emoji/reaction_name.png
Expand Down
24 changes: 0 additions & 24 deletions modules/base/tool.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,8 +21,6 @@ import (
"time"
"unicode/utf8"

"code.gitea.io/gitea/modules/git"
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/setting"

"github.com/dustin/go-humanize"
Expand Down Expand Up @@ -165,28 +163,6 @@ func Int64sToStrings(ints []int64) []string {
return strs
}

// EntryIcon returns the octicon class for displaying files/directories
func EntryIcon(entry *git.TreeEntry) string {
switch {
case entry.IsLink():
te, err := entry.FollowLink()
if err != nil {
log.Debug(err.Error())
return "file-symlink-file"
}
if te.IsDir() {
return "file-directory-symlink"
}
return "file-symlink-file"
case entry.IsDir():
return "file-directory-fill"
case entry.IsSubModule():
return "file-submodule"
}

return "file"
}

// SetupGiteaRoot Sets GITEA_ROOT if it is not already set and returns the value
func SetupGiteaRoot() string {
giteaRoot := os.Getenv("GITEA_ROOT")
Expand Down
2 changes: 0 additions & 2 deletions modules/base/tool_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -168,8 +168,6 @@ func TestInt64sToStrings(t *testing.T) {
)
}

// TODO: Test EntryIcon

func TestSetupGiteaRoot(t *testing.T) {
_ = os.Setenv("GITEA_ROOT", "test")
assert.Equal(t, "test", SetupGiteaRoot())
Expand Down
105 changes: 105 additions & 0 deletions modules/fileicon/fileicon.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,105 @@
package fileicon

import (
"context"
"html/template"
"path"
"strings"

"code.gitea.io/gitea/modules/git"
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/svg"
lru "github.com/hashicorp/golang-lru/v2"
)

var fileIconCache *lru.Cache[string, string]

func init() {
var err error
fileIconCache, err = lru.New[string, string](1000)
if err != nil {
log.Fatal("Failed to create file icon cache: %v", err)
}
}

func getBasicFileIconName(entry *git.TreeEntry) string {
switch {
case entry.IsLink():
te, err := entry.FollowLink()
if err != nil {
log.Debug(err.Error())
return "octicon-file-symlink-file"
}
if te.IsDir() {
return "octicon-file-directory-symlink"
}
return "octicon-file-symlink-file"
case entry.IsDir():
return "octicon-file-directory-fill"
case entry.IsSubModule():
return "octicon-file-submodule"
}

return "octicon-file"
}

// getFileIconNames returns a list of possible icon names for a file or directory
// Folder named `sub-folder` =>
// - `folder_sub-folder“ (. will be replaced with _)
// - `folder`
//
// File named `.gitignore` =>
// - `file__gitignore` (. will be replaced with _)
// - `file_`
//
// File named `README.md` =>
// - `file_readme_md`
// - `file_md`
func getFileIconNames(entry *git.TreeEntry) []string {
fileName := strings.ReplaceAll(strings.ToLower(path.Base(entry.Name())), ".", "_")

if entry.IsDir() {
return []string{"folder_" + fileName, "folder"}
}

if entry.IsRegular() {
ext := strings.ToLower(strings.TrimPrefix(path.Ext(entry.Name()), "."))
return []string{"file_" + fileName, "file_" + ext, "file"}
}

return nil
}

type fileIconBackend interface {

Check failure on line 74 in modules/fileicon/fileicon.go

View workflow job for this annotation

GitHub Actions / lint-backend

type `fileIconBackend` is unused (unused)

Check failure on line 74 in modules/fileicon/fileicon.go

View workflow job for this annotation

GitHub Actions / lint-go-windows

type `fileIconBackend` is unused (unused)

Check failure on line 74 in modules/fileicon/fileicon.go

View workflow job for this annotation

GitHub Actions / lint-go-gogit

type `fileIconBackend` is unused (unused)
GetIcon(string) (string, error)
}

// FileIcon returns a custom icon from a folder or the default octicon for displaying files/directories
func FileIcon(ctx context.Context, entry *git.TreeEntry) template.HTML {
backend := &fileIconHTTPBackend{
theme: setting.UI.FileIconTheme,
baseURL: "https://raw.githubusercontent.com/anbraten/gitea-icons/refs/heads/master/gitea/",
}

iconTheme := setting.UI.FileIconTheme
if iconTheme != "" {
iconNames := getFileIconNames(entry)

// Try to load the custom icon
for _, iconName := range iconNames {
if icon, err := backend.GetIcon(iconName); err == nil {
if icon, ok := fileIconCache.Get(iconName); ok {
return svg.RenderHTMLFromString(icon)
}

fileIconCache.Add(iconName, string(icon))

Check failure on line 96 in modules/fileicon/fileicon.go

View workflow job for this annotation

GitHub Actions / lint-backend

unnecessary conversion (unconvert)

Check failure on line 96 in modules/fileicon/fileicon.go

View workflow job for this annotation

GitHub Actions / lint-go-windows

unnecessary conversion (unconvert)

Check failure on line 96 in modules/fileicon/fileicon.go

View workflow job for this annotation

GitHub Actions / lint-go-gogit

unnecessary conversion (unconvert)

return svg.RenderHTMLFromString(string(icon))

Check failure on line 98 in modules/fileicon/fileicon.go

View workflow job for this annotation

GitHub Actions / lint-backend

unnecessary conversion (unconvert)

Check failure on line 98 in modules/fileicon/fileicon.go

View workflow job for this annotation

GitHub Actions / lint-go-windows

unnecessary conversion (unconvert)

Check failure on line 98 in modules/fileicon/fileicon.go

View workflow job for this annotation

GitHub Actions / lint-go-gogit

unnecessary conversion (unconvert)
}
}
}

// If no custom icon was found or an error occurred, return the default icon
return svg.RenderHTML(getBasicFileIconName(entry))
}
31 changes: 31 additions & 0 deletions modules/fileicon/fs.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
package fileicon

import (
"os"
"path"

"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/setting"
)

type fileIconFolderBackend struct {

Check failure on line 11 in modules/fileicon/fs.go

View workflow job for this annotation

GitHub Actions / lint-backend

type `fileIconFolderBackend` is unused (unused)

Check failure on line 11 in modules/fileicon/fs.go

View workflow job for this annotation

GitHub Actions / lint-go-windows

type `fileIconFolderBackend` is unused (unused)

Check failure on line 11 in modules/fileicon/fs.go

View workflow job for this annotation

GitHub Actions / lint-go-gogit

type `fileIconFolderBackend` is unused (unused)
theme string
}

func (f *fileIconFolderBackend) GetIcon(iconName string) (string, error) {

Check failure on line 15 in modules/fileicon/fs.go

View workflow job for this annotation

GitHub Actions / lint-backend

func `(*fileIconFolderBackend).GetIcon` is unused (unused)

Check failure on line 15 in modules/fileicon/fs.go

View workflow job for this annotation

GitHub Actions / lint-go-windows

func `(*fileIconFolderBackend).GetIcon` is unused (unused)

Check failure on line 15 in modules/fileicon/fs.go

View workflow job for this annotation

GitHub Actions / lint-go-gogit

func `(*fileIconFolderBackend).GetIcon` is unused (unused)
iconPath := path.Join(setting.AppDataPath, "icons", f.theme, iconName+".svg")

log.Debug("Loading custom icon from %s", iconPath)

// Try to load the icon from the filesystem
if _, err := os.Stat(iconPath); err != nil {
return "", err
}

iconData, err := os.ReadFile(iconPath)
if err != nil {
return "", err
}

return string(iconData), nil
}
38 changes: 38 additions & 0 deletions modules/fileicon/http.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
package fileicon

import (
"fmt"
"io"
"net/http"
"path"

"code.gitea.io/gitea/modules/log"
)

type fileIconHTTPBackend struct {
theme string
baseURL string
}

func (f *fileIconHTTPBackend) GetIcon(iconName string) (string, error) {
iconPath := path.Join(f.baseURL, f.theme, iconName+".svg")

log.Info("Loading custom icon from %s", iconPath)

// Try to load the icon via HTTP get
res, err := http.Get(iconPath)
if err != nil {
return "", err
}

if res.StatusCode != http.StatusOK {
return "", fmt.Errorf("Failed to load icon: %s", res.Status)
}

resBody, err := io.ReadAll(res.Body)
if err != nil {
return "", err
}

return string(resBody), nil
}
1 change: 1 addition & 0 deletions modules/setting/ui.go
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ var UI = struct {
DefaultShowFullName bool
DefaultTheme string
Themes []string
FileIconTheme string
Reactions []string
ReactionsLookup container.Set[string] `ini:"-"`
CustomEmojis []string
Expand Down
28 changes: 19 additions & 9 deletions modules/svg/svg.go
Original file line number Diff line number Diff line change
Expand Up @@ -60,16 +60,26 @@ func MockIcon(icon string) func() {
func RenderHTML(icon string, others ...any) template.HTML {
size, class := gitea_html.ParseSizeAndClass(defaultSize, "", others...)
if svgStr, ok := svgIcons[icon]; ok {
// the code is somewhat hacky, but it just works, because the SVG contents are all normalized
if size != defaultSize {
svgStr = strings.Replace(svgStr, fmt.Sprintf(`width="%d"`, defaultSize), fmt.Sprintf(`width="%d"`, size), 1)
svgStr = strings.Replace(svgStr, fmt.Sprintf(`height="%d"`, defaultSize), fmt.Sprintf(`height="%d"`, size), 1)
}
if class != "" {
svgStr = strings.Replace(svgStr, `class="`, fmt.Sprintf(`class="%s `, class), 1)
}
return template.HTML(svgStr)
return RenderHTMLFromString(svgStr, size, class)
}

// during test (or something wrong happens), there is no SVG loaded, so use a dummy span to tell that the icon is missing
return template.HTML(fmt.Sprintf("<span>%s(%d/%s)</span>", template.HTMLEscapeString(icon), size, template.HTMLEscapeString(class)))
}

// RenderHTMLFromString renders icons from a string - arguments SVG string (string), size (int), class (string)
func RenderHTMLFromString(svgStr string, others ...any) template.HTML {
svgStr = string(Normalize([]byte(svgStr), defaultSize))

size, class := gitea_html.ParseSizeAndClass(defaultSize, "", others...)

// the code is somewhat hacky, but it just works, because the SVG contents are all normalized
if size != defaultSize {
svgStr = strings.Replace(svgStr, fmt.Sprintf(`width="%d"`, defaultSize), fmt.Sprintf(`width="%d"`, size), 1)
svgStr = strings.Replace(svgStr, fmt.Sprintf(`height="%d"`, defaultSize), fmt.Sprintf(`height="%d"`, size), 1)
}
if class != "" {
svgStr = strings.Replace(svgStr, `class="`, fmt.Sprintf(`class="%s `, class), 1)
}
return template.HTML(svgStr)
}
3 changes: 2 additions & 1 deletion modules/templates/helper.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ import (

user_model "code.gitea.io/gitea/models/user"
"code.gitea.io/gitea/modules/base"
"code.gitea.io/gitea/modules/fileicon"
"code.gitea.io/gitea/modules/markup"
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/svg"
Expand Down Expand Up @@ -58,7 +59,7 @@ func NewFuncMap() template.FuncMap {
// -----------------------------------------------------------------
// svg / avatar / icon / color
"svg": svg.RenderHTML,
"EntryIcon": base.EntryIcon,
"FileIcon": fileicon.FileIcon,
"MigrationIcon": migrationIcon,
"ActionIcon": actionIcon,
"SortArrow": sortArrow,
Expand Down
4 changes: 1 addition & 3 deletions templates/repo/view_list.tmpl
Original file line number Diff line number Diff line change
Expand Up @@ -24,8 +24,8 @@
<tr data-entryname="{{$entry.Name}}" data-ready="{{if $commit}}true{{else}}false{{end}}" class="{{if not $commit}}not{{end}}ready entry">
<td class="name four wide">
<span class="truncate">
{{FileIcon $.Context $entry}}
{{if $entry.IsSubModule}}
{{svg "octicon-file-submodule"}}
{{$refURL := $subModuleFile.RefURL AppUrl $.Repository.FullName $.SSHDomain}} {{/* FIXME: the usage of AppUrl seems incorrect, it would be fixed in the future, use AppSubUrl instead */}}
{{if $refURL}}
<a class="muted" href="{{$refURL}}">{{$entry.Name}}</a><span class="at">@</span><a href="{{$refURL}}/commit/{{PathEscape $subModuleFile.RefID}}">{{ShortSha $subModuleFile.RefID}}</a>
Expand All @@ -35,7 +35,6 @@
{{else}}
{{if $entry.IsDir}}
{{$subJumpablePathName := $entry.GetSubJumpablePathName}}
{{svg "octicon-file-directory-fill"}}
<a class="muted" href="{{$.TreeLink}}/{{PathEscapeSegments $subJumpablePathName}}" title="{{$subJumpablePathName}}">
{{$subJumpablePathFields := StringUtils.Split $subJumpablePathName "/"}}
{{$subJumpablePathFieldLast := (Eval (len $subJumpablePathFields) "-" 1)}}
Expand All @@ -47,7 +46,6 @@
{{end}}
</a>
{{else}}
{{svg (printf "octicon-%s" (EntryIcon $entry))}}
<a class="muted" href="{{$.TreeLink}}/{{PathEscapeSegments $entry.Name}}" title="{{$entry.Name}}">{{$entry.Name}}</a>
{{end}}
{{end}}
Expand Down
Loading