2021-07-26 12:28:57 -04:00
|
|
|
// Copyright 2021 The Hugo Authors. All rights reserved.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
package htime
|
|
|
|
|
|
|
|
import (
|
2023-01-04 12:24:36 -05:00
|
|
|
"log"
|
2021-07-26 12:28:57 -04:00
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
2022-04-26 13:57:04 -04:00
|
|
|
"github.com/bep/clock"
|
2021-07-28 12:02:42 -04:00
|
|
|
"github.com/spf13/cast"
|
|
|
|
|
2021-08-02 10:24:04 -04:00
|
|
|
"github.com/gohugoio/locales"
|
2021-07-26 12:28:57 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
longDayNames = []string{
|
|
|
|
"Sunday",
|
|
|
|
"Monday",
|
|
|
|
"Tuesday",
|
|
|
|
"Wednesday",
|
|
|
|
"Thursday",
|
|
|
|
"Friday",
|
|
|
|
"Saturday",
|
|
|
|
}
|
|
|
|
|
|
|
|
shortDayNames = []string{
|
|
|
|
"Sun",
|
|
|
|
"Mon",
|
|
|
|
"Tue",
|
|
|
|
"Wed",
|
|
|
|
"Thu",
|
|
|
|
"Fri",
|
|
|
|
"Sat",
|
|
|
|
}
|
|
|
|
|
|
|
|
shortMonthNames = []string{
|
|
|
|
"Jan",
|
|
|
|
"Feb",
|
|
|
|
"Mar",
|
|
|
|
"Apr",
|
|
|
|
"May",
|
|
|
|
"Jun",
|
|
|
|
"Jul",
|
|
|
|
"Aug",
|
|
|
|
"Sep",
|
|
|
|
"Oct",
|
|
|
|
"Nov",
|
|
|
|
"Dec",
|
|
|
|
}
|
|
|
|
|
|
|
|
longMonthNames = []string{
|
|
|
|
"January",
|
|
|
|
"February",
|
|
|
|
"March",
|
|
|
|
"April",
|
|
|
|
"May",
|
|
|
|
"June",
|
|
|
|
"July",
|
|
|
|
"August",
|
|
|
|
"September",
|
|
|
|
"October",
|
|
|
|
"November",
|
|
|
|
"December",
|
|
|
|
}
|
2022-05-07 08:10:32 -04:00
|
|
|
|
|
|
|
Clock = clock.System()
|
2021-07-26 12:28:57 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
func NewTimeFormatter(ltr locales.Translator) TimeFormatter {
|
|
|
|
if ltr == nil {
|
|
|
|
panic("must provide a locales.Translator")
|
|
|
|
}
|
|
|
|
return TimeFormatter{
|
|
|
|
ltr: ltr,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TimeFormatter is locale aware.
|
|
|
|
type TimeFormatter struct {
|
|
|
|
ltr locales.Translator
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f TimeFormatter) Format(t time.Time, layout string) string {
|
|
|
|
if layout == "" {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
if layout[0] == ':' {
|
|
|
|
// It may be one of Hugo's custom layouts.
|
|
|
|
switch strings.ToLower(layout[1:]) {
|
|
|
|
case "date_full":
|
|
|
|
return f.ltr.FmtDateFull(t)
|
|
|
|
case "date_long":
|
|
|
|
return f.ltr.FmtDateLong(t)
|
|
|
|
case "date_medium":
|
|
|
|
return f.ltr.FmtDateMedium(t)
|
|
|
|
case "date_short":
|
|
|
|
return f.ltr.FmtDateShort(t)
|
|
|
|
case "time_full":
|
|
|
|
return f.ltr.FmtTimeFull(t)
|
|
|
|
case "time_long":
|
|
|
|
return f.ltr.FmtTimeLong(t)
|
|
|
|
case "time_medium":
|
|
|
|
return f.ltr.FmtTimeMedium(t)
|
|
|
|
case "time_short":
|
|
|
|
return f.ltr.FmtTimeShort(t)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
s := t.Format(layout)
|
|
|
|
|
|
|
|
monthIdx := t.Month() - 1 // Month() starts at 1.
|
|
|
|
dayIdx := t.Weekday()
|
|
|
|
|
|
|
|
s = strings.ReplaceAll(s, longMonthNames[monthIdx], f.ltr.MonthWide(t.Month()))
|
2021-11-01 10:20:57 -04:00
|
|
|
if !strings.Contains(s, f.ltr.MonthWide(t.Month())) {
|
|
|
|
s = strings.ReplaceAll(s, shortMonthNames[monthIdx], f.ltr.MonthAbbreviated(t.Month()))
|
|
|
|
}
|
2021-07-26 12:28:57 -04:00
|
|
|
s = strings.ReplaceAll(s, longDayNames[dayIdx], f.ltr.WeekdayWide(t.Weekday()))
|
2021-11-01 10:20:57 -04:00
|
|
|
if !strings.Contains(s, f.ltr.WeekdayWide(t.Weekday())) {
|
|
|
|
s = strings.ReplaceAll(s, shortDayNames[dayIdx], f.ltr.WeekdayAbbreviated(t.Weekday()))
|
|
|
|
}
|
2021-07-26 12:28:57 -04:00
|
|
|
|
|
|
|
return s
|
|
|
|
}
|
2021-07-28 12:02:42 -04:00
|
|
|
|
2022-03-17 17:03:27 -04:00
|
|
|
func ToTimeInDefaultLocationE(i any, location *time.Location) (tim time.Time, err error) {
|
2021-07-28 12:02:42 -04:00
|
|
|
switch vv := i.(type) {
|
2022-06-06 03:48:40 -04:00
|
|
|
case AsTimeProvider:
|
2021-07-28 12:02:42 -04:00
|
|
|
return vv.AsTime(location), nil
|
2021-10-25 13:49:44 -04:00
|
|
|
// issue #8895
|
|
|
|
// datetimes parsed by `go-toml` have empty zone name
|
|
|
|
// convert back them into string and use `cast`
|
2022-06-06 03:48:40 -04:00
|
|
|
// TODO(bep) add tests, make sure we really need this.
|
2021-10-25 13:49:44 -04:00
|
|
|
case time.Time:
|
|
|
|
i = vv.Format(time.RFC3339)
|
2021-07-28 12:02:42 -04:00
|
|
|
}
|
|
|
|
return cast.ToTimeInDefaultLocationE(i, location)
|
|
|
|
}
|
2022-04-26 13:57:04 -04:00
|
|
|
|
2022-05-07 08:10:32 -04:00
|
|
|
// Now returns time.Now() or time value based on the `clock` flag.
|
2022-04-26 13:57:04 -04:00
|
|
|
// Use this function to fake time inside hugo.
|
|
|
|
func Now() time.Time {
|
|
|
|
return Clock.Now()
|
|
|
|
}
|
|
|
|
|
|
|
|
func Since(t time.Time) time.Duration {
|
2022-05-07 08:10:32 -04:00
|
|
|
return Clock.Since(t)
|
2022-04-26 13:57:04 -04:00
|
|
|
}
|
2022-06-06 03:48:40 -04:00
|
|
|
|
|
|
|
// AsTimeProvider is implemented by go-toml's LocalDate and LocalDateTime.
|
|
|
|
type AsTimeProvider interface {
|
|
|
|
AsTime(zone *time.Location) time.Time
|
|
|
|
}
|
2023-01-04 12:24:36 -05:00
|
|
|
|
|
|
|
// StopWatch is a simple helper to measure time during development.
|
|
|
|
func StopWatch(name string) func() {
|
|
|
|
start := time.Now()
|
|
|
|
return func() {
|
|
|
|
log.Printf("StopWatch %q took %s", name, time.Since(start))
|
|
|
|
}
|
|
|
|
}
|