1 package ko
2
3 import (
4 "math"
5 "strconv"
6 "time"
7
8 "github.com/go-playground/locales"
9 "github.com/go-playground/locales/currency"
10 )
11
12 type ko struct {
13 locale string
14 pluralsCardinal []locales.PluralRule
15 pluralsOrdinal []locales.PluralRule
16 pluralsRange []locales.PluralRule
17 decimal string
18 group string
19 minus string
20 percent string
21 perMille string
22 timeSeparator string
23 inifinity string
24 currencies []string
25 currencyNegativePrefix string
26 currencyNegativeSuffix string
27 monthsAbbreviated []string
28 monthsNarrow []string
29 monthsWide []string
30 daysAbbreviated []string
31 daysNarrow []string
32 daysShort []string
33 daysWide []string
34 periodsAbbreviated []string
35 periodsNarrow []string
36 periodsShort []string
37 periodsWide []string
38 erasAbbreviated []string
39 erasNarrow []string
40 erasWide []string
41 timezones map[string]string
42 }
43
44
45 func New() locales.Translator {
46 return &ko{
47 locale: "ko",
48 pluralsCardinal: []locales.PluralRule{6},
49 pluralsOrdinal: []locales.PluralRule{6},
50 pluralsRange: []locales.PluralRule{6},
51 decimal: ".",
52 group: ",",
53 minus: "-",
54 percent: "%",
55 perMille: "‰",
56 timeSeparator: ":",
57 inifinity: "∞",
58 currencies: []string{"ADP", "AED", "AFA", "AFN", "ALK", "ALL", "AMD", "ANG", "AOA", "AOK", "AON", "AOR", "ARA", "ARL", "ARM", "ARP", "ARS", "ATS", "AU$", "AWG", "AZM", "AZN", "BAD", "BAM", "BAN", "BBD", "BDT", "BEC", "BEF", "BEL", "BGL", "BGM", "BGN", "BGO", "BHD", "BIF", "BMD", "BND", "BOB", "BOL", "BOP", "BOV", "BRB", "BRC", "BRE", "R$", "BRN", "BRR", "BRZ", "BSD", "BTN", "BUK", "BWP", "BYB", "BYN", "BYR", "BZD", "CA$", "CDF", "CHE", "CHF", "CHW", "CLE", "CLF", "CLP", "CNH", "CNX", "CN¥", "COP", "COU", "CRC", "CSD", "CSK", "CUC", "CUP", "CVE", "CYP", "CZK", "DDM", "DEM", "DJF", "DKK", "DOP", "DZD", "ECS", "ECV", "EEK", "EGP", "ERN", "ESA", "ESB", "ESP", "ETB", "€", "FIM", "FJD", "FKP", "FRF", "£", "GEK", "GEL", "GHC", "GHS", "GIP", "GMD", "GNF", "GNS", "GQE", "GRD", "GTQ", "GWE", "GWP", "GYD", "HK$", "HNL", "HRD", "HRK", "HTG", "HUF", "IDR", "IEP", "ILP", "ILR", "₪", "₹", "IQD", "IRR", "ISJ", "ISK", "ITL", "JMD", "JOD", "JP¥", "KES", "KGS", "KHR", "KMF", "KPW", "KRH", "KRO", "₩", "KWD", "KYD", "KZT", "LAK", "LBP", "LKR", "LRD", "LSL", "LTL", "LTT", "LUC", "LUF", "LUL", "LVL", "LVR", "LYD", "MAD", "MAF", "MCF", "MDC", "MDL", "MGA", "MGF", "MKD", "MKN", "MLF", "MMK", "MNT", "MOP", "MRO", "MRU", "MTL", "MTP", "MUR", "MVP", "MVR", "MWK", "MX$", "MXP", "MXV", "MYR", "MZE", "MZM", "MZN", "NAD", "NGN", "NIC", "NIO", "NLG", "NOK", "NPR", "NZ$", "OMR", "PAB", "PEI", "PEN", "PES", "PGK", "PHP", "PKR", "PLN", "PLZ", "PTE", "PYG", "QAR", "RHD", "ROL", "RON", "RSD", "RUB", "RUR", "RWF", "SAR", "SBD", "SCR", "SDD", "SDG", "SDP", "SEK", "SGD", "SHP", "SIT", "SKK", "SLL", "SOS", "SRD", "SRG", "SSP", "STD", "STN", "SUR", "SVC", "SYP", "SZL", "THB", "TJR", "TJS", "TMM", "TMT", "TND", "TOP", "TPE", "TRL", "TRY", "TTD", "NT$", "TZS", "UAH", "UAK", "UGS", "UGX", "US$", "USN", "USS", "UYI", "UYP", "UYU", "UYW", "UZS", "VEB", "VEF", "VES", "₫", "VNN", "VUV", "WST", "FCFA", "XAG", "XAU", "XBA", "XBB", "XBC", "XBD", "EC$", "XDR", "XEU", "XFO", "XFU", "CFA", "XPD", "CFPF", "XPT", "XRE", "XSU", "XTS", "XUA", "XXX", "YDD", "YER", "YUD", "YUM", "YUN", "YUR", "ZAL", "ZAR", "ZMK", "ZMW", "ZRN", "ZRZ", "ZWD", "ZWL", "ZWR"},
59 currencyNegativePrefix: "(",
60 currencyNegativeSuffix: ")",
61 monthsAbbreviated: []string{"", "1월", "2월", "3월", "4월", "5월", "6월", "7월", "8월", "9월", "10월", "11월", "12월"},
62 monthsNarrow: []string{"", "1월", "2월", "3월", "4월", "5월", "6월", "7월", "8월", "9월", "10월", "11월", "12월"},
63 monthsWide: []string{"", "1월", "2월", "3월", "4월", "5월", "6월", "7월", "8월", "9월", "10월", "11월", "12월"},
64 daysAbbreviated: []string{"일", "월", "화", "수", "목", "금", "토"},
65 daysNarrow: []string{"일", "월", "화", "수", "목", "금", "토"},
66 daysShort: []string{"일", "월", "화", "수", "목", "금", "토"},
67 daysWide: []string{"일요일", "월요일", "화요일", "수요일", "목요일", "금요일", "토요일"},
68 periodsAbbreviated: []string{"AM", "PM"},
69 periodsNarrow: []string{"AM", "PM"},
70 periodsWide: []string{"오전", "오후"},
71 erasAbbreviated: []string{"BC", "AD"},
72 erasNarrow: []string{"", ""},
73 erasWide: []string{"기원전", "서기"},
74 timezones: map[string]string{"ACDT": "오스트레일리아 중부 하계 표준시", "ACST": "오스트레일리아 중부 표준시", "ACWDT": "오스트레일리아 중서부 하계 표준시", "ACWST": "오스트레일리아 중서부 표준시", "ADT": "대서양 하계 표준시", "AEDT": "오스트레일리아 동부 하계 표준시", "AEST": "오스트레일리아 동부 표준시", "AKDT": "알래스카 하계 표준시", "AKST": "알래스카 표준시", "ARST": "아르헨티나 하계 표준시", "ART": "아르헨티나 표준시", "AST": "대서양 표준시", "AWDT": "오스트레일리아 서부 하계 표준시", "AWST": "오스트레일리아 서부 표준시", "BOT": "볼리비아 시간", "BT": "부탄 시간", "CAT": "중앙아프리카 시간", "CDT": "미 중부 하계 표준시", "CHADT": "채텀 하계 표준시", "CHAST": "채텀 표준시", "CLST": "칠레 하계 표준시", "CLT": "칠레 표준시", "COST": "콜롬비아 하계 표준시", "COT": "콜롬비아 표준시", "CST": "미 중부 표준시", "ChST": "차모로 시간", "EAT": "동아프리카 시간", "ECT": "에콰도르 시간", "EDT": "미 동부 하계 표준시", "EST": "미 동부 표준시", "GFT": "프랑스령 가이아나 시간", "GMT": "그리니치 표준시", "GST": "걸프만 표준시", "GYT": "가이아나 시간", "HADT": "하와이 알류샨 하계 표준시", "HAST": "하와이 알류샨 표준시", "HAT": "뉴펀들랜드 하계 표준시", "HECU": "쿠바 하계 표준시", "HEEG": "그린란드 동부 하계 표준시", "HENOMX": "멕시코 북서부 하계 표준시", "HEOG": "그린란드 서부 하계 표준시", "HEPM": "세인트피에르 미클롱 하계 표준시", "HEPMX": "멕시코 태평양 하계 표준시", "HKST": "홍콩 하계 표준시", "HKT": "홍콩 표준시", "HNCU": "쿠바 표준시", "HNEG": "그린란드 동부 표준시", "HNNOMX": "멕시코 북서부 표준시", "HNOG": "그린란드 서부 표준시", "HNPM": "세인트피에르 미클롱 표준시", "HNPMX": "멕시코 태평양 표준시", "HNT": "뉴펀들랜드 표준시", "IST": "인도 표준시", "JDT": "일본 하계 표준시", "JST": "일본 표준시", "LHDT": "로드 하우 하계 표준시", "LHST": "로드 하우 표준시", "MDT": "미 산지 하계 표준시", "MESZ": "중부 유럽 하계 표준시", "MEZ": "중부 유럽 표준시", "MST": "미 산악 표준시", "MYT": "말레이시아 시간", "NZDT": "뉴질랜드 하계 표준시", "NZST": "뉴질랜드 표준시", "OESZ": "동유럽 하계 표준시", "OEZ": "동유럽 표준시", "PDT": "미 태평양 하계 표준시", "PST": "미 태평양 표준시", "SAST": "남아프리카 시간", "SGT": "싱가포르 표준시", "SRT": "수리남 시간", "TMST": "투르크메니스탄 하계 표준시", "TMT": "투르크메니스탄 표준시", "UYST": "우루과이 하계 표준시", "UYT": "우루과이 표준시", "VET": "베네수엘라 시간", "WARST": "아르헨티나 서부 하계 표준시", "WART": "아르헨티나 서부 표준시", "WAST": "서아프리카 하계 표준시", "WAT": "서아프리카 표준시", "WESZ": "서유럽 하계 표준시", "WEZ": "서유럽 표준시", "WIB": "서부 인도네시아 시간", "WIT": "동부 인도네시아 시간", "WITA": "중부 인도네시아 시간", "∅∅∅": "브라질리아 하계 표준시"},
75 }
76 }
77
78
79 func (ko *ko) Locale() string {
80 return ko.locale
81 }
82
83
84 func (ko *ko) PluralsCardinal() []locales.PluralRule {
85 return ko.pluralsCardinal
86 }
87
88
89 func (ko *ko) PluralsOrdinal() []locales.PluralRule {
90 return ko.pluralsOrdinal
91 }
92
93
94 func (ko *ko) PluralsRange() []locales.PluralRule {
95 return ko.pluralsRange
96 }
97
98
99 func (ko *ko) CardinalPluralRule(num float64, v uint64) locales.PluralRule {
100 return locales.PluralRuleOther
101 }
102
103
104 func (ko *ko) OrdinalPluralRule(num float64, v uint64) locales.PluralRule {
105 return locales.PluralRuleOther
106 }
107
108
109 func (ko *ko) RangePluralRule(num1 float64, v1 uint64, num2 float64, v2 uint64) locales.PluralRule {
110 return locales.PluralRuleOther
111 }
112
113
114 func (ko *ko) MonthAbbreviated(month time.Month) string {
115 return ko.monthsAbbreviated[month]
116 }
117
118
119 func (ko *ko) MonthsAbbreviated() []string {
120 return ko.monthsAbbreviated[1:]
121 }
122
123
124 func (ko *ko) MonthNarrow(month time.Month) string {
125 return ko.monthsNarrow[month]
126 }
127
128
129 func (ko *ko) MonthsNarrow() []string {
130 return ko.monthsNarrow[1:]
131 }
132
133
134 func (ko *ko) MonthWide(month time.Month) string {
135 return ko.monthsWide[month]
136 }
137
138
139 func (ko *ko) MonthsWide() []string {
140 return ko.monthsWide[1:]
141 }
142
143
144 func (ko *ko) WeekdayAbbreviated(weekday time.Weekday) string {
145 return ko.daysAbbreviated[weekday]
146 }
147
148
149 func (ko *ko) WeekdaysAbbreviated() []string {
150 return ko.daysAbbreviated
151 }
152
153
154 func (ko *ko) WeekdayNarrow(weekday time.Weekday) string {
155 return ko.daysNarrow[weekday]
156 }
157
158
159 func (ko *ko) WeekdaysNarrow() []string {
160 return ko.daysNarrow
161 }
162
163
164 func (ko *ko) WeekdayShort(weekday time.Weekday) string {
165 return ko.daysShort[weekday]
166 }
167
168
169 func (ko *ko) WeekdaysShort() []string {
170 return ko.daysShort
171 }
172
173
174 func (ko *ko) WeekdayWide(weekday time.Weekday) string {
175 return ko.daysWide[weekday]
176 }
177
178
179 func (ko *ko) WeekdaysWide() []string {
180 return ko.daysWide
181 }
182
183
184 func (ko *ko) Decimal() string {
185 return ko.decimal
186 }
187
188
189 func (ko *ko) Group() string {
190 return ko.group
191 }
192
193
194 func (ko *ko) Minus() string {
195 return ko.minus
196 }
197
198
199 func (ko *ko) FmtNumber(num float64, v uint64) string {
200
201 s := strconv.FormatFloat(math.Abs(num), 'f', int(v), 64)
202 l := len(s) + 2 + 1*len(s[:len(s)-int(v)-1])/3
203 count := 0
204 inWhole := v == 0
205 b := make([]byte, 0, l)
206
207 for i := len(s) - 1; i >= 0; i-- {
208
209 if s[i] == '.' {
210 b = append(b, ko.decimal[0])
211 inWhole = true
212 continue
213 }
214
215 if inWhole {
216 if count == 3 {
217 b = append(b, ko.group[0])
218 count = 1
219 } else {
220 count++
221 }
222 }
223
224 b = append(b, s[i])
225 }
226
227 if num < 0 {
228 b = append(b, ko.minus[0])
229 }
230
231
232 for i, j := 0, len(b)-1; i < j; i, j = i+1, j-1 {
233 b[i], b[j] = b[j], b[i]
234 }
235
236 return string(b)
237 }
238
239
240
241 func (ko *ko) FmtPercent(num float64, v uint64) string {
242 s := strconv.FormatFloat(math.Abs(num), 'f', int(v), 64)
243 l := len(s) + 3
244 b := make([]byte, 0, l)
245
246 for i := len(s) - 1; i >= 0; i-- {
247
248 if s[i] == '.' {
249 b = append(b, ko.decimal[0])
250 continue
251 }
252
253 b = append(b, s[i])
254 }
255
256 if num < 0 {
257 b = append(b, ko.minus[0])
258 }
259
260
261 for i, j := 0, len(b)-1; i < j; i, j = i+1, j-1 {
262 b[i], b[j] = b[j], b[i]
263 }
264
265 b = append(b, ko.percent...)
266
267 return string(b)
268 }
269
270
271 func (ko *ko) FmtCurrency(num float64, v uint64, currency currency.Type) string {
272
273 s := strconv.FormatFloat(math.Abs(num), 'f', int(v), 64)
274 symbol := ko.currencies[currency]
275 l := len(s) + len(symbol) + 2 + 1*len(s[:len(s)-int(v)-1])/3
276 count := 0
277 inWhole := v == 0
278 b := make([]byte, 0, l)
279
280 for i := len(s) - 1; i >= 0; i-- {
281
282 if s[i] == '.' {
283 b = append(b, ko.decimal[0])
284 inWhole = true
285 continue
286 }
287
288 if inWhole {
289 if count == 3 {
290 b = append(b, ko.group[0])
291 count = 1
292 } else {
293 count++
294 }
295 }
296
297 b = append(b, s[i])
298 }
299
300 for j := len(symbol) - 1; j >= 0; j-- {
301 b = append(b, symbol[j])
302 }
303
304 if num < 0 {
305 b = append(b, ko.minus[0])
306 }
307
308
309 for i, j := 0, len(b)-1; i < j; i, j = i+1, j-1 {
310 b[i], b[j] = b[j], b[i]
311 }
312
313 if int(v) < 2 {
314
315 if v == 0 {
316 b = append(b, ko.decimal...)
317 }
318
319 for i := 0; i < 2-int(v); i++ {
320 b = append(b, '0')
321 }
322 }
323
324 return string(b)
325 }
326
327
328
329 func (ko *ko) FmtAccounting(num float64, v uint64, currency currency.Type) string {
330
331 s := strconv.FormatFloat(math.Abs(num), 'f', int(v), 64)
332 symbol := ko.currencies[currency]
333 l := len(s) + len(symbol) + 4 + 1*len(s[:len(s)-int(v)-1])/3
334 count := 0
335 inWhole := v == 0
336 b := make([]byte, 0, l)
337
338 for i := len(s) - 1; i >= 0; i-- {
339
340 if s[i] == '.' {
341 b = append(b, ko.decimal[0])
342 inWhole = true
343 continue
344 }
345
346 if inWhole {
347 if count == 3 {
348 b = append(b, ko.group[0])
349 count = 1
350 } else {
351 count++
352 }
353 }
354
355 b = append(b, s[i])
356 }
357
358 if num < 0 {
359
360 for j := len(symbol) - 1; j >= 0; j-- {
361 b = append(b, symbol[j])
362 }
363
364 b = append(b, ko.currencyNegativePrefix[0])
365
366 } else {
367
368 for j := len(symbol) - 1; j >= 0; j-- {
369 b = append(b, symbol[j])
370 }
371
372 }
373
374
375 for i, j := 0, len(b)-1; i < j; i, j = i+1, j-1 {
376 b[i], b[j] = b[j], b[i]
377 }
378
379 if int(v) < 2 {
380
381 if v == 0 {
382 b = append(b, ko.decimal...)
383 }
384
385 for i := 0; i < 2-int(v); i++ {
386 b = append(b, '0')
387 }
388 }
389
390 if num < 0 {
391 b = append(b, ko.currencyNegativeSuffix...)
392 }
393
394 return string(b)
395 }
396
397
398 func (ko *ko) FmtDateShort(t time.Time) string {
399
400 b := make([]byte, 0, 32)
401
402 if t.Year() > 9 {
403 b = append(b, strconv.Itoa(t.Year())[2:]...)
404 } else {
405 b = append(b, strconv.Itoa(t.Year())[1:]...)
406 }
407
408 b = append(b, []byte{0x2e, 0x20}...)
409 b = strconv.AppendInt(b, int64(t.Month()), 10)
410 b = append(b, []byte{0x2e, 0x20}...)
411 b = strconv.AppendInt(b, int64(t.Day()), 10)
412 b = append(b, []byte{0x2e}...)
413
414 return string(b)
415 }
416
417
418 func (ko *ko) FmtDateMedium(t time.Time) string {
419
420 b := make([]byte, 0, 32)
421
422 if t.Year() > 0 {
423 b = strconv.AppendInt(b, int64(t.Year()), 10)
424 } else {
425 b = strconv.AppendInt(b, int64(-t.Year()), 10)
426 }
427
428 b = append(b, []byte{0x2e, 0x20}...)
429 b = strconv.AppendInt(b, int64(t.Month()), 10)
430 b = append(b, []byte{0x2e, 0x20}...)
431 b = strconv.AppendInt(b, int64(t.Day()), 10)
432 b = append(b, []byte{0x2e}...)
433
434 return string(b)
435 }
436
437
438 func (ko *ko) FmtDateLong(t time.Time) string {
439
440 b := make([]byte, 0, 32)
441
442 if t.Year() > 0 {
443 b = strconv.AppendInt(b, int64(t.Year()), 10)
444 } else {
445 b = strconv.AppendInt(b, int64(-t.Year()), 10)
446 }
447
448 b = append(b, []byte{0xeb, 0x85, 0x84, 0x20}...)
449 b = strconv.AppendInt(b, int64(t.Month()), 10)
450 b = append(b, []byte{0xec, 0x9b, 0x94, 0x20}...)
451 b = strconv.AppendInt(b, int64(t.Day()), 10)
452 b = append(b, []byte{0xec, 0x9d, 0xbc}...)
453
454 return string(b)
455 }
456
457
458 func (ko *ko) FmtDateFull(t time.Time) string {
459
460 b := make([]byte, 0, 32)
461
462 if t.Year() > 0 {
463 b = strconv.AppendInt(b, int64(t.Year()), 10)
464 } else {
465 b = strconv.AppendInt(b, int64(-t.Year()), 10)
466 }
467
468 b = append(b, []byte{0xeb, 0x85, 0x84, 0x20}...)
469 b = strconv.AppendInt(b, int64(t.Month()), 10)
470 b = append(b, []byte{0xec, 0x9b, 0x94, 0x20}...)
471 b = strconv.AppendInt(b, int64(t.Day()), 10)
472 b = append(b, []byte{0xec, 0x9d, 0xbc, 0x20}...)
473 b = append(b, ko.daysWide[t.Weekday()]...)
474
475 return string(b)
476 }
477
478
479 func (ko *ko) FmtTimeShort(t time.Time) string {
480
481 b := make([]byte, 0, 32)
482
483 if t.Hour() < 12 {
484 b = append(b, ko.periodsAbbreviated[0]...)
485 } else {
486 b = append(b, ko.periodsAbbreviated[1]...)
487 }
488
489 b = append(b, []byte{0x20}...)
490
491 h := t.Hour()
492
493 if h > 12 {
494 h -= 12
495 }
496
497 b = strconv.AppendInt(b, int64(h), 10)
498 b = append(b, ko.timeSeparator...)
499
500 if t.Minute() < 10 {
501 b = append(b, '0')
502 }
503
504 b = strconv.AppendInt(b, int64(t.Minute()), 10)
505
506 return string(b)
507 }
508
509
510 func (ko *ko) FmtTimeMedium(t time.Time) string {
511
512 b := make([]byte, 0, 32)
513
514 if t.Hour() < 12 {
515 b = append(b, ko.periodsAbbreviated[0]...)
516 } else {
517 b = append(b, ko.periodsAbbreviated[1]...)
518 }
519
520 b = append(b, []byte{0x20}...)
521
522 h := t.Hour()
523
524 if h > 12 {
525 h -= 12
526 }
527
528 b = strconv.AppendInt(b, int64(h), 10)
529 b = append(b, ko.timeSeparator...)
530
531 if t.Minute() < 10 {
532 b = append(b, '0')
533 }
534
535 b = strconv.AppendInt(b, int64(t.Minute()), 10)
536 b = append(b, ko.timeSeparator...)
537
538 if t.Second() < 10 {
539 b = append(b, '0')
540 }
541
542 b = strconv.AppendInt(b, int64(t.Second()), 10)
543
544 return string(b)
545 }
546
547
548 func (ko *ko) FmtTimeLong(t time.Time) string {
549
550 b := make([]byte, 0, 32)
551
552 if t.Hour() < 12 {
553 b = append(b, ko.periodsAbbreviated[0]...)
554 } else {
555 b = append(b, ko.periodsAbbreviated[1]...)
556 }
557
558 b = append(b, []byte{0x20}...)
559
560 h := t.Hour()
561
562 if h > 12 {
563 h -= 12
564 }
565
566 b = strconv.AppendInt(b, int64(h), 10)
567 b = append(b, []byte{0xec, 0x8b, 0x9c, 0x20}...)
568 b = strconv.AppendInt(b, int64(t.Minute()), 10)
569 b = append(b, []byte{0xeb, 0xb6, 0x84, 0x20}...)
570 b = strconv.AppendInt(b, int64(t.Second()), 10)
571 b = append(b, []byte{0xec, 0xb4, 0x88, 0x20}...)
572
573 tz, _ := t.Zone()
574 b = append(b, tz...)
575
576 return string(b)
577 }
578
579
580 func (ko *ko) FmtTimeFull(t time.Time) string {
581
582 b := make([]byte, 0, 32)
583
584 if t.Hour() < 12 {
585 b = append(b, ko.periodsAbbreviated[0]...)
586 } else {
587 b = append(b, ko.periodsAbbreviated[1]...)
588 }
589
590 b = append(b, []byte{0x20}...)
591
592 h := t.Hour()
593
594 if h > 12 {
595 h -= 12
596 }
597
598 b = strconv.AppendInt(b, int64(h), 10)
599 b = append(b, []byte{0xec, 0x8b, 0x9c, 0x20}...)
600 b = strconv.AppendInt(b, int64(t.Minute()), 10)
601 b = append(b, []byte{0xeb, 0xb6, 0x84, 0x20}...)
602 b = strconv.AppendInt(b, int64(t.Second()), 10)
603 b = append(b, []byte{0xec, 0xb4, 0x88, 0x20}...)
604
605 tz, _ := t.Zone()
606
607 if btz, ok := ko.timezones[tz]; ok {
608 b = append(b, btz...)
609 } else {
610 b = append(b, tz...)
611 }
612
613 return string(b)
614 }
615
View as plain text