Make time diff translatable (#2057)
This commit is contained in:
		
							parent
							
								
									9d8fba62b3
								
							
						
					
					
						commit
						32fc44aa83
					
				| @ -47,8 +47,8 @@ func SendTestMail(email string) error { | ||||
| func SendUserMail(c *macaron.Context, u *User, tpl base.TplName, code, subject, info string) { | ||||
| 	data := map[string]interface{}{ | ||||
| 		"Username":          u.DisplayName(), | ||||
| 		"ActiveCodeLives":   base.MinutesToFriendly(setting.Service.ActiveCodeLives), | ||||
| 		"ResetPwdCodeLives": base.MinutesToFriendly(setting.Service.ResetPwdCodeLives), | ||||
| 		"ActiveCodeLives":   base.MinutesToFriendly(setting.Service.ActiveCodeLives, c.Locale.Language()), | ||||
| 		"ResetPwdCodeLives": base.MinutesToFriendly(setting.Service.ResetPwdCodeLives, c.Locale.Language()), | ||||
| 		"Code":              code, | ||||
| 	} | ||||
| 
 | ||||
| @ -79,7 +79,7 @@ func SendResetPasswordMail(c *macaron.Context, u *User) { | ||||
| func SendActivateEmailMail(c *macaron.Context, u *User, email *EmailAddress) { | ||||
| 	data := map[string]interface{}{ | ||||
| 		"Username":        u.DisplayName(), | ||||
| 		"ActiveCodeLives": base.MinutesToFriendly(setting.Service.ActiveCodeLives), | ||||
| 		"ActiveCodeLives": base.MinutesToFriendly(setting.Service.ActiveCodeLives, c.Locale.Language()), | ||||
| 		"Code":            u.GenerateEmailActivateCode(email.Email), | ||||
| 		"Email":           email.Email, | ||||
| 	} | ||||
|  | ||||
| @ -219,59 +219,59 @@ const ( | ||||
| 	Year   = 12 * Month | ||||
| ) | ||||
| 
 | ||||
| func computeTimeDiff(diff int64) (int64, string) { | ||||
| func computeTimeDiff(diff int64, lang string) (int64, string) { | ||||
| 	diffStr := "" | ||||
| 	switch { | ||||
| 	case diff <= 0: | ||||
| 		diff = 0 | ||||
| 		diffStr = "now" | ||||
| 		diffStr = i18n.Tr(lang, "tool.now") | ||||
| 	case diff < 2: | ||||
| 		diff = 0 | ||||
| 		diffStr = "1 second" | ||||
| 		diffStr = i18n.Tr(lang, "tool.1s") | ||||
| 	case diff < 1*Minute: | ||||
| 		diffStr = fmt.Sprintf("%d seconds", diff) | ||||
| 		diffStr = i18n.Tr(lang, "tool.seconds", diff) | ||||
| 		diff = 0 | ||||
| 
 | ||||
| 	case diff < 2*Minute: | ||||
| 		diff -= 1 * Minute | ||||
| 		diffStr = "1 minute" | ||||
| 		diffStr = i18n.Tr(lang, "tool.1m") | ||||
| 	case diff < 1*Hour: | ||||
| 		diffStr = fmt.Sprintf("%d minutes", diff/Minute) | ||||
| 		diffStr = i18n.Tr(lang, "tool.minutes", diff/Minute) | ||||
| 		diff -= diff / Minute * Minute | ||||
| 
 | ||||
| 	case diff < 2*Hour: | ||||
| 		diff -= 1 * Hour | ||||
| 		diffStr = "1 hour" | ||||
| 		diffStr = i18n.Tr(lang, "tool.1h") | ||||
| 	case diff < 1*Day: | ||||
| 		diffStr = fmt.Sprintf("%d hours", diff/Hour) | ||||
| 		diffStr = i18n.Tr(lang, "tool.hours", diff/Hour) | ||||
| 		diff -= diff / Hour * Hour | ||||
| 
 | ||||
| 	case diff < 2*Day: | ||||
| 		diff -= 1 * Day | ||||
| 		diffStr = "1 day" | ||||
| 		diffStr = i18n.Tr(lang, "tool.1d") | ||||
| 	case diff < 1*Week: | ||||
| 		diffStr = fmt.Sprintf("%d days", diff/Day) | ||||
| 		diffStr = i18n.Tr(lang, "tool.days", diff/Day) | ||||
| 		diff -= diff / Day * Day | ||||
| 
 | ||||
| 	case diff < 2*Week: | ||||
| 		diff -= 1 * Week | ||||
| 		diffStr = "1 week" | ||||
| 		diffStr = i18n.Tr(lang, "tool.1w") | ||||
| 	case diff < 1*Month: | ||||
| 		diffStr = fmt.Sprintf("%d weeks", diff/Week) | ||||
| 		diffStr = i18n.Tr(lang, "tool.weeks", diff/Week) | ||||
| 		diff -= diff / Week * Week | ||||
| 
 | ||||
| 	case diff < 2*Month: | ||||
| 		diff -= 1 * Month | ||||
| 		diffStr = "1 month" | ||||
| 		diffStr = i18n.Tr(lang, "tool.1mon") | ||||
| 	case diff < 1*Year: | ||||
| 		diffStr = fmt.Sprintf("%d months", diff/Month) | ||||
| 		diffStr = i18n.Tr(lang, "tool.months", diff/Month) | ||||
| 		diff -= diff / Month * Month | ||||
| 
 | ||||
| 	case diff < 2*Year: | ||||
| 		diff -= 1 * Year | ||||
| 		diffStr = "1 year" | ||||
| 		diffStr = i18n.Tr(lang, "tool.1y") | ||||
| 	default: | ||||
| 		diffStr = fmt.Sprintf("%d years", diff/Year) | ||||
| 		diffStr = i18n.Tr(lang, "tool.years", diff/Year) | ||||
| 		diff -= (diff / Year) * Year | ||||
| 	} | ||||
| 	return diff, diffStr | ||||
| @ -279,24 +279,24 @@ func computeTimeDiff(diff int64) (int64, string) { | ||||
| 
 | ||||
| // MinutesToFriendly returns a user friendly string with number of minutes
 | ||||
| // converted to hours and minutes.
 | ||||
| func MinutesToFriendly(minutes int) string { | ||||
| func MinutesToFriendly(minutes int, lang string) string { | ||||
| 	duration := time.Duration(minutes) * time.Minute | ||||
| 	return TimeSincePro(time.Now().Add(-duration)) | ||||
| 	return TimeSincePro(time.Now().Add(-duration), lang) | ||||
| } | ||||
| 
 | ||||
| // TimeSincePro calculates the time interval and generate full user-friendly string.
 | ||||
| func TimeSincePro(then time.Time) string { | ||||
| 	return timeSincePro(then, time.Now()) | ||||
| func TimeSincePro(then time.Time, lang string) string { | ||||
| 	return timeSincePro(then, time.Now(), lang) | ||||
| } | ||||
| 
 | ||||
| func timeSincePro(then, now time.Time) string { | ||||
| func timeSincePro(then, now time.Time, lang string) string { | ||||
| 	diff := now.Unix() - then.Unix() | ||||
| 
 | ||||
| 	if then.After(now) { | ||||
| 		return "future" | ||||
| 		return i18n.Tr(lang, "tool.future") | ||||
| 	} | ||||
| 	if diff == 0 { | ||||
| 		return "now" | ||||
| 		return i18n.Tr(lang, "tool.now") | ||||
| 	} | ||||
| 
 | ||||
| 	var timeStr, diffStr string | ||||
| @ -305,58 +305,25 @@ func timeSincePro(then, now time.Time) string { | ||||
| 			break | ||||
| 		} | ||||
| 
 | ||||
| 		diff, diffStr = computeTimeDiff(diff) | ||||
| 		diff, diffStr = computeTimeDiff(diff, lang) | ||||
| 		timeStr += ", " + diffStr | ||||
| 	} | ||||
| 	return strings.TrimPrefix(timeStr, ", ") | ||||
| } | ||||
| 
 | ||||
| func timeSince(then, now time.Time, lang string) string { | ||||
| 	lbl := i18n.Tr(lang, "tool.ago") | ||||
| 	lbl := "tool.ago" | ||||
| 	diff := now.Unix() - then.Unix() | ||||
| 	if then.After(now) { | ||||
| 		lbl = i18n.Tr(lang, "tool.from_now") | ||||
| 		lbl = "tool.from_now" | ||||
| 		diff = then.Unix() - now.Unix() | ||||
| 	} | ||||
| 
 | ||||
| 	switch { | ||||
| 	case diff <= 0: | ||||
| 	if diff <= 0 { | ||||
| 		return i18n.Tr(lang, "tool.now") | ||||
| 	case diff <= 1: | ||||
| 		return i18n.Tr(lang, "tool.1s", lbl) | ||||
| 	case diff < 1*Minute: | ||||
| 		return i18n.Tr(lang, "tool.seconds", diff, lbl) | ||||
| 
 | ||||
| 	case diff < 2*Minute: | ||||
| 		return i18n.Tr(lang, "tool.1m", lbl) | ||||
| 	case diff < 1*Hour: | ||||
| 		return i18n.Tr(lang, "tool.minutes", diff/Minute, lbl) | ||||
| 
 | ||||
| 	case diff < 2*Hour: | ||||
| 		return i18n.Tr(lang, "tool.1h", lbl) | ||||
| 	case diff < 1*Day: | ||||
| 		return i18n.Tr(lang, "tool.hours", diff/Hour, lbl) | ||||
| 
 | ||||
| 	case diff < 2*Day: | ||||
| 		return i18n.Tr(lang, "tool.1d", lbl) | ||||
| 	case diff < 1*Week: | ||||
| 		return i18n.Tr(lang, "tool.days", diff/Day, lbl) | ||||
| 
 | ||||
| 	case diff < 2*Week: | ||||
| 		return i18n.Tr(lang, "tool.1w", lbl) | ||||
| 	case diff < 1*Month: | ||||
| 		return i18n.Tr(lang, "tool.weeks", diff/Week, lbl) | ||||
| 
 | ||||
| 	case diff < 2*Month: | ||||
| 		return i18n.Tr(lang, "tool.1mon", lbl) | ||||
| 	case diff < 1*Year: | ||||
| 		return i18n.Tr(lang, "tool.months", diff/Month, lbl) | ||||
| 
 | ||||
| 	case diff < 2*Year: | ||||
| 		return i18n.Tr(lang, "tool.1y", lbl) | ||||
| 	default: | ||||
| 		return i18n.Tr(lang, "tool.years", diff/Year, lbl) | ||||
| 	} | ||||
| 
 | ||||
| 	_, diffStr := computeTimeDiff(diff, lang) | ||||
| 	return i18n.Tr(lang, lbl, diffStr) | ||||
| } | ||||
| 
 | ||||
| // RawTimeSince retrieves i18n key of time since t
 | ||||
|  | ||||
| @ -145,7 +145,7 @@ func TestComputeTimeDiff(t *testing.T) { | ||||
| 	// computeTimeDiff(base + offset) == (offset, str)
 | ||||
| 	test := func(base int64, str string, offsets ...int64) { | ||||
| 		for _, offset := range offsets { | ||||
| 			diff, diffStr := computeTimeDiff(base + offset) | ||||
| 			diff, diffStr := computeTimeDiff(base+offset, "en") | ||||
| 			assert.Equal(t, offset, diff) | ||||
| 			assert.Equal(t, str, diffStr) | ||||
| 		} | ||||
| @ -170,7 +170,7 @@ func TestComputeTimeDiff(t *testing.T) { | ||||
| func TestMinutesToFriendly(t *testing.T) { | ||||
| 	// test that a number of minutes yields the expected string
 | ||||
| 	test := func(expected string, minutes int) { | ||||
| 		actual := MinutesToFriendly(minutes) | ||||
| 		actual := MinutesToFriendly(minutes, "en") | ||||
| 		assert.Equal(t, expected, actual) | ||||
| 	} | ||||
| 	test("1 minute", 1) | ||||
| @ -186,13 +186,11 @@ func TestTimeSince(t *testing.T) { | ||||
| 
 | ||||
| 	// test that each diff in `diffs` yields the expected string
 | ||||
| 	test := func(expected string, diffs ...time.Duration) { | ||||
| 		ago := i18n.Tr("en", "tool.ago") | ||||
| 		fromNow := i18n.Tr("en", "tool.from_now") | ||||
| 		for _, diff := range diffs { | ||||
| 			actual := timeSince(BaseDate, BaseDate.Add(diff), "en") | ||||
| 			assert.Equal(t, expected+" "+ago, actual) | ||||
| 			assert.Equal(t, i18n.Tr("en", "tool.ago", expected), actual) | ||||
| 			actual = timeSince(BaseDate.Add(diff), BaseDate, "en") | ||||
| 			assert.Equal(t, expected+" "+fromNow, actual) | ||||
| 			assert.Equal(t, i18n.Tr("en", "tool.from_now", expected), actual) | ||||
| 		} | ||||
| 	} | ||||
| 	test("1 second", time.Second, time.Second+50*time.Millisecond) | ||||
| @ -212,13 +210,13 @@ func TestTimeSince(t *testing.T) { | ||||
| } | ||||
| 
 | ||||
| func TestTimeSincePro(t *testing.T) { | ||||
| 	assert.Equal(t, "now", timeSincePro(BaseDate, BaseDate)) | ||||
| 	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)) | ||||
| 		actual := timeSincePro(BaseDate, BaseDate.Add(diff), "en") | ||||
| 		assert.Equal(t, expected, actual) | ||||
| 		assert.Equal(t, "future", timeSincePro(BaseDate.Add(diff), BaseDate)) | ||||
| 		assert.Equal(t, "future", timeSincePro(BaseDate.Add(diff), BaseDate, "en")) | ||||
| 	} | ||||
| 	test("1 second", time.Second) | ||||
| 	test("2 seconds", 2*time.Second) | ||||
|  | ||||
| @ -1385,23 +1385,24 @@ push_tag = pushed tag <a href="%s/src/%s">%[2]s</a> to <a href="%[1]s">%[3]s</a> | ||||
| compare_commits = Compare %d commits | ||||
| 
 | ||||
| [tool] | ||||
| ago = ago | ||||
| from_now = from now | ||||
| ago = %s ago | ||||
| from_now = %s from now | ||||
| now = now | ||||
| 1s = 1 second %s | ||||
| 1m = 1 minute %s | ||||
| 1h = 1 hour %s | ||||
| 1d = 1 day %s | ||||
| 1w = 1 week %s | ||||
| 1mon = 1 month %s | ||||
| 1y = 1 year %s | ||||
| seconds = %d seconds %s | ||||
| minutes = %d minutes %s | ||||
| hours = %d hours %s | ||||
| days = %d days %s | ||||
| weeks = %d weeks %s | ||||
| months = %d months %s | ||||
| years = %d years %s | ||||
| future = future | ||||
| 1s = 1 second | ||||
| 1m = 1 minute | ||||
| 1h = 1 hour | ||||
| 1d = 1 day | ||||
| 1w = 1 week | ||||
| 1mon = 1 month | ||||
| 1y = 1 year | ||||
| seconds = %d seconds | ||||
| minutes = %d minutes | ||||
| hours = %d hours | ||||
| days = %d days | ||||
| weeks = %d weeks | ||||
| months = %d months | ||||
| years = %d years | ||||
| raw_seconds = seconds | ||||
| raw_minutes = minutes | ||||
| 
 | ||||
|  | ||||
| @ -73,7 +73,7 @@ var sysStatus struct { | ||||
| } | ||||
| 
 | ||||
| func updateSystemStatus() { | ||||
| 	sysStatus.Uptime = base.TimeSincePro(startTime) | ||||
| 	sysStatus.Uptime = base.TimeSincePro(startTime, "en") | ||||
| 
 | ||||
| 	m := new(runtime.MemStats) | ||||
| 	runtime.ReadMemStats(m) | ||||
|  | ||||
| @ -18,8 +18,8 @@ func TemplatePreview(ctx *context.Context) { | ||||
| 	ctx.Data["AppVer"] = setting.AppVer | ||||
| 	ctx.Data["AppUrl"] = setting.AppURL | ||||
| 	ctx.Data["Code"] = "2014031910370000009fff6782aadb2162b4a997acb69d4400888e0b9274657374" | ||||
| 	ctx.Data["ActiveCodeLives"] = base.MinutesToFriendly(setting.Service.ActiveCodeLives) | ||||
| 	ctx.Data["ResetPwdCodeLives"] = base.MinutesToFriendly(setting.Service.ResetPwdCodeLives) | ||||
| 	ctx.Data["ActiveCodeLives"] = base.MinutesToFriendly(setting.Service.ActiveCodeLives, ctx.Locale.Language()) | ||||
| 	ctx.Data["ResetPwdCodeLives"] = base.MinutesToFriendly(setting.Service.ResetPwdCodeLives, ctx.Locale.Language()) | ||||
| 	ctx.Data["CurDbValue"] = "" | ||||
| 
 | ||||
| 	ctx.HTML(200, base.TplName(ctx.Params("*"))) | ||||
|  | ||||
| @ -677,7 +677,7 @@ func LinkAccountPostRegister(ctx *context.Context, cpt *captcha.Captcha, form au | ||||
| 		models.SendActivateAccountMail(ctx.Context, u) | ||||
| 		ctx.Data["IsSendRegisterMail"] = true | ||||
| 		ctx.Data["Email"] = u.Email | ||||
| 		ctx.Data["ActiveCodeLives"] = base.MinutesToFriendly(setting.Service.ActiveCodeLives) | ||||
| 		ctx.Data["ActiveCodeLives"] = base.MinutesToFriendly(setting.Service.ActiveCodeLives, ctx.Locale.Language()) | ||||
| 		ctx.HTML(200, TplActivate) | ||||
| 
 | ||||
| 		if err := ctx.Cache.Put("MailResendLimit_"+u.LowerName, u.LowerName, 180); err != nil { | ||||
| @ -792,7 +792,7 @@ func SignUpPost(ctx *context.Context, cpt *captcha.Captcha, form auth.RegisterFo | ||||
| 		models.SendActivateAccountMail(ctx.Context, u) | ||||
| 		ctx.Data["IsSendRegisterMail"] = true | ||||
| 		ctx.Data["Email"] = u.Email | ||||
| 		ctx.Data["ActiveCodeLives"] = base.MinutesToFriendly(setting.Service.ActiveCodeLives) | ||||
| 		ctx.Data["ActiveCodeLives"] = base.MinutesToFriendly(setting.Service.ActiveCodeLives, ctx.Locale.Language()) | ||||
| 		ctx.HTML(200, TplActivate) | ||||
| 
 | ||||
| 		if err := ctx.Cache.Put("MailResendLimit_"+u.LowerName, u.LowerName, 180); err != nil { | ||||
| @ -818,7 +818,7 @@ func Activate(ctx *context.Context) { | ||||
| 			if ctx.Cache.IsExist("MailResendLimit_" + ctx.User.LowerName) { | ||||
| 				ctx.Data["ResendLimited"] = true | ||||
| 			} else { | ||||
| 				ctx.Data["ActiveCodeLives"] = base.MinutesToFriendly(setting.Service.ActiveCodeLives) | ||||
| 				ctx.Data["ActiveCodeLives"] = base.MinutesToFriendly(setting.Service.ActiveCodeLives, ctx.Locale.Language()) | ||||
| 				models.SendActivateAccountMail(ctx.Context, ctx.User) | ||||
| 
 | ||||
| 				if err := ctx.Cache.Put("MailResendLimit_"+ctx.User.LowerName, ctx.User.LowerName, 180); err != nil { | ||||
| @ -913,7 +913,7 @@ func ForgotPasswdPost(ctx *context.Context) { | ||||
| 	u, err := models.GetUserByEmail(email) | ||||
| 	if err != nil { | ||||
| 		if models.IsErrUserNotExist(err) { | ||||
| 			ctx.Data["ResetPwdCodeLives"] = base.MinutesToFriendly(setting.Service.ResetPwdCodeLives) | ||||
| 			ctx.Data["ResetPwdCodeLives"] = base.MinutesToFriendly(setting.Service.ResetPwdCodeLives, ctx.Locale.Language()) | ||||
| 			ctx.Data["IsResetSent"] = true | ||||
| 			ctx.HTML(200, tplForgotPassword) | ||||
| 			return | ||||
| @ -940,7 +940,7 @@ func ForgotPasswdPost(ctx *context.Context) { | ||||
| 		log.Error(4, "Set cache(MailResendLimit) fail: %v", err) | ||||
| 	} | ||||
| 
 | ||||
| 	ctx.Data["ResetPwdCodeLives"] = base.MinutesToFriendly(setting.Service.ResetPwdCodeLives) | ||||
| 	ctx.Data["ResetPwdCodeLives"] = base.MinutesToFriendly(setting.Service.ResetPwdCodeLives, ctx.Locale.Language()) | ||||
| 	ctx.Data["IsResetSent"] = true | ||||
| 	ctx.HTML(200, tplForgotPassword) | ||||
| } | ||||
|  | ||||
| @ -415,7 +415,7 @@ func RegisterOpenIDPost(ctx *context.Context, cpt *captcha.Captcha, form auth.Si | ||||
| 		models.SendActivateAccountMail(ctx.Context, u) | ||||
| 		ctx.Data["IsSendRegisterMail"] = true | ||||
| 		ctx.Data["Email"] = u.Email | ||||
| 		ctx.Data["ActiveCodeLives"] = base.MinutesToFriendly(setting.Service.ActiveCodeLives) | ||||
| 		ctx.Data["ActiveCodeLives"] = base.MinutesToFriendly(setting.Service.ActiveCodeLives, ctx.Locale.Language()) | ||||
| 		ctx.HTML(200, TplActivate) | ||||
| 
 | ||||
| 		if err := ctx.Cache.Put("MailResendLimit_"+u.LowerName, u.LowerName, 180); err != nil { | ||||
|  | ||||
| @ -298,7 +298,7 @@ func SettingsEmailPost(ctx *context.Context, form auth.AddEmailForm) { | ||||
| 		if err := ctx.Cache.Put("MailResendLimit_"+ctx.User.LowerName, ctx.User.LowerName, 180); err != nil { | ||||
| 			log.Error(4, "Set cache(MailResendLimit) fail: %v", err) | ||||
| 		} | ||||
| 		ctx.Flash.Info(ctx.Tr("settings.add_email_confirmation_sent", email.Email, base.MinutesToFriendly(setting.Service.ActiveCodeLives))) | ||||
| 		ctx.Flash.Info(ctx.Tr("settings.add_email_confirmation_sent", email.Email, base.MinutesToFriendly(setting.Service.ActiveCodeLives, ctx.Locale.Language()))) | ||||
| 	} else { | ||||
| 		ctx.Flash.Success(ctx.Tr("settings.add_email_success")) | ||||
| 	} | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user