-
Notifications
You must be signed in to change notification settings - Fork 24
/
Copy pathformat.go
162 lines (151 loc) · 5.76 KB
/
format.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
// Copyright 2015 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package date
import (
"fmt"
"io"
"strings"
)
// These are predefined layouts for use in [Date.Format] and [Parse].
// The reference date used in the layouts is the same date used by the
// time package in the standard library:
//
// Monday, Jan 2, 2006
//
// To define your own format, write down what the reference date would look
// like formatted your way; see the values of the predefined layouts for
// examples. The model is to demonstrate what the reference date looks like
// so that the [Parse] function and [Date.Format] method can apply the same
// transformation to a general date value.
const (
ISO8601 = "2006-01-02" // ISO 8601 extended format
ISO8601B = "20060102" // ISO 8601 basic format
RFC822 = "02-Jan-06"
RFC822W = "Mon, 02-Jan-06" // RFC822 with day of the week
RFC850 = "Monday, 02-Jan-06"
RFC1123 = "02 Jan 2006"
RFC1123W = "Mon, 02 Jan 2006" // RFC1123 with day of the week
RFC3339 = "2006-01-02"
)
// String returns the time formatted in ISO 8601 extended format
// (e.g. "2006-01-02"). If the year of the date falls outside the
// [0,9999] range, this format produces an expanded year representation
// with possibly extra year digits beyond the prescribed four-digit minimum
// and with a + or - sign prefix (e.g. , "+12345-06-07", "-0987-06-05").
func (d Date) String() string {
buf := &strings.Builder{}
buf.Grow(12)
d.WriteTo(buf)
return buf.String()
}
// WriteTo is as per String, albeit writing to an io.Writer.
func (d Date) WriteTo(w io.Writer) (n64 int64, err error) {
var n int
year, month, day := d.Date()
if 0 <= year && year < 10000 {
n, err = fmt.Fprintf(w, "%04d-%02d-%02d", year, month, day)
} else {
n, err = fmt.Fprintf(w, "%+05d-%02d-%02d", year, month, day)
}
return int64(n), err
}
// FormatOrdinal returns a textual representation of the date value formatted
// according to the ordinal date variant of the ISO 8601 format.
// The year of the date is represented as a signed integer. The three-digit
// ordinal day number is appended, e.g. "2024-107".
func (d Date) FormatOrdinal() string {
t := decode(d)
year := t.Year()
ordinal := d.YearDay()
return fmt.Sprintf("%04d-%03d", year, ordinal)
}
// FormatISO returns a textual representation of the date value formatted
// according to the expanded year variant of the ISO 8601 extended format;
// the year of the date is represented as a signed integer using the
// specified number of digits (ignored if less than four).
// The string representation of the year will take more than the specified
// number of digits if the magnitude of the year is too large to fit.
//
// [Date.Format] can be used to format Date values in other formats,
// but it is currently not able to format dates according to the expanded
// year variant of the ISO 8601 format.
func (d Date) FormatISO(yearDigits int) string {
n := 5 // four-digit minimum plus sign
if yearDigits > 4 {
n += yearDigits - 4
}
year, month, day := d.Date()
return fmt.Sprintf("%+0*d-%02d-%02d", n, year, month, day)
}
// Format returns a textual representation of the date value formatted according
// to layout, which defines the format by showing how the reference date,
// defined to be
//
// Mon, Jan 2, 2006
//
// would be displayed if it were the value; it serves as an example of the
// desired output.
//
// This function actually uses [time.Time.Format] to format the input and can use any
// layout accepted by [time.Time.Format] by extending its date to a time at
// 00:00:00.000 UTC.
//
// Additionally, it is able to insert the day-number suffix into the output string.
// This is done by including "nd" in the format string, which will become
//
// Mon, Jan 2nd, 2006
//
// For example, New Year's Day might be rendered as "Fri, Jan 1st, 2016". To alter
// the suffix strings for a different locale, change [DaySuffixes] or use
// [Date.FormatWithSuffixes] instead.
//
// This function cannot currently format Date values according to the expanded
// year variant of ISO 8601; you should use [Date.FormatISO] to do that.
func (d Date) Format(layout string) string {
return d.FormatWithSuffixes(layout, DaySuffixes)
}
// FormatWithSuffixes is the same as Format, except the suffix strings can be specified
// explicitly, which allows multiple locales to be supported. The suffixes slice should
// contain 31 strings covering the days 1 (index 0) to 31 (index 30).
func (d Date) FormatWithSuffixes(layout string, suffixes []string) string {
t := decode(d)
parts := strings.Split(layout, "nd")
switch len(parts) {
case 1:
return t.Format(layout)
default:
// If the format contains "Monday", it has been split so repair it.
i := 1
for i < len(parts) {
if i > 0 && strings.HasSuffix(parts[i-1], "Mo") && strings.HasPrefix(parts[i], "ay") {
parts[i-1] = parts[i-1] + "nd" + parts[i]
copy(parts[i:], parts[i+1:])
parts = parts[:len(parts)-1]
} else {
i++
}
}
a := make([]string, 0, 2*len(parts)-1)
for i, p := range parts {
if i > 0 {
a = append(a, suffixes[d.Day()-1])
}
a = append(a, t.Format(p))
}
return strings.Join(a, "")
}
}
// DaySuffixes is the default array of strings used as suffixes when a format string
// contains "nd" (as in "second"). This can be altered at startup in order to change
// the default locale strings used for formatting dates. It supports every locale that
// uses the Gregorian calendar and has a suffix after the day-of-month number.
var DaySuffixes = []string{
"st", "nd", "rd", "th", "th", // 1 - 5
"th", "th", "th", "th", "th", // 6 - 10
"th", "th", "th", "th", "th", // 11 - 15
"th", "th", "th", "th", "th", // 16 - 20
"st", "nd", "rd", "th", "th", // 21 - 25
"th", "th", "th", "th", "th", // 26 - 30
"st", // 31
}