宽窄优行-由【嘉易行】项目成品而来
younger_times
2023-04-06 a1ae6802080a22e6e6ce6d0935e95facb1daca5c
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
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
//
//  SwiftDate
//  Parse, validate, manipulate, and display dates, time and timezones in Swift
//
//  Created by Daniele Margutti
//   - Web: https://www.danielemargutti.com
//   - Twitter: https://twitter.com/danielemargutti
//   - Mail: hello@danielemargutti.com
//
//  Copyright © 2019 Daniele Margutti. Licensed under MIT License.
//
 
import Foundation
 
internal class RelativeFormatterLanguagesCache {
 
    static let shared = RelativeFormatterLanguagesCache()
 
    private(set) var cachedValues = [String: [String: Any]]()
 
    func flavoursForLocaleID(_ langID: String) -> [String: Any]? {
        do {
            guard let fullURL = Bundle(for: RelativeFormatter.self).resourceURL?.appendingPathComponent("langs/\(langID).json") else {
                return nil
            }
            let data = try Data(contentsOf: fullURL)
            let json = try JSONSerialization.jsonObject(with: data, options: JSONSerialization.ReadingOptions.allowFragments)
 
            if let value = json as? [String: Any] {
                cachedValues[langID] = value
                return value
            } else {
                return nil
            }
 
        } catch {
            debugPrint("Failed to read data for language id: \(langID)")
            return nil
        }
    }
}
 
