mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-11-04 01:34:27 +00:00 
			
		
		
		
	Try to backport #33315, the only trivial conflict is in the helper functions map in the helper.go Fix #33333 Co-authored-by: Sysoev, Vladimir <i@vsysoev.ru>
This commit is contained in:
		@@ -48,7 +48,7 @@ func (s Stopwatch) Seconds() int64 {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
// Duration returns a human-readable duration string based on local server time
 | 
					// Duration returns a human-readable duration string based on local server time
 | 
				
			||||||
func (s Stopwatch) Duration() string {
 | 
					func (s Stopwatch) Duration() string {
 | 
				
			||||||
	return util.SecToTime(s.Seconds())
 | 
						return util.SecToHours(s.Seconds())
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func getStopwatch(ctx context.Context, userID, issueID int64) (sw *Stopwatch, exists bool, err error) {
 | 
					func getStopwatch(ctx context.Context, userID, issueID int64) (sw *Stopwatch, exists bool, err error) {
 | 
				
			||||||
@@ -201,7 +201,7 @@ func FinishIssueStopwatch(ctx context.Context, user *user_model.User, issue *Iss
 | 
				
			|||||||
		Doer:    user,
 | 
							Doer:    user,
 | 
				
			||||||
		Issue:   issue,
 | 
							Issue:   issue,
 | 
				
			||||||
		Repo:    issue.Repo,
 | 
							Repo:    issue.Repo,
 | 
				
			||||||
		Content: util.SecToTime(timediff),
 | 
							Content: util.SecToHours(timediff),
 | 
				
			||||||
		Type:    CommentTypeStopTracking,
 | 
							Type:    CommentTypeStopTracking,
 | 
				
			||||||
		TimeID:  tt.ID,
 | 
							TimeID:  tt.ID,
 | 
				
			||||||
	}); err != nil {
 | 
						}); err != nil {
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -70,7 +70,7 @@ func NewFuncMap() template.FuncMap {
 | 
				
			|||||||
		// time / number / format
 | 
							// time / number / format
 | 
				
			||||||
		"FileSize": base.FileSize,
 | 
							"FileSize": base.FileSize,
 | 
				
			||||||
		"CountFmt": base.FormatNumberSI,
 | 
							"CountFmt": base.FormatNumberSI,
 | 
				
			||||||
		"Sec2Time": util.SecToTime,
 | 
							"Sec2Time": util.SecToHours,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		"TimeEstimateString": timeEstimateString,
 | 
							"TimeEstimateString": timeEstimateString,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -8,59 +8,17 @@ import (
 | 
				
			|||||||
	"strings"
 | 
						"strings"
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// SecToTime converts an amount of seconds to a human-readable string. E.g.
 | 
					// SecToHours converts an amount of seconds to a human-readable hours string.
 | 
				
			||||||
// 66s			-> 1 minute 6 seconds
 | 
					// This is stable for planning and managing timesheets.
 | 
				
			||||||
// 52410s		-> 14 hours 33 minutes
 | 
					// Here it only supports hours and minutes, because a work day could contain 6 or 7 or 8 hours.
 | 
				
			||||||
// 563418		-> 6 days 12 hours
 | 
					func SecToHours(durationVal any) string {
 | 
				
			||||||
// 1563418		-> 2 weeks 4 days
 | 
					 | 
				
			||||||
// 3937125s     -> 1 month 2 weeks
 | 
					 | 
				
			||||||
// 45677465s	-> 1 year 6 months
 | 
					 | 
				
			||||||
func SecToTime(durationVal any) string {
 | 
					 | 
				
			||||||
	duration, _ := ToInt64(durationVal)
 | 
						duration, _ := ToInt64(durationVal)
 | 
				
			||||||
 | 
						hours := duration / 3600
 | 
				
			||||||
 | 
						minutes := (duration / 60) % 60
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	formattedTime := ""
 | 
						formattedTime := ""
 | 
				
			||||||
 | 
						formattedTime = formatTime(hours, "hour", formattedTime)
 | 
				
			||||||
	// The following four variables are calculated by taking
 | 
						formattedTime = formatTime(minutes, "minute", formattedTime)
 | 
				
			||||||
	// into account the previously calculated variables, this avoids
 | 
					 | 
				
			||||||
	// pitfalls when using remainders. As that could lead to incorrect
 | 
					 | 
				
			||||||
	// results when the calculated number equals the quotient number.
 | 
					 | 
				
			||||||
	remainingDays := duration / (60 * 60 * 24)
 | 
					 | 
				
			||||||
	years := remainingDays / 365
 | 
					 | 
				
			||||||
	remainingDays -= years * 365
 | 
					 | 
				
			||||||
	months := remainingDays * 12 / 365
 | 
					 | 
				
			||||||
	remainingDays -= months * 365 / 12
 | 
					 | 
				
			||||||
	weeks := remainingDays / 7
 | 
					 | 
				
			||||||
	remainingDays -= weeks * 7
 | 
					 | 
				
			||||||
	days := remainingDays
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	// The following three variables are calculated without depending
 | 
					 | 
				
			||||||
	// on the previous calculated variables.
 | 
					 | 
				
			||||||
	hours := (duration / 3600) % 24
 | 
					 | 
				
			||||||
	minutes := (duration / 60) % 60
 | 
					 | 
				
			||||||
	seconds := duration % 60
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	// Extract only the relevant information of the time
 | 
					 | 
				
			||||||
	// If the time is greater than a year, it makes no sense to display seconds.
 | 
					 | 
				
			||||||
	switch {
 | 
					 | 
				
			||||||
	case years > 0:
 | 
					 | 
				
			||||||
		formattedTime = formatTime(years, "year", formattedTime)
 | 
					 | 
				
			||||||
		formattedTime = formatTime(months, "month", formattedTime)
 | 
					 | 
				
			||||||
	case months > 0:
 | 
					 | 
				
			||||||
		formattedTime = formatTime(months, "month", formattedTime)
 | 
					 | 
				
			||||||
		formattedTime = formatTime(weeks, "week", formattedTime)
 | 
					 | 
				
			||||||
	case weeks > 0:
 | 
					 | 
				
			||||||
		formattedTime = formatTime(weeks, "week", formattedTime)
 | 
					 | 
				
			||||||
		formattedTime = formatTime(days, "day", formattedTime)
 | 
					 | 
				
			||||||
	case days > 0:
 | 
					 | 
				
			||||||
		formattedTime = formatTime(days, "day", formattedTime)
 | 
					 | 
				
			||||||
		formattedTime = formatTime(hours, "hour", formattedTime)
 | 
					 | 
				
			||||||
	case hours > 0:
 | 
					 | 
				
			||||||
		formattedTime = formatTime(hours, "hour", formattedTime)
 | 
					 | 
				
			||||||
		formattedTime = formatTime(minutes, "minute", formattedTime)
 | 
					 | 
				
			||||||
	default:
 | 
					 | 
				
			||||||
		formattedTime = formatTime(minutes, "minute", formattedTime)
 | 
					 | 
				
			||||||
		formattedTime = formatTime(seconds, "second", formattedTime)
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// The formatTime() function always appends a space at the end. This will be trimmed
 | 
						// The formatTime() function always appends a space at the end. This will be trimmed
 | 
				
			||||||
	return strings.TrimRight(formattedTime, " ")
 | 
						return strings.TrimRight(formattedTime, " ")
 | 
				
			||||||
@@ -76,6 +34,5 @@ func formatTime(value int64, name, formattedTime string) string {
 | 
				
			|||||||
	} else if value > 1 {
 | 
						} else if value > 1 {
 | 
				
			||||||
		formattedTime = fmt.Sprintf("%s%d %ss ", formattedTime, value, name)
 | 
							formattedTime = fmt.Sprintf("%s%d %ss ", formattedTime, value, name)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					 | 
				
			||||||
	return formattedTime
 | 
						return formattedTime
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -9,22 +9,17 @@ import (
 | 
				
			|||||||
	"github.com/stretchr/testify/assert"
 | 
						"github.com/stretchr/testify/assert"
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestSecToTime(t *testing.T) {
 | 
					func TestSecToHours(t *testing.T) {
 | 
				
			||||||
	second := int64(1)
 | 
						second := int64(1)
 | 
				
			||||||
	minute := 60 * second
 | 
						minute := 60 * second
 | 
				
			||||||
	hour := 60 * minute
 | 
						hour := 60 * minute
 | 
				
			||||||
	day := 24 * hour
 | 
						day := 24 * hour
 | 
				
			||||||
	year := 365 * day
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	assert.Equal(t, "1 minute 6 seconds", SecToTime(minute+6*second))
 | 
						assert.Equal(t, "1 minute", SecToHours(minute+6*second))
 | 
				
			||||||
	assert.Equal(t, "1 hour", SecToTime(hour))
 | 
						assert.Equal(t, "1 hour", SecToHours(hour))
 | 
				
			||||||
	assert.Equal(t, "1 hour", SecToTime(hour+second))
 | 
						assert.Equal(t, "1 hour", SecToHours(hour+second))
 | 
				
			||||||
	assert.Equal(t, "14 hours 33 minutes", SecToTime(14*hour+33*minute+30*second))
 | 
						assert.Equal(t, "14 hours 33 minutes", SecToHours(14*hour+33*minute+30*second))
 | 
				
			||||||
	assert.Equal(t, "6 days 12 hours", SecToTime(6*day+12*hour+30*minute+18*second))
 | 
						assert.Equal(t, "156 hours 30 minutes", SecToHours(6*day+12*hour+30*minute+18*second))
 | 
				
			||||||
	assert.Equal(t, "2 weeks 4 days", SecToTime((2*7+4)*day+2*hour+16*minute+58*second))
 | 
						assert.Equal(t, "98 hours 16 minutes", SecToHours(4*day+2*hour+16*minute+58*second))
 | 
				
			||||||
	assert.Equal(t, "4 weeks", SecToTime(4*7*day))
 | 
						assert.Equal(t, "672 hours", SecToHours(4*7*day))
 | 
				
			||||||
	assert.Equal(t, "4 weeks 1 day", SecToTime((4*7+1)*day))
 | 
					 | 
				
			||||||
	assert.Equal(t, "1 month 2 weeks", SecToTime((6*7+3)*day+13*hour+38*minute+45*second))
 | 
					 | 
				
			||||||
	assert.Equal(t, "11 months", SecToTime(year-25*day))
 | 
					 | 
				
			||||||
	assert.Equal(t, "1 year 5 months", SecToTime(year+163*day+10*hour+11*minute+5*second))
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -81,7 +81,7 @@ func DeleteTime(c *context.Context) {
 | 
				
			|||||||
		return
 | 
							return
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	c.Flash.Success(c.Tr("repo.issues.del_time_history", util.SecToTime(t.Time)))
 | 
						c.Flash.Success(c.Tr("repo.issues.del_time_history", util.SecToHours(t.Time)))
 | 
				
			||||||
	c.JSONRedirect("")
 | 
						c.JSONRedirect("")
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -16,6 +16,7 @@ import (
 | 
				
			|||||||
	"code.gitea.io/gitea/modules/log"
 | 
						"code.gitea.io/gitea/modules/log"
 | 
				
			||||||
	"code.gitea.io/gitea/modules/setting"
 | 
						"code.gitea.io/gitea/modules/setting"
 | 
				
			||||||
	api "code.gitea.io/gitea/modules/structs"
 | 
						api "code.gitea.io/gitea/modules/structs"
 | 
				
			||||||
 | 
						"code.gitea.io/gitea/modules/util"
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func ToIssue(ctx context.Context, doer *user_model.User, issue *issues_model.Issue) *api.Issue {
 | 
					func ToIssue(ctx context.Context, doer *user_model.User, issue *issues_model.Issue) *api.Issue {
 | 
				
			||||||
@@ -186,7 +187,7 @@ func ToStopWatches(ctx context.Context, sws []*issues_model.Stopwatch) (api.Stop
 | 
				
			|||||||
		result = append(result, api.StopWatch{
 | 
							result = append(result, api.StopWatch{
 | 
				
			||||||
			Created:       sw.CreatedUnix.AsTime(),
 | 
								Created:       sw.CreatedUnix.AsTime(),
 | 
				
			||||||
			Seconds:       sw.Seconds(),
 | 
								Seconds:       sw.Seconds(),
 | 
				
			||||||
			Duration:      sw.Duration(),
 | 
								Duration:      util.SecToHours(sw.Seconds()),
 | 
				
			||||||
			IssueIndex:    issue.Index,
 | 
								IssueIndex:    issue.Index,
 | 
				
			||||||
			IssueTitle:    issue.Title,
 | 
								IssueTitle:    issue.Title,
 | 
				
			||||||
			RepoOwnerName: repo.OwnerName,
 | 
								RepoOwnerName: repo.OwnerName,
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -74,7 +74,7 @@ func ToTimelineComment(ctx context.Context, repo *repo_model.Repository, c *issu
 | 
				
			|||||||
			c.Content[0] == '|' {
 | 
								c.Content[0] == '|' {
 | 
				
			||||||
			// TimeTracking Comments from v1.21 on store the seconds instead of an formatted string
 | 
								// TimeTracking Comments from v1.21 on store the seconds instead of an formatted string
 | 
				
			||||||
			// so we check for the "|" delimiter and convert new to legacy format on demand
 | 
								// so we check for the "|" delimiter and convert new to legacy format on demand
 | 
				
			||||||
			c.Content = util.SecToTime(c.Content[1:])
 | 
								c.Content = util.SecToHours(c.Content[1:])
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		if c.Type == issues_model.CommentTypeChangeTimeEstimate {
 | 
							if c.Type == issues_model.CommentTypeChangeTimeEstimate {
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,6 +1,6 @@
 | 
				
			|||||||
import {createTippy} from '../modules/tippy.ts';
 | 
					import {createTippy} from '../modules/tippy.ts';
 | 
				
			||||||
import {GET} from '../modules/fetch.ts';
 | 
					import {GET} from '../modules/fetch.ts';
 | 
				
			||||||
import {hideElem, showElem} from '../utils/dom.ts';
 | 
					import {hideElem, queryElems, showElem} from '../utils/dom.ts';
 | 
				
			||||||
import {logoutFromWorker} from '../modules/worker.ts';
 | 
					import {logoutFromWorker} from '../modules/worker.ts';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const {appSubUrl, notificationSettings, enableTimeTracking, assetVersionEncoded} = window.config;
 | 
					const {appSubUrl, notificationSettings, enableTimeTracking, assetVersionEncoded} = window.config;
 | 
				
			||||||
@@ -144,23 +144,10 @@ function updateStopwatchData(data) {
 | 
				
			|||||||
  return Boolean(data.length);
 | 
					  return Boolean(data.length);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// TODO: This flickers on page load, we could avoid this by making a custom
 | 
					// TODO: This flickers on page load, we could avoid this by making a custom element to render time periods.
 | 
				
			||||||
// element to render time periods. Feeding a datetime in backend does not work
 | 
					function updateStopwatchTime(seconds: number) {
 | 
				
			||||||
// when time zone between server and client differs.
 | 
					  const hours = seconds / 3600 || 0;
 | 
				
			||||||
function updateStopwatchTime(seconds) {
 | 
					  const minutes = seconds / 60 || 0;
 | 
				
			||||||
  if (!Number.isFinite(seconds)) return;
 | 
					  const timeText = hours >= 1 ? `${Math.round(hours)}h` : `${Math.round(minutes)}m`;
 | 
				
			||||||
  const datetime = (new Date(Date.now() - seconds * 1000)).toISOString();
 | 
					  queryElems(document, '.header-stopwatch-dot', (el) => el.textContent = timeText);
 | 
				
			||||||
  for (const parent of document.querySelectorAll('.header-stopwatch-dot')) {
 | 
					 | 
				
			||||||
    const existing = parent.querySelector(':scope > relative-time');
 | 
					 | 
				
			||||||
    if (existing) {
 | 
					 | 
				
			||||||
      existing.setAttribute('datetime', datetime);
 | 
					 | 
				
			||||||
    } else {
 | 
					 | 
				
			||||||
      const el = document.createElement('relative-time');
 | 
					 | 
				
			||||||
      el.setAttribute('format', 'micro');
 | 
					 | 
				
			||||||
      el.setAttribute('datetime', datetime);
 | 
					 | 
				
			||||||
      el.setAttribute('lang', 'en-US');
 | 
					 | 
				
			||||||
      el.setAttribute('title', ''); // make <relative-time> show no title and therefor no tooltip
 | 
					 | 
				
			||||||
      parent.append(el);
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user