2023-04-29 15:02:29 +03:00
// Copyright 2023 The Gitea Authors. All rights reserved.
// SPDX-License-Identifier: MIT
package templates
import (
"context"
"html/template"
"mime"
"path/filepath"
2023-08-08 13:44:19 +03:00
"strconv"
2023-04-29 15:02:29 +03:00
"strings"
"time"
activities_model "code.gitea.io/gitea/models/activities"
repo_model "code.gitea.io/gitea/models/repo"
"code.gitea.io/gitea/modules/git"
giturl "code.gitea.io/gitea/modules/git/url"
"code.gitea.io/gitea/modules/json"
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/repository"
"code.gitea.io/gitea/modules/svg"
"github.com/editorconfig/editorconfig-core-go/v2"
)
func SortArrow ( normSort , revSort , urlSort string , isDefault bool ) template . HTML {
// if needed
if len ( normSort ) == 0 || len ( urlSort ) == 0 {
return ""
}
if len ( urlSort ) == 0 && isDefault {
// if sort is sorted as default add arrow tho this table header
if isDefault {
return svg . RenderHTML ( "octicon-triangle-down" , 16 )
}
} else {
// if sort arg is in url test if it correlates with column header sort arguments
// the direction of the arrow should indicate the "current sort order", up means ASC(normal), down means DESC(rev)
if urlSort == normSort {
// the table is sorted with this header normal
return svg . RenderHTML ( "octicon-triangle-up" , 16 )
} else if urlSort == revSort {
// the table is sorted with this header reverse
return svg . RenderHTML ( "octicon-triangle-down" , 16 )
}
}
// the table is NOT sorted with this header
return ""
}
// IsMultilineCommitMessage checks to see if a commit message contains multiple lines.
func IsMultilineCommitMessage ( msg string ) bool {
return strings . Count ( strings . TrimSpace ( msg ) , "\n" ) >= 1
}
// Actioner describes an action
type Actioner interface {
GetOpType ( ) activities_model . ActionType
2023-09-29 15:12:54 +03:00
GetActUserName ( ctx context . Context ) string
GetRepoUserName ( ctx context . Context ) string
GetRepoName ( ctx context . Context ) string
GetRepoPath ( ctx context . Context ) string
GetRepoLink ( ctx context . Context ) string
2023-04-29 15:02:29 +03:00
GetBranch ( ) string
GetContent ( ) string
GetCreate ( ) time . Time
GetIssueInfos ( ) [ ] string
}
// ActionIcon accepts an action operation type and returns an icon class name.
func ActionIcon ( opType activities_model . ActionType ) string {
switch opType {
case activities_model . ActionCreateRepo , activities_model . ActionTransferRepo , activities_model . ActionRenameRepo :
return "repo"
2023-10-08 02:26:27 +03:00
case activities_model . ActionCommitRepo :
2023-04-29 15:02:29 +03:00
return "git-commit"
2023-10-08 02:26:27 +03:00
case activities_model . ActionDeleteBranch :
return "git-branch"
2023-04-29 15:02:29 +03:00
case activities_model . ActionMergePullRequest , activities_model . ActionAutoMergePullRequest :
return "git-merge"
2023-10-08 02:26:27 +03:00
case activities_model . ActionCreatePullRequest :
return "git-pull-request"
case activities_model . ActionClosePullRequest :
return "git-pull-request-closed"
case activities_model . ActionCreateIssue :
return "issue-opened"
case activities_model . ActionCloseIssue :
2023-04-29 15:02:29 +03:00
return "issue-closed"
case activities_model . ActionReopenIssue , activities_model . ActionReopenPullRequest :
return "issue-reopened"
2023-10-08 02:26:27 +03:00
case activities_model . ActionCommentIssue , activities_model . ActionCommentPull :
return "comment-discussion"
2023-04-29 15:02:29 +03:00
case activities_model . ActionMirrorSyncPush , activities_model . ActionMirrorSyncCreate , activities_model . ActionMirrorSyncDelete :
return "mirror"
case activities_model . ActionApprovePullRequest :
return "check"
case activities_model . ActionRejectPullRequest :
2023-10-08 02:26:27 +03:00
return "file-diff"
case activities_model . ActionPublishRelease , activities_model . ActionPushTag , activities_model . ActionDeleteTag :
2023-04-29 15:02:29 +03:00
return "tag"
case activities_model . ActionPullReviewDismissed :
return "x"
default :
return "question"
}
}
// ActionContent2Commits converts action content to push commits
func ActionContent2Commits ( act Actioner ) * repository . PushCommits {
push := repository . NewPushCommits ( )
if act == nil || act . GetContent ( ) == "" {
return push
}
if err := json . Unmarshal ( [ ] byte ( act . GetContent ( ) ) , push ) ; err != nil {
log . Error ( "json.Unmarshal:\n%s\nERROR: %v" , act . GetContent ( ) , err )
}
if push . Len == 0 {
push . Len = len ( push . Commits )
}
return push
}
// MigrationIcon returns a SVG name matching the service an issue/comment was migrated from
func MigrationIcon ( hostname string ) string {
switch hostname {
case "github.com" :
return "octicon-mark-github"
default :
return "gitea-git"
}
}
type remoteAddress struct {
Address string
Username string
Password string
}
2024-04-13 18:41:57 +03:00
func mirrorRemoteAddress ( ctx context . Context , m * repo_model . Repository , remoteName string ) remoteAddress {
ret := remoteAddress { }
remoteURL , err := git . GetRemoteAddress ( ctx , m . RepoPath ( ) , remoteName )
if err != nil {
log . Error ( "GetRemoteURL %v" , err )
return ret
2023-04-29 15:02:29 +03:00
}
u , err := giturl . Parse ( remoteURL )
if err != nil {
log . Error ( "giturl.Parse %v" , err )
2024-04-13 18:41:57 +03:00
return ret
2023-04-29 15:02:29 +03:00
}
if u . Scheme != "ssh" && u . Scheme != "file" {
if u . User != nil {
2024-04-13 18:41:57 +03:00
ret . Username = u . User . Username ( )
ret . Password , _ = u . User . Password ( )
2023-04-29 15:02:29 +03:00
}
}
2024-04-13 18:41:57 +03:00
// The URL stored in the git repo could contain authentication,
// erase it, or it will be shown in the UI.
u . User = nil
ret . Address = u . String ( )
// Why not use m.OriginalURL to set ret.Address?
// It should be OK to use it, since m.OriginalURL should be the same as the authentication-erased URL from the Git repository.
// However, the old code has already stored authentication in m.OriginalURL when updating mirror settings.
// That means we need to use "giturl.Parse" for m.OriginalURL again to ensure authentication is erased.
// Instead of doing this, why not directly use the authentication-erased URL from the Git repository?
// It should be the same as long as there are no bugs.
return ret
2023-04-29 15:02:29 +03:00
}
func FilenameIsImage ( filename string ) bool {
mimeType := mime . TypeByExtension ( filepath . Ext ( filename ) )
return strings . HasPrefix ( mimeType , "image/" )
}
2023-08-08 13:44:19 +03:00
func TabSizeClass ( ec * editorconfig . Editorconfig , filename string ) string {
2023-04-29 15:02:29 +03:00
if ec != nil {
2023-08-08 13:44:19 +03:00
def , err := ec . GetDefinitionForFilename ( filename )
if err == nil && def . TabWidth >= 1 && def . TabWidth <= 16 {
return "tab-size-" + strconv . Itoa ( def . TabWidth )
2023-04-29 15:02:29 +03:00
}
}
2023-08-08 13:44:19 +03:00
return "tab-size-4"
2023-04-29 15:02:29 +03:00
}