You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

helper.go 13 kB

11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
Shows total tracked time in issue and milestone list (#3341) * Show total tracked time in issue and milestone list Show total tracked time at issue page Signed-off-by: Jonas Franz <info@jonasfranz.software> * Optimizing TotalTimes by using SumInt Signed-off-by: Jonas Franz <info@jonasfranz.software> * Fixing wrong total times for milestones caused by a missing JOIN Adding unit tests for total times Signed-off-by: Jonas Franz <info@jonasfranz.software> * Logging error instead of ignoring it Signed-off-by: Jonas Franz <info@jonasfranz.software> * Correcting spelling mistakes Signed-off-by: Jonas Franz <info@jonasfranz.software> * Change error message to a short version Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add error handling to TotalTimes Add variable for totalTimes Signed-off-by: Jonas Franz <info@jonasfranz.de> * Introduce TotalTrackedTimes as variable of issue Load TotalTrackedTimes by loading attributes of IssueList Load TotalTrackedTimes by loading attributes of single issue Add Sec2Time as helper to use it in templates Signed-off-by: Jonas Franz <info@jonasfranz.software> * Fixed test + gofmt Signed-off-by: Jonas Franz <info@jonasfranz.software> * Load TotalTrackedTimes via MilestoneList instead of single requests Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add documentation for MilestoneList Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add documentation for MilestoneList Signed-off-by: Jonas Franz <info@jonasfranz.software> * Fix test Signed-off-by: Jonas Franz <info@jonasfranz.software> * Change comment from SQL query to description Signed-off-by: Jonas Franz <info@jonasfranz.software> * Fix unit test by using int64 instead of int Signed-off-by: Jonas Franz <info@jonasfranz.software> * Fix unit test by using int64 instead of int Signed-off-by: Jonas Franz <info@jonasfranz.software> * Check if timetracker is enabled Signed-off-by: Jonas Franz <info@jonasfranz.software> * Fix test by enabling timetracking Signed-off-by: Jonas Franz <info@jonasfranz.de>
7 years ago
Issue due date (#3794) * Started adding deadline to ui * Implemented basic issue due date managing * Improved UI for due date managing * Added at least write access to the repo in order to modify issue due dates * Ui improvements * Added issue comments creation when adding/modifying/removing a due date * Show due date in issue list * Added api support for issue due dates * Fixed lint suggestions * Added deadline to sdk * Updated css * Added support for adding/modifiying deadlines for pull requests via api * Fixed comments not created when updating or removing a deadline * update sdk (will do properly once go-gitea/go-sdk#103 is merged) * enhanced updateIssueDeadline * Removed unnessecary Issue.DeadlineString * UI improvements * Small improvments to comment creation + ui & validation improvements * Check if an issue is overdue is now a seperate function * Updated go-sdk with govendor as it was merged * Simplified isOverdue method * removed unessecary deadline to 0 set * Update swagger definitions * Added missing return * Added an explanary comment * Improved updateIssueDeadline method so it'll only update `deadline_unix` * Small changes and improvements * no need to explicitly load the issue when updating a deadline, just use whats already there * small optimisations * Added check if a deadline was modified before updating it * Moved comment creating logic into its own function * Code cleanup for creating deadline comment * locale improvement * When modifying a deadline, the old deadline is saved with the comment * small improvments to xorm session handling when updating an issue deadline + style nitpicks * style nitpicks * Moved checking for if the user has write acces to middleware
7 years ago
Pull request review/approval and comment on code (#3748) * Initial ui components for pull request review * Add Review Add IssueComment types Signed-off-by: Jonas Franz <info@jonasfranz.software> (cherry picked from commit 2b4daab) Signed-off-by: Jonas Franz <info@jonasfranz.software> * Replace ReviewComment with Content Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add load functions Add ReviewID to findComments Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add create review comment implementation Add migration for review Other small changes Signed-off-by: Jonas Franz <info@jonasfranz.software> * Simplified create and find functions for review Signed-off-by: Jonas Franz <info@jonasfranz.software> * Moved "Pending" to first position Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add GetCurrentReview to simplify fetching current review Signed-off-by: Jonas Franz <info@jonasfranz.software> * Preview for listing comments Signed-off-by: Jonas Franz <info@jonasfranz.software> * Move new comment form to its own file Signed-off-by: Jonas Franz <info@jonasfranz.software> * Implement Review form Show Review comments on comment stream Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add support for single comments Showing buttons in context Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add pending tag to pending review comments Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add unit tests for Review Signed-off-by: Jonas Franz <info@jonasfranz.software> * Fetch all review ids at once Add unit tests Signed-off-by: Jonas Franz <info@jonasfranz.software> * gofmt Signed-off-by: Jonas Franz <info@jonasfranz.software> * Improved comment rendering in "Files" view by adding Comments to DiffLine Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add support for invalidating comments Signed-off-by: Jonas Franz <info@jonasfranz.software> * Switched back to code.gitea.io/git Signed-off-by: Jonas Franz <info@jonasfranz.software> * Moved review migration from v64 to v65 Signed-off-by: Jonas Franz <info@jonasfranz.software> * Rebuild css Signed-off-by: Jonas Franz <info@jonasfranz.software> * gofmt Signed-off-by: Jonas Franz <info@jonasfranz.software> * Improve translations Signed-off-by: Jonas Franz <info@jonasfranz.software> * Fix unit tests by updating fixtures and updating outdated test Signed-off-by: Jonas Franz <info@jonasfranz.software> * Comments will be shown at the right place now Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add support for deleting CodeComments Signed-off-by: Jonas Franz <info@jonasfranz.software> * Fix problems caused by files in subdirectories Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add support for showing code comments of reviews in conversation Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add support for "Show/Hide outdated" Signed-off-by: Jonas Franz <info@jonasfranz.software> * Update code.gitea.io/git Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add support for new webhooks Signed-off-by: Jonas Franz <info@jonasfranz.software> * Update comparison Signed-off-by: Jonas Franz <info@jonasfranz.software> * Resolve conflicts Signed-off-by: Jonas Franz <info@jonasfranz.software> * Minor UI improvements * update code.gitea.io/git * Fix ui bug reported by @lunny causing wrong position of add button Add functionality to "Cancel" button Add scale effects to add button Hide "Cancel" button for existing comments Signed-off-by: Jonas Franz <info@jonasfranz.software> * Prepare solving conflicts Signed-off-by: Jonas Franz <info@jonasfranz.software> * Show add button only if no comments already exist for the line Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add missing vendor files Signed-off-by: Jonas Franz <info@jonasfranz.software> * Check if reviewer is nil Signed-off-by: Jonas Franz <info@jonasfranz.software> * Show forms only to users who are logged in Signed-off-by: Jonas Franz <info@jonasfranz.software> * Revert "Show forms only to users who are logged in" This reverts commit c083682 Signed-off-by: Jonas Franz <info@jonasfranz.software> * Save patch in comment Render patch for code comments Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add link to comment in code Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add reply form to comment list Show forms only to signed in users Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add 'Reply' as translatable Add CODE_COMMENT_LINES setting Signed-off-by: Jonas Franz <info@jonasfranz.software> * gofmt Signed-off-by: Jonas Franz <info@jonasfranz.software> * Fix problems introduced by checking for singed in user Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add v70 Signed-off-by: Jonas Franz <info@jonasfranz.software> * Update generated stylesheet Signed-off-by: Jonas Franz <info@jonasfranz.software> * Fix preview Beginn with new review comment patch system Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add new algo to generate diff for line range Remove old algo used for cutting big diffs (it was very buggy) * Add documentation and example for CutDiffAroundLine * Fix example of CutDiffAroundLine * Fix some comment UI rendering bugs * Add code comment edit mode * Send notifications / actions to users until review gets published Fix diff generation bug Fix wrong hashtag * Fix vet errors * Send notifications also for single comments * Fix some notification bugs, fix link * Fix: add comment icon is only shown on code lines * Add lint comment * Add unit tests for git diff * Add more error messages * Regenerated css Signed-off-by: Jonas Franz <info@jonasfranz.software> * fmt Signed-off-by: Jonas Franz <info@jonasfranz.software> * Regenerated CSS with latest less version Signed-off-by: Jonas Franz <info@jonasfranz.software> * Fix test by updating comment type to new ID Signed-off-by: Jonas Franz <info@jonasfranz.software> * Introducing CodeComments as type for map[string]map[int64][]*Comment Other minor code improvements Signed-off-by: Jonas Franz <info@jonasfranz.software> * Fix data-tab issues Signed-off-by: Jonas Franz <info@jonasfranz.software> * Remove unnecessary change Signed-off-by: Jonas Franz <info@jonasfranz.software> * refactored checkForInvalidation Signed-off-by: Jonas Franz <info@jonasfranz.software> * Append comments instead of setting Signed-off-by: Jonas Franz <info@jonasfranz.software> * Use HeadRepo instead of BaseRepo Signed-off-by: Jonas Franz <info@jonasfranz.software> * Update migration Signed-off-by: Jonas Franz <info@jonasfranz.de> * Regenerated CSS Signed-off-by: Jonas Franz <info@jonasfranz.software> * Add copyright Signed-off-by: Jonas Franz <info@jonasfranz.software> * Update index.css Signed-off-by: Jonas Franz <info@jonasfranz.software>
6 years ago
9 years ago
11 years ago
11 years ago
11 years ago
10 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486
  1. // Copyright 2014 The Gogs Authors. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. package templates
  5. import (
  6. "bytes"
  7. "container/list"
  8. "encoding/json"
  9. "errors"
  10. "fmt"
  11. "html"
  12. "html/template"
  13. "mime"
  14. "net/url"
  15. "path/filepath"
  16. "runtime"
  17. "strings"
  18. "time"
  19. "code.gitea.io/gitea/models"
  20. "code.gitea.io/gitea/modules/base"
  21. "code.gitea.io/gitea/modules/log"
  22. "code.gitea.io/gitea/modules/markup"
  23. "code.gitea.io/gitea/modules/setting"
  24. "golang.org/x/net/html/charset"
  25. "golang.org/x/text/transform"
  26. "gopkg.in/editorconfig/editorconfig-core-go.v1"
  27. )
  28. // NewFuncMap returns functions for injecting to templates
  29. func NewFuncMap() []template.FuncMap {
  30. return []template.FuncMap{map[string]interface{}{
  31. "GoVer": func() string {
  32. return strings.Title(runtime.Version())
  33. },
  34. "UseHTTPS": func() bool {
  35. return strings.HasPrefix(setting.AppURL, "https")
  36. },
  37. "AppName": func() string {
  38. return setting.AppName
  39. },
  40. "AppSubUrl": func() string {
  41. return setting.AppSubURL
  42. },
  43. "AppUrl": func() string {
  44. return setting.AppURL
  45. },
  46. "AppVer": func() string {
  47. return setting.AppVer
  48. },
  49. "AppBuiltWith": func() string {
  50. return setting.AppBuiltWith
  51. },
  52. "AppDomain": func() string {
  53. return setting.Domain
  54. },
  55. "DisableGravatar": func() bool {
  56. return setting.DisableGravatar
  57. },
  58. "ShowFooterTemplateLoadTime": func() bool {
  59. return setting.ShowFooterTemplateLoadTime
  60. },
  61. "LoadTimes": func(startTime time.Time) string {
  62. return fmt.Sprint(time.Since(startTime).Nanoseconds()/1e6) + "ms"
  63. },
  64. "AvatarLink": base.AvatarLink,
  65. "Safe": Safe,
  66. "SafeJS": SafeJS,
  67. "Str2html": Str2html,
  68. "TimeSince": base.TimeSince,
  69. "TimeSinceUnix": base.TimeSinceUnix,
  70. "RawTimeSince": base.RawTimeSince,
  71. "FileSize": base.FileSize,
  72. "Subtract": base.Subtract,
  73. "EntryIcon": base.EntryIcon,
  74. "Add": func(a, b int) int {
  75. return a + b
  76. },
  77. "ActionIcon": ActionIcon,
  78. "DateFmtLong": func(t time.Time) string {
  79. return t.Format(time.RFC1123Z)
  80. },
  81. "DateFmtShort": func(t time.Time) string {
  82. return t.Format("Jan 02, 2006")
  83. },
  84. "SizeFmt": func(s int64) string {
  85. return base.FileSize(s)
  86. },
  87. "List": List,
  88. "SubStr": func(str string, start, length int) string {
  89. if len(str) == 0 {
  90. return ""
  91. }
  92. end := start + length
  93. if length == -1 {
  94. end = len(str)
  95. }
  96. if len(str) < end {
  97. return str
  98. }
  99. return str[start:end]
  100. },
  101. "EllipsisString": base.EllipsisString,
  102. "DiffTypeToStr": DiffTypeToStr,
  103. "DiffLineTypeToStr": DiffLineTypeToStr,
  104. "Sha1": Sha1,
  105. "ShortSha": base.ShortSha,
  106. "MD5": base.EncodeMD5,
  107. "ActionContent2Commits": ActionContent2Commits,
  108. "PathEscape": url.PathEscape,
  109. "EscapePound": func(str string) string {
  110. return strings.NewReplacer("%", "%25", "#", "%23", " ", "%20", "?", "%3F").Replace(str)
  111. },
  112. "RenderCommitMessage": RenderCommitMessage,
  113. "RenderCommitMessageLink": RenderCommitMessageLink,
  114. "RenderCommitBody": RenderCommitBody,
  115. "IsMultilineCommitMessage": IsMultilineCommitMessage,
  116. "ThemeColorMetaTag": func() string {
  117. return setting.UI.ThemeColorMetaTag
  118. },
  119. "MetaAuthor": func() string {
  120. return setting.UI.Meta.Author
  121. },
  122. "MetaDescription": func() string {
  123. return setting.UI.Meta.Description
  124. },
  125. "MetaKeywords": func() string {
  126. return setting.UI.Meta.Keywords
  127. },
  128. "FilenameIsImage": func(filename string) bool {
  129. mimeType := mime.TypeByExtension(filepath.Ext(filename))
  130. return strings.HasPrefix(mimeType, "image/")
  131. },
  132. "TabSizeClass": func(ec *editorconfig.Editorconfig, filename string) string {
  133. if ec != nil {
  134. def := ec.GetDefinitionForFilename(filename)
  135. if def.TabWidth > 0 {
  136. return fmt.Sprintf("tab-size-%d", def.TabWidth)
  137. }
  138. }
  139. return "tab-size-8"
  140. },
  141. "SubJumpablePath": func(str string) []string {
  142. var path []string
  143. index := strings.LastIndex(str, "/")
  144. if index != -1 && index != len(str) {
  145. path = append(path, str[0:index+1])
  146. path = append(path, str[index+1:])
  147. } else {
  148. path = append(path, str)
  149. }
  150. return path
  151. },
  152. "JsonPrettyPrint": func(in string) string {
  153. var out bytes.Buffer
  154. err := json.Indent(&out, []byte(in), "", " ")
  155. if err != nil {
  156. return ""
  157. }
  158. return out.String()
  159. },
  160. "DisableGitHooks": func() bool {
  161. return setting.DisableGitHooks
  162. },
  163. "DisableImportLocal": func() bool {
  164. return !setting.ImportLocalPaths
  165. },
  166. "TrN": TrN,
  167. "Dict": func(values ...interface{}) (map[string]interface{}, error) {
  168. if len(values)%2 != 0 {
  169. return nil, errors.New("invalid dict call")
  170. }
  171. dict := make(map[string]interface{}, len(values)/2)
  172. for i := 0; i < len(values); i += 2 {
  173. key, ok := values[i].(string)
  174. if !ok {
  175. return nil, errors.New("dict keys must be strings")
  176. }
  177. dict[key] = values[i+1]
  178. }
  179. return dict, nil
  180. },
  181. "Printf": fmt.Sprintf,
  182. "Escape": Escape,
  183. "Sec2Time": models.SecToTime,
  184. "ParseDeadline": func(deadline string) []string {
  185. return strings.Split(deadline, "|")
  186. },
  187. "DefaultTheme": func() string {
  188. return setting.UI.DefaultTheme
  189. },
  190. "dict": func(values ...interface{}) (map[string]interface{}, error) {
  191. if len(values) == 0 {
  192. return nil, errors.New("invalid dict call")
  193. }
  194. dict := make(map[string]interface{})
  195. for i := 0; i < len(values); i++ {
  196. switch key := values[i].(type) {
  197. case string:
  198. i++
  199. if i == len(values) {
  200. return nil, errors.New("specify the key for non array values")
  201. }
  202. dict[key] = values[i]
  203. case map[string]interface{}:
  204. m := values[i].(map[string]interface{})
  205. for i, v := range m {
  206. dict[i] = v
  207. }
  208. default:
  209. return nil, errors.New("dict values must be maps")
  210. }
  211. }
  212. return dict, nil
  213. },
  214. }}
  215. }
  216. // Safe render raw as HTML
  217. func Safe(raw string) template.HTML {
  218. return template.HTML(raw)
  219. }
  220. // SafeJS renders raw as JS
  221. func SafeJS(raw string) template.JS {
  222. return template.JS(raw)
  223. }
  224. // Str2html render Markdown text to HTML
  225. func Str2html(raw string) template.HTML {
  226. return template.HTML(markup.Sanitize(raw))
  227. }
  228. // Escape escapes a HTML string
  229. func Escape(raw string) string {
  230. return html.EscapeString(raw)
  231. }
  232. // List traversings the list
  233. func List(l *list.List) chan interface{} {
  234. e := l.Front()
  235. c := make(chan interface{})
  236. go func() {
  237. for e != nil {
  238. c <- e.Value
  239. e = e.Next()
  240. }
  241. close(c)
  242. }()
  243. return c
  244. }
  245. // Sha1 returns sha1 sum of string
  246. func Sha1(str string) string {
  247. return base.EncodeSha1(str)
  248. }
  249. // ToUTF8WithErr converts content to UTF8 encoding
  250. func ToUTF8WithErr(content []byte) (string, error) {
  251. charsetLabel, err := base.DetectEncoding(content)
  252. if err != nil {
  253. return "", err
  254. } else if charsetLabel == "UTF-8" {
  255. return string(content), nil
  256. }
  257. encoding, _ := charset.Lookup(charsetLabel)
  258. if encoding == nil {
  259. return string(content), fmt.Errorf("Unknown encoding: %s", charsetLabel)
  260. }
  261. // If there is an error, we concatenate the nicely decoded part and the
  262. // original left over. This way we won't loose data.
  263. result, n, err := transform.String(encoding.NewDecoder(), string(content))
  264. if err != nil {
  265. result = result + string(content[n:])
  266. }
  267. return result, err
  268. }
  269. // ToUTF8 converts content to UTF8 encoding and ignore error
  270. func ToUTF8(content string) string {
  271. res, _ := ToUTF8WithErr([]byte(content))
  272. return res
  273. }
  274. // ReplaceLeft replaces all prefixes 'old' in 's' with 'new'.
  275. func ReplaceLeft(s, old, new string) string {
  276. oldLen, newLen, i, n := len(old), len(new), 0, 0
  277. for ; i < len(s) && strings.HasPrefix(s[i:], old); n++ {
  278. i += oldLen
  279. }
  280. // simple optimization
  281. if n == 0 {
  282. return s
  283. }
  284. // allocating space for the new string
  285. curLen := n*newLen + len(s[i:])
  286. replacement := make([]byte, curLen, curLen)
  287. j := 0
  288. for ; j < n*newLen; j += newLen {
  289. copy(replacement[j:j+newLen], new)
  290. }
  291. copy(replacement[j:], s[i:])
  292. return string(replacement)
  293. }
  294. // RenderCommitMessage renders commit message with XSS-safe and special links.
  295. func RenderCommitMessage(msg, urlPrefix string, metas map[string]string) template.HTML {
  296. return RenderCommitMessageLink(msg, urlPrefix, "", metas)
  297. }
  298. // RenderCommitMessageLink renders commit message as a XXS-safe link to the provided
  299. // default url, handling for special links.
  300. func RenderCommitMessageLink(msg, urlPrefix, urlDefault string, metas map[string]string) template.HTML {
  301. cleanMsg := template.HTMLEscapeString(msg)
  302. // we can safely assume that it will not return any error, since there
  303. // shouldn't be any special HTML.
  304. fullMessage, err := markup.RenderCommitMessage([]byte(cleanMsg), urlPrefix, urlDefault, metas)
  305. if err != nil {
  306. log.Error(3, "RenderCommitMessage: %v", err)
  307. return ""
  308. }
  309. msgLines := strings.Split(strings.TrimSpace(string(fullMessage)), "\n")
  310. if len(msgLines) == 0 {
  311. return template.HTML("")
  312. }
  313. return template.HTML(msgLines[0])
  314. }
  315. // RenderCommitBody extracts the body of a commit message without its title.
  316. func RenderCommitBody(msg, urlPrefix string, metas map[string]string) template.HTML {
  317. cleanMsg := template.HTMLEscapeString(msg)
  318. fullMessage, err := markup.RenderCommitMessage([]byte(cleanMsg), urlPrefix, "", metas)
  319. if err != nil {
  320. log.Error(3, "RenderCommitMessage: %v", err)
  321. return ""
  322. }
  323. body := strings.Split(strings.TrimSpace(string(fullMessage)), "\n")
  324. if len(body) == 0 {
  325. return template.HTML("")
  326. }
  327. return template.HTML(strings.Join(body[1:], "\n"))
  328. }
  329. // IsMultilineCommitMessage checks to see if a commit message contains multiple lines.
  330. func IsMultilineCommitMessage(msg string) bool {
  331. return strings.Count(strings.TrimSpace(msg), "\n") >= 1
  332. }
  333. // Actioner describes an action
  334. type Actioner interface {
  335. GetOpType() models.ActionType
  336. GetActUserName() string
  337. GetRepoUserName() string
  338. GetRepoName() string
  339. GetRepoPath() string
  340. GetRepoLink() string
  341. GetBranch() string
  342. GetContent() string
  343. GetCreate() time.Time
  344. GetIssueInfos() []string
  345. }
  346. // ActionIcon accepts an action operation type and returns an icon class name.
  347. func ActionIcon(opType models.ActionType) string {
  348. switch opType {
  349. case models.ActionCreateRepo, models.ActionTransferRepo:
  350. return "repo"
  351. case models.ActionCommitRepo, models.ActionPushTag, models.ActionDeleteTag, models.ActionDeleteBranch:
  352. return "git-commit"
  353. case models.ActionCreateIssue:
  354. return "issue-opened"
  355. case models.ActionCreatePullRequest:
  356. return "git-pull-request"
  357. case models.ActionCommentIssue:
  358. return "comment-discussion"
  359. case models.ActionMergePullRequest:
  360. return "git-merge"
  361. case models.ActionCloseIssue, models.ActionClosePullRequest:
  362. return "issue-closed"
  363. case models.ActionReopenIssue, models.ActionReopenPullRequest:
  364. return "issue-reopened"
  365. case models.ActionMirrorSyncPush, models.ActionMirrorSyncCreate, models.ActionMirrorSyncDelete:
  366. return "repo-clone"
  367. default:
  368. return "invalid type"
  369. }
  370. }
  371. // ActionContent2Commits converts action content to push commits
  372. func ActionContent2Commits(act Actioner) *models.PushCommits {
  373. push := models.NewPushCommits()
  374. if err := json.Unmarshal([]byte(act.GetContent()), push); err != nil {
  375. log.Error(4, "json.Unmarshal:\n%s\nERROR: %v", act.GetContent(), err)
  376. }
  377. return push
  378. }
  379. // DiffTypeToStr returns diff type name
  380. func DiffTypeToStr(diffType int) string {
  381. diffTypes := map[int]string{
  382. 1: "add", 2: "modify", 3: "del", 4: "rename",
  383. }
  384. return diffTypes[diffType]
  385. }
  386. // DiffLineTypeToStr returns diff line type name
  387. func DiffLineTypeToStr(diffType int) string {
  388. switch diffType {
  389. case 2:
  390. return "add"
  391. case 3:
  392. return "del"
  393. case 4:
  394. return "tag"
  395. }
  396. return "same"
  397. }
  398. // Language specific rules for translating plural texts
  399. var trNLangRules = map[string]func(int64) int{
  400. "en-US": func(cnt int64) int {
  401. if cnt == 1 {
  402. return 0
  403. }
  404. return 1
  405. },
  406. "lv-LV": func(cnt int64) int {
  407. if cnt%10 == 1 && cnt%100 != 11 {
  408. return 0
  409. }
  410. return 1
  411. },
  412. "ru-RU": func(cnt int64) int {
  413. if cnt%10 == 1 && cnt%100 != 11 {
  414. return 0
  415. }
  416. return 1
  417. },
  418. "zh-CN": func(cnt int64) int {
  419. return 0
  420. },
  421. "zh-HK": func(cnt int64) int {
  422. return 0
  423. },
  424. "zh-TW": func(cnt int64) int {
  425. return 0
  426. },
  427. }
  428. // TrN returns key to be used for plural text translation
  429. func TrN(lang string, cnt interface{}, key1, keyN string) string {
  430. var c int64
  431. if t, ok := cnt.(int); ok {
  432. c = int64(t)
  433. } else if t, ok := cnt.(int16); ok {
  434. c = int64(t)
  435. } else if t, ok := cnt.(int32); ok {
  436. c = int64(t)
  437. } else if t, ok := cnt.(int64); ok {
  438. c = t
  439. } else {
  440. return keyN
  441. }
  442. ruleFunc, ok := trNLangRules[lang]
  443. if !ok {
  444. ruleFunc = trNLangRules["en-US"]
  445. }
  446. if ruleFunc(c) == 0 {
  447. return key1
  448. }
  449. return keyN
  450. }