public enum RelativeFormatterLanguage: String, CaseIterable {
    case af = "af" // Locales.afrikaans
    case am = "am" // Locales.amharic
    case ar_AE = "ar_AE" // Locales.arabicUnitedArabEmirates
    case ar = "ar" // Locales.arabic
    case `as` = "as" // Locales.assamese
    case az = "az" // Locales.assamese
    case be = "be" // Locales.belarusian
    case bg = "bg" // Locales.bulgarian
    case bn = "bn" // Locales.bengali
    case br = "br" // Locales.breton
    case bs = "bs" // Locales.bosnian
    case bs_Cyrl = "bs-Cyrl" // Locales.belarusian
    case ca = "ca" // Locales.catalan
    case cz = "cz" // Locales.czech
    case cy = "cy" // Locales.welsh
    case cs = "cs" // Locales.czech
    case da = "da" // Locales.danish
    case de = "de" // Locales.dutch
    case dsb = "dsb" // Locales.lowerSorbian
    case dz = "dz" // Locales.dzongkha
    case ee = "ee" // Locales.ewe
    case el = "el" // Locales.greek
    case en = "en" // Locales.english
    case es_AR = "es_AR" // Locales.spanishArgentina
    case es_PY = "es_PY" // Locales.spanishParaguay
    case es_MX = "es_MX" // Locales.spanishMexico
    case es_US = "es_US" // Locales.spanishUnitedStates
    case es = "es" // Locales.spanish
    case et = "et" // Locales.estonian
    case eu = "eu" // Locales.basque
    case fa = "fa" // Locales.persian
    case fi = "fi" // Locales.finnish
    case fil = "fil" // Locales.filipino
    case fo = "fo" // Locales.faroese
    case fr_CA = "fr_CA" // French (Canada)
    case fr = "fr" // French
    case fur = "fur" // Friulian
    case fy = "fy" // Western Frisian
    case ga = "ga" // Irish
    case gd = "gd" // Scottish Gaelic
    case gl = "gl" // Galician
    case gu = "gu" // Gujarati
    case he = "he" // Hebrew
    case hi = "hi" // Hindi
    case hr = "hr" // Croatian
    case hsb = "hsb" // Upper Sorbian
    case hu = "hu" // Hungarian
    case hy = "hy" // Armenian
    case id = "id" // Indonesian
    case `is` = "is" // Icelandic
    case it = "it" // Locales.italian
    case ja = "ja" // Japanese
    case jgo = "jgo" // Ngomba
    case ka = "ka" // Georgian
    case kea = "kea" // Kabuverdianu
    case kk = "kk" // Kazakh
    case kl = "kl" // Kalaallisut
    case km = "km" // Khmer
    case kn = "kn" // Kannada
    case ko = "ko" // Korean
    case kok = "kok" // Konkani
    case ksh = "ksh" // Colognian
    case ky = "ky" // Kyrgyz
    case lb = "lb" // Luxembourgish
    case lkt = "lkt" // Lakota
    case lo = "lo" // Lao
    case lt = "lt" // Lithuanian
    case lv = "lv" // Latvian
    case mk = "mk" // Macedonian
    case ml = "ml" // Malayalam
    case mn = "mn" // Mongolian
    case mr = "mr" // Marathi
    case ms = "ms" // Malay
    case mt = "mt" // Maltese
    case my = "my" // Burmese
    case mzn = "mzn" // Mazanderani
    case nb = "nb" // Norwegian Bokmål
    case ne = "ne" // Nepali
    case nl = "nl" // Netherland
    case nn = "nn" // Norwegian Nynorsk
    case or = "or" // Odia
    case pa = "pa" // Punjabi
    case pl = "pl" // Polish
    case ps = "ps" // Pashto
    case pt = "pt" // Portuguese
    case ro = "ro" // Romanian
    case ru = "ru" // Russian
    case sah = "sah" // Sakha
    case sd = "sd" // Sindhi
    case se_FI = "se_FI" // Northern Sami (Finland)
    case se = "se" // Northern Sami
    case si = "si" // Sinhala
    case sk = "sk" // Slovak
    case sl = "sl" // Slovenian
    case sq = "sq" // Albanian
    case sr_Latn = "sr_Latn" // Serbian (Latin)
    case sr = "sr" // Serbian
    case sv = "sv" // Swedish
    case sw = "sw" // Swedish
    case ta = "ta" // Tamil
    case te = "te" // Telugu
    case th = "th" // Thai
    case ti = "ti" // Tigrinya
    case tk = "tk" // Turkmen
    case to = "to" // Tongan
    case tr = "tr" // Turkish
    case ug = "ug" // Uyghur
    case uk = "uk" // Ukrainian
    case ur_IN = "ur_IN" // Urdu (India)
    case ur = "ur" // Urdu
    case uz_Cyrl = "uz_Cyrl" // Uzbek (Cyrillic)
    case uz = "uz" // Uzbek (Cyrillic)
    case vi = "vi" // Vietnamese
    case wae = "wae" // Walser
    case yue_Hans = "yue_Hans" // Cantonese (Simplified)
    case yue_Hant = "yue_Hant" // Cantonese (Traditional)
    case zh_Hans_HK = "zh_Hans_HK" // Chinese (Simplified, Hong Kong [China])
    case zh_Hans_MO = "zh_Hans_MO" // Chinese (Simplified, Macau [China])
    case zh_Hans_SG = "zh_Hans_SG" // Chinese (Simplified, Singapore)
    case zh_Hant_HK = "zh_Hant_HK" // Chinese (Traditional, Hong Kong [China])
    case zh_Hant_MO = "zh_Hant_MO" // Chinese (Traditional, Macau [China])
    case zh_Hans = "zh_Hans" // Chinese (Simplified)
    case zh_Hant = "zh_Hant" // Chinese (Traditional)
    case zh = "zh" // Chinese
    case zu = "zu" // Zulu
 
    /// Table with the data of the language.
    /// Data is structured in:
    /// { flavour: { unit : { data } } }
    public var flavours: [String: Any] {
        return RelativeFormatterLanguagesCache.shared.flavoursForLocaleID(self.rawValue) ?? [:]
    }
 
    public var identifier: String {
        return self.rawValue
    }
 
