mirror of
				https://github.com/go-gitea/gitea
				synced 2025-10-26 08:58:24 +00:00 
			
		
		
		
	Display ui time with customize time location (#7792)
* display ui time with customize time location * fix lint * rename UILocation to DefaultUILocation * move time related functions to modules/timeutil * fix tests * fix tests * fix build * fix swagger
This commit is contained in:
		| @@ -12,7 +12,6 @@ import ( | ||||
| 	"encoding/base64" | ||||
| 	"encoding/hex" | ||||
| 	"fmt" | ||||
| 	"html/template" | ||||
| 	"io" | ||||
| 	"math" | ||||
| 	"net/http" | ||||
| @@ -29,10 +28,8 @@ import ( | ||||
| 	"code.gitea.io/gitea/modules/git" | ||||
| 	"code.gitea.io/gitea/modules/log" | ||||
| 	"code.gitea.io/gitea/modules/setting" | ||||
| 	"code.gitea.io/gitea/modules/util" | ||||
|  | ||||
| 	"github.com/Unknwon/com" | ||||
| 	"github.com/Unknwon/i18n" | ||||
| ) | ||||
|  | ||||
| // EncodeMD5 encodes string to md5 hex value. | ||||
| @@ -217,154 +214,6 @@ func AvatarLink(email string) string { | ||||
| 	return SizedAvatarLink(email, DefaultAvatarSize) | ||||
| } | ||||
|  | ||||
| // Seconds-based time units | ||||
| const ( | ||||
| 	Minute = 60 | ||||
| 	Hour   = 60 * Minute | ||||
| 	Day    = 24 * Hour | ||||
| 	Week   = 7 * Day | ||||
| 	Month  = 30 * Day | ||||
| 	Year   = 12 * Month | ||||
| ) | ||||
|  | ||||
| func computeTimeDiff(diff int64, lang string) (int64, string) { | ||||
| 	diffStr := "" | ||||
| 	switch { | ||||
| 	case diff <= 0: | ||||
| 		diff = 0 | ||||
| 		diffStr = i18n.Tr(lang, "tool.now") | ||||
| 	case diff < 2: | ||||
| 		diff = 0 | ||||
| 		diffStr = i18n.Tr(lang, "tool.1s") | ||||
| 	case diff < 1*Minute: | ||||
| 		diffStr = i18n.Tr(lang, "tool.seconds", diff) | ||||
| 		diff = 0 | ||||
|  | ||||
| 	case diff < 2*Minute: | ||||
| 		diff -= 1 * Minute | ||||
| 		diffStr = i18n.Tr(lang, "tool.1m") | ||||
| 	case diff < 1*Hour: | ||||
| 		diffStr = i18n.Tr(lang, "tool.minutes", diff/Minute) | ||||
| 		diff -= diff / Minute * Minute | ||||
|  | ||||
| 	case diff < 2*Hour: | ||||
| 		diff -= 1 * Hour | ||||
| 		diffStr = i18n.Tr(lang, "tool.1h") | ||||
| 	case diff < 1*Day: | ||||
| 		diffStr = i18n.Tr(lang, "tool.hours", diff/Hour) | ||||
| 		diff -= diff / Hour * Hour | ||||
|  | ||||
| 	case diff < 2*Day: | ||||
| 		diff -= 1 * Day | ||||
| 		diffStr = i18n.Tr(lang, "tool.1d") | ||||
| 	case diff < 1*Week: | ||||
| 		diffStr = i18n.Tr(lang, "tool.days", diff/Day) | ||||
| 		diff -= diff / Day * Day | ||||
|  | ||||
| 	case diff < 2*Week: | ||||
| 		diff -= 1 * Week | ||||
| 		diffStr = i18n.Tr(lang, "tool.1w") | ||||
| 	case diff < 1*Month: | ||||
| 		diffStr = i18n.Tr(lang, "tool.weeks", diff/Week) | ||||
| 		diff -= diff / Week * Week | ||||
|  | ||||
| 	case diff < 2*Month: | ||||
| 		diff -= 1 * Month | ||||
| 		diffStr = i18n.Tr(lang, "tool.1mon") | ||||
| 	case diff < 1*Year: | ||||
| 		diffStr = i18n.Tr(lang, "tool.months", diff/Month) | ||||
| 		diff -= diff / Month * Month | ||||
|  | ||||
| 	case diff < 2*Year: | ||||
| 		diff -= 1 * Year | ||||
| 		diffStr = i18n.Tr(lang, "tool.1y") | ||||
| 	default: | ||||
| 		diffStr = i18n.Tr(lang, "tool.years", diff/Year) | ||||
| 		diff -= (diff / Year) * Year | ||||
| 	} | ||||
| 	return diff, diffStr | ||||
| } | ||||
|  | ||||
| // MinutesToFriendly returns a user friendly string with number of minutes | ||||
| // converted to hours and minutes. | ||||
| func MinutesToFriendly(minutes int, lang string) string { | ||||
| 	duration := time.Duration(minutes) * time.Minute | ||||
| 	return TimeSincePro(time.Now().Add(-duration), lang) | ||||
| } | ||||
|  | ||||
| // TimeSincePro calculates the time interval and generate full user-friendly string. | ||||
| func TimeSincePro(then time.Time, lang string) string { | ||||
| 	return timeSincePro(then, time.Now(), lang) | ||||
| } | ||||
|  | ||||
| func timeSincePro(then, now time.Time, lang string) string { | ||||
| 	diff := now.Unix() - then.Unix() | ||||
|  | ||||
| 	if then.After(now) { | ||||
| 		return i18n.Tr(lang, "tool.future") | ||||
| 	} | ||||
| 	if diff == 0 { | ||||
| 		return i18n.Tr(lang, "tool.now") | ||||
| 	} | ||||
|  | ||||
| 	var timeStr, diffStr string | ||||
| 	for { | ||||
| 		if diff == 0 { | ||||
| 			break | ||||
| 		} | ||||
|  | ||||
| 		diff, diffStr = computeTimeDiff(diff, lang) | ||||
| 		timeStr += ", " + diffStr | ||||
| 	} | ||||
| 	return strings.TrimPrefix(timeStr, ", ") | ||||
| } | ||||
|  | ||||
| func timeSince(then, now time.Time, lang string) string { | ||||
| 	return timeSinceUnix(then.Unix(), now.Unix(), lang) | ||||
| } | ||||
|  | ||||
| func timeSinceUnix(then, now int64, lang string) string { | ||||
| 	lbl := "tool.ago" | ||||
| 	diff := now - then | ||||
| 	if then > now { | ||||
| 		lbl = "tool.from_now" | ||||
| 		diff = then - now | ||||
| 	} | ||||
| 	if diff <= 0 { | ||||
| 		return i18n.Tr(lang, "tool.now") | ||||
| 	} | ||||
|  | ||||
| 	_, diffStr := computeTimeDiff(diff, lang) | ||||
| 	return i18n.Tr(lang, lbl, diffStr) | ||||
| } | ||||
|  | ||||
| // RawTimeSince retrieves i18n key of time since t | ||||
| func RawTimeSince(t time.Time, lang string) string { | ||||
| 	return timeSince(t, time.Now(), lang) | ||||
| } | ||||
|  | ||||
| // TimeSince calculates the time interval and generate user-friendly string. | ||||
| func TimeSince(then time.Time, lang string) template.HTML { | ||||
| 	return htmlTimeSince(then, time.Now(), lang) | ||||
| } | ||||
|  | ||||
| func htmlTimeSince(then, now time.Time, lang string) template.HTML { | ||||
| 	return template.HTML(fmt.Sprintf(`<span class="time-since" title="%s">%s</span>`, | ||||
| 		then.Format(setting.TimeFormat), | ||||
| 		timeSince(then, now, lang))) | ||||
| } | ||||
|  | ||||
| // TimeSinceUnix calculates the time interval and generate user-friendly string. | ||||
| func TimeSinceUnix(then util.TimeStamp, lang string) template.HTML { | ||||
| 	return htmlTimeSinceUnix(then, util.TimeStamp(time.Now().Unix()), lang) | ||||
| } | ||||
|  | ||||
| func htmlTimeSinceUnix(then, now util.TimeStamp, lang string) template.HTML { | ||||
| 	return template.HTML(fmt.Sprintf(`<span class="time-since" title="%s">%s</span>`, | ||||
| 		then.Format(setting.TimeFormat), | ||||
| 		timeSinceUnix(int64(then), int64(now), lang))) | ||||
| } | ||||
|  | ||||
| // Storage space size types | ||||
| const ( | ||||
| 	Byte  = 1 | ||||
|   | ||||
| @@ -2,43 +2,13 @@ package base | ||||
|  | ||||
| import ( | ||||
| 	"net/url" | ||||
| 	"os" | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"code.gitea.io/gitea/modules/setting" | ||||
|  | ||||
| 	"github.com/Unknwon/i18n" | ||||
| 	macaroni18n "github.com/go-macaron/i18n" | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| ) | ||||
|  | ||||
| var BaseDate time.Time | ||||
|  | ||||
| // time durations | ||||
| const ( | ||||
| 	DayDur   = 24 * time.Hour | ||||
| 	WeekDur  = 7 * DayDur | ||||
| 	MonthDur = 30 * DayDur | ||||
| 	YearDur  = 12 * MonthDur | ||||
| ) | ||||
|  | ||||
| func TestMain(m *testing.M) { | ||||
| 	// setup | ||||
| 	macaroni18n.I18n(macaroni18n.Options{ | ||||
| 		Directory:   "../../options/locale/", | ||||
| 		DefaultLang: "en-US", | ||||
| 		Langs:       []string{"en-US"}, | ||||
| 		Names:       []string{"english"}, | ||||
| 	}) | ||||
| 	BaseDate = time.Date(2000, time.January, 1, 0, 0, 0, 0, time.UTC) | ||||
|  | ||||
| 	// run the tests | ||||
| 	retVal := m.Run() | ||||
|  | ||||
| 	os.Exit(retVal) | ||||
| } | ||||
|  | ||||
| func TestEncodeMD5(t *testing.T) { | ||||
| 	assert.Equal(t, | ||||
| 		"3858f62230ac3c915f300c664312c63f", | ||||
| @@ -131,125 +101,6 @@ func TestAvatarLink(t *testing.T) { | ||||
| 	) | ||||
| } | ||||
|  | ||||
| func TestComputeTimeDiff(t *testing.T) { | ||||
| 	// test that for each offset in offsets, | ||||
| 	// computeTimeDiff(base + offset) == (offset, str) | ||||
| 	test := func(base int64, str string, offsets ...int64) { | ||||
| 		for _, offset := range offsets { | ||||
| 			diff, diffStr := computeTimeDiff(base+offset, "en") | ||||
| 			assert.Equal(t, offset, diff) | ||||
| 			assert.Equal(t, str, diffStr) | ||||
| 		} | ||||
| 	} | ||||
| 	test(0, "now", 0) | ||||
| 	test(1, "1 second", 0) | ||||
| 	test(2, "2 seconds", 0) | ||||
| 	test(Minute, "1 minute", 0, 1, 30, Minute-1) | ||||
| 	test(2*Minute, "2 minutes", 0, Minute-1) | ||||
| 	test(Hour, "1 hour", 0, 1, Hour-1) | ||||
| 	test(5*Hour, "5 hours", 0, Hour-1) | ||||
| 	test(Day, "1 day", 0, 1, Day-1) | ||||
| 	test(5*Day, "5 days", 0, Day-1) | ||||
| 	test(Week, "1 week", 0, 1, Week-1) | ||||
| 	test(3*Week, "3 weeks", 0, 4*Day+25000) | ||||
| 	test(Month, "1 month", 0, 1, Month-1) | ||||
| 	test(10*Month, "10 months", 0, Month-1) | ||||
| 	test(Year, "1 year", 0, Year-1) | ||||
| 	test(3*Year, "3 years", 0, Year-1) | ||||
| } | ||||
|  | ||||
| func TestMinutesToFriendly(t *testing.T) { | ||||
| 	// test that a number of minutes yields the expected string | ||||
| 	test := func(expected string, minutes int) { | ||||
| 		actual := MinutesToFriendly(minutes, "en") | ||||
| 		assert.Equal(t, expected, actual) | ||||
| 	} | ||||
| 	test("1 minute", 1) | ||||
| 	test("2 minutes", 2) | ||||
| 	test("1 hour", 60) | ||||
| 	test("1 hour, 1 minute", 61) | ||||
| 	test("1 hour, 2 minutes", 62) | ||||
| 	test("2 hours", 120) | ||||
| } | ||||
|  | ||||
| func TestTimeSince(t *testing.T) { | ||||
| 	assert.Equal(t, "now", timeSince(BaseDate, BaseDate, "en")) | ||||
|  | ||||
| 	// test that each diff in `diffs` yields the expected string | ||||
| 	test := func(expected string, diffs ...time.Duration) { | ||||
| 		for _, diff := range diffs { | ||||
| 			actual := timeSince(BaseDate, BaseDate.Add(diff), "en") | ||||
| 			assert.Equal(t, i18n.Tr("en", "tool.ago", expected), actual) | ||||
| 			actual = timeSince(BaseDate.Add(diff), BaseDate, "en") | ||||
| 			assert.Equal(t, i18n.Tr("en", "tool.from_now", expected), actual) | ||||
| 		} | ||||
| 	} | ||||
| 	test("1 second", time.Second, time.Second+50*time.Millisecond) | ||||
| 	test("2 seconds", 2*time.Second, 2*time.Second+50*time.Millisecond) | ||||
| 	test("1 minute", time.Minute, time.Minute+30*time.Second) | ||||
| 	test("2 minutes", 2*time.Minute, 2*time.Minute+30*time.Second) | ||||
| 	test("1 hour", time.Hour, time.Hour+30*time.Minute) | ||||
| 	test("2 hours", 2*time.Hour, 2*time.Hour+30*time.Minute) | ||||
| 	test("1 day", DayDur, DayDur+12*time.Hour) | ||||
| 	test("2 days", 2*DayDur, 2*DayDur+12*time.Hour) | ||||
| 	test("1 week", WeekDur, WeekDur+3*DayDur) | ||||
| 	test("2 weeks", 2*WeekDur, 2*WeekDur+3*DayDur) | ||||
| 	test("1 month", MonthDur, MonthDur+15*DayDur) | ||||
| 	test("2 months", 2*MonthDur, 2*MonthDur+15*DayDur) | ||||
| 	test("1 year", YearDur, YearDur+6*MonthDur) | ||||
| 	test("2 years", 2*YearDur, 2*YearDur+6*MonthDur) | ||||
| } | ||||
|  | ||||
| func TestTimeSincePro(t *testing.T) { | ||||
| 	assert.Equal(t, "now", timeSincePro(BaseDate, BaseDate, "en")) | ||||
|  | ||||
| 	// test that a difference of `diff` yields the expected string | ||||
| 	test := func(expected string, diff time.Duration) { | ||||
| 		actual := timeSincePro(BaseDate, BaseDate.Add(diff), "en") | ||||
| 		assert.Equal(t, expected, actual) | ||||
| 		assert.Equal(t, "future", timeSincePro(BaseDate.Add(diff), BaseDate, "en")) | ||||
| 	} | ||||
| 	test("1 second", time.Second) | ||||
| 	test("2 seconds", 2*time.Second) | ||||
| 	test("1 minute", time.Minute) | ||||
| 	test("1 minute, 1 second", time.Minute+time.Second) | ||||
| 	test("1 minute, 59 seconds", time.Minute+59*time.Second) | ||||
| 	test("2 minutes", 2*time.Minute) | ||||
| 	test("1 hour", time.Hour) | ||||
| 	test("1 hour, 1 second", time.Hour+time.Second) | ||||
| 	test("1 hour, 59 minutes, 59 seconds", time.Hour+59*time.Minute+59*time.Second) | ||||
| 	test("2 hours", 2*time.Hour) | ||||
| 	test("1 day", DayDur) | ||||
| 	test("1 day, 23 hours, 59 minutes, 59 seconds", | ||||
| 		DayDur+23*time.Hour+59*time.Minute+59*time.Second) | ||||
| 	test("2 days", 2*DayDur) | ||||
| 	test("1 week", WeekDur) | ||||
| 	test("2 weeks", 2*WeekDur) | ||||
| 	test("1 month", MonthDur) | ||||
| 	test("3 months", 3*MonthDur) | ||||
| 	test("1 year", YearDur) | ||||
| 	test("2 years, 3 months, 1 week, 2 days, 4 hours, 12 minutes, 17 seconds", | ||||
| 		2*YearDur+3*MonthDur+WeekDur+2*DayDur+4*time.Hour+ | ||||
| 			12*time.Minute+17*time.Second) | ||||
| } | ||||
|  | ||||
| func TestHtmlTimeSince(t *testing.T) { | ||||
| 	setting.TimeFormat = time.UnixDate | ||||
| 	// test that `diff` yields a result containing `expected` | ||||
| 	test := func(expected string, diff time.Duration) { | ||||
| 		actual := htmlTimeSince(BaseDate, BaseDate.Add(diff), "en") | ||||
| 		assert.Contains(t, actual, `title="Sat Jan  1 00:00:00 UTC 2000"`) | ||||
| 		assert.Contains(t, actual, expected) | ||||
| 	} | ||||
| 	test("1 second", time.Second) | ||||
| 	test("3 minutes", 3*time.Minute+5*time.Second) | ||||
| 	test("1 day", DayDur+18*time.Hour) | ||||
| 	test("1 week", WeekDur+6*DayDur) | ||||
| 	test("3 months", 3*MonthDur+3*WeekDur) | ||||
| 	test("2 years", 2*YearDur) | ||||
| 	test("3 years", 3*YearDur+11*MonthDur+4*WeekDur) | ||||
| } | ||||
|  | ||||
| func TestFileSize(t *testing.T) { | ||||
| 	var size int64 = 512 | ||||
| 	assert.Equal(t, "512B", FileSize(size)) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user