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
use icu_locid::{LanguageIdentifier, ParserError};
use language_matcher::LanguageMatcher;
use serde::{Deserialize, Serialize};
use std::{fmt::Display, str::FromStr, sync::LazyLock};
use sys_locale::get_locale;
static MATCHER: LazyLock<LanguageMatcher> = LazyLock::new(LanguageMatcher::new);
#[derive(Debug, Default, Clone, PartialEq, Eq, Hash, Serialize, Deserialize)]
#[repr(transparent)]
pub struct Locale(pub LanguageIdentifier);
impl Locale {
pub fn current() -> Self {
get_locale()
.and_then(|loc| loc.parse().ok())
.unwrap_or_default()
}
pub fn choose_from<'a>(
&self,
locales: impl IntoIterator<Item = impl Into<&'a Locale>>,
) -> Option<&'a Locale> {
MATCHER
.matches(self.0.clone(), locales.into_iter().map(|loc| loc.into()))
.map(|(lang, _)| lang)
}
}
impl Display for Locale {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
self.0.fmt(f)
}
}
impl FromStr for Locale {
type Err = ParserError;
fn from_str(s: &str) -> Result<Self, Self::Err> {
Ok(Self(s.parse()?))
}
}
impl AsRef<LanguageIdentifier> for Locale {
fn as_ref(&self) -> &LanguageIdentifier {
&self.0
}
}
#[doc(hidden)]
pub use icu_locid::langid as __icu_langid;
#[macro_export]
macro_rules! locale {
($langid:literal) => {
$crate::Locale($crate::__icu_langid!($langid))
};
}
#[cfg(test)]
mod test {
use crate::locale;
#[test]
fn parse() {
assert_eq!(locale!("zh-Hans").to_string(), "zh-Hans");
}
#[test]
fn accept() {
let accepts = [
locale!("en"),
locale!("ja"),
locale!("zh-Hans"),
locale!("zh-Hant"),
];
assert_eq!(
locale!("zh-CN").choose_from(&accepts),
Some(&locale!("zh-Hans"))
);
assert_eq!(
locale!("zh-TW").choose_from(&accepts),
Some(&locale!("zh-Hant"))
);
}
}