    public func quantifyKey(forValue value: Double) -> RelativeFormatter.PluralForm? {
        switch self {
 
        case .sr_Latn, .sr, .uk:
            let mod10 = Int(value) % 10
            let mod100 = Int(value) % 100
 
            switch mod10 {
            case 1:
                switch mod100 {
                case 11:
                    break
                default:
                    return .one
                }
            case 2, 3, 4:
                switch mod100 {
                case 12, 13, 14:
                    break
                default:
                    return .few
                }
            default:
                break
            }
 
            return .many
 
        case .ru, .sk, .sl:
            let mod10 = Int(value) % 10
            let mod100 = Int(value) % 100
 
            switch mod100 {
            case 11...14:
                break
 
            default:
                switch mod10 {
                case 1:
                    return .one
                case 2...4:
                    return .few
                default:
                    break
                }
 
            }
            return .many
 
        case .ro:
            let mod100 = Int(value) % 100
 
            switch value {
            case 0:
                return .few
            case 1:
                return .one
            default:
                if mod100 > 1 && mod100 <= 19 {
                    return .few
                }
            }
 
            return .other
 
        case .pa:
            switch value {
            case 0, 1:
                return .one
            default:
                return .other
            }
 
        case .mt:
            switch value {
            case 1: return .one
            case 0: return .few
            case 2...10: return .few
            case 11...19: return .many
            default: return .other
            }
 
        case .lt, .lv:
            let mod10 = Int(value) % 10
            let mod100 = Int(value) % 100
 
            if value == 0 {
                return .zero
            }
 
            if value == 1 {
                return .one
            }
 
            switch mod10 {
            case 1:
                if mod100 != 11 {
                    return .one
                }
                return .many
            default:
                return .many
            }
 
        case .ksh, .se:
            switch value {
            case 0: return .zero
            case 1: return .one
            default: return .other
            }
 
        case .`is`:
            let mod10 = Int(value) % 10
            let mod100 = Int(value) % 100
 
            if value == 0 {
                return .zero
            }
 
            if value == 1 {
                return .one
            }
 
            switch mod10 {
            case 1:
                if mod100 != 11 {
                    return .one
                }
            default:
                break
            }
 
            return .many
 
        case .id, .ja, .ms, .my, .mzn, .sah, .se_FI, .si, .th, .yue_Hans, .yue_Hant,
             .zh_Hans_HK, .zh_Hans_MO, .zh_Hans_SG, .zh_Hant_HK, .zh_Hant_MO, .zh:
 
            return .other
 
        case .hy:
            return (value >= 0 && value < 2 ? .one : .other)
 
        case .ga, .gd:
            switch Int(value) {
            case 1: return .one
            case 2: return .two
            case 3...6: return .few
            case 7...10: return .many
            default: return .other
            }
 
        case .fr_CA, .fr:
            return (value >= 0 && value < 2 ? .one : .other)
 
        case .dz, .kea, .ko, .kok, .lkt, .lo:
            return nil
 
        case .cs: // Locales.czech
            switch value {
            case 1:
                return .one
            case 2, 3, 4:
                return .few
            default:
                return .other
            }
 
        case .cy:
            switch value {
            case 0:    return .zero
            case 1: return .one
            case 2: return .two
            case 3: return .few
            case 6: return .many
            default: return .other
            }
 
        case .cz, .dsb:
            switch value {
            case 1:
                return .one
            case 2, 3, 4:
                return .few
            default:
                return .other
            }
 
        case .br:
            let n = Int(value)
            return n % 10 == 1 && n % 100 != 11 && n % 100 != 71 && n % 100 != 91 ? .zero : n % 10 == 2 && n % 100 != 12 && n % 100 != 72 && n % 100 != 92 ? .one : (n % 10 == 3 || n % 10 == 4 || n % 10 == 9) && n % 100 != 13 && n % 100 != 14 && n % 100 != 19 && n % 100 != 73 && n % 100 != 74 && n % 100 != 79 && n % 100 != 93 && n % 100 != 94 && n % 100 != 99 ? .two : n % 1_000_000 == 0 && n != 0 ? .many : .other
 
        case .be, .bs, .bs_Cyrl, .hr, .hsb, .pl:
            let mod10 = Int(value) % 10
            let mod100 = Int(value) % 100
 
            switch mod10 {
            case 1:
                switch mod100 {
                case 11:
                    break
                default:
                    return .one
                }
            case 2, 3, 4:
                switch mod100 {
                case 12, 13, 14:
                    break
                default:
                    return .few
                }
            default:
                break
            }
            return .many
 
        case .ar, .ar_AE, .he:
            switch value {
            case 0: return .zero
            case 1: return .one
            case 2: return .two
            default:
                let mod100 = Int(value) % 100
                if mod100 >= 3 && mod100 <= 10 {
                    return .few
                } else if mod100 >= 11 {
                    return .many
                } else {
                    return .other
                }
            }
 
        case .am, .bn, .fa, .gu, .kn, .mr, .zu:
            return (value >= 0 && value <= 1 ? .one : .other)
 
        default:
            return (value == 1 ? .one : .other)
 
        }
    }
 
}