use core::fmt::{self, Display, Formatter};
#[derive(Clone, Copy, Debug)]
pub struct CharConversionError;
impl Display for CharConversionError {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
write!(f, "{self:?}")
}
}
impl core::error::Error for CharConversionError {}
#[derive(Clone, Copy, Default, Eq, PartialEq, PartialOrd, Ord, Hash)]
#[repr(transparent)]
pub struct Char8(u8);
impl TryFrom<char> for Char8 {
type Error = CharConversionError;
fn try_from(value: char) -> Result<Self, Self::Error> {
let code_point = u32::from(value);
u8::try_from(code_point)
.map(Char8)
.map_err(|_| CharConversionError)
}
}
impl From<Char8> for char {
fn from(char: Char8) -> Self {
Self::from(char.0)
}
}
impl From<u8> for Char8 {
fn from(value: u8) -> Self {
Self(value)
}
}
impl From<Char8> for u8 {
fn from(char: Char8) -> Self {
char.0
}
}
impl fmt::Debug for Char8 {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
<char as fmt::Debug>::fmt(&From::from(self.0), f)
}
}
impl fmt::Display for Char8 {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
<char as fmt::Display>::fmt(&From::from(self.0), f)
}
}
impl PartialEq<char> for Char8 {
fn eq(&self, other: &char) -> bool {
u32::from(self.0) == u32::from(*other)
}
}
pub const NUL_8: Char8 = Char8(0);
#[derive(Clone, Copy, Default, Eq, PartialEq, PartialOrd, Ord, Hash)]
#[repr(transparent)]
pub struct Char16(u16);
impl Char16 {
#[must_use]
pub const unsafe fn from_u16_unchecked(val: u16) -> Self {
Self(val)
}
#[must_use]
pub const fn is_ascii(&self) -> bool {
self.0 <= 127
}
}
impl TryFrom<char> for Char16 {
type Error = CharConversionError;
fn try_from(value: char) -> Result<Self, Self::Error> {
let code_point = u32::from(value);
u16::try_from(code_point)
.map(Char16)
.map_err(|_| CharConversionError)
}
}
impl From<Char16> for char {
fn from(char: Char16) -> Self {
u32::from(char.0).try_into().unwrap()
}
}
impl TryFrom<u16> for Char16 {
type Error = CharConversionError;
fn try_from(value: u16) -> Result<Self, Self::Error> {
let res: Result<char, _> = u32::from(value).try_into();
if let Ok(ch) = res {
ch.try_into()
} else {
Err(CharConversionError)
}
}
}
impl From<Char16> for u16 {
fn from(char: Char16) -> Self {
char.0
}
}
impl fmt::Debug for Char16 {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
if let Ok(c) = u32::from(self.0).try_into() {
<char as fmt::Debug>::fmt(&c, f)
} else {
write!(f, "Char16({:?})", self.0)
}
}
}
impl fmt::Display for Char16 {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
if let Ok(c) = u32::from(self.0).try_into() {
<char as fmt::Display>::fmt(&c, f)
} else {
write!(f, "{}", core::char::REPLACEMENT_CHARACTER)
}
}
}
impl PartialEq<char> for Char16 {
fn eq(&self, other: &char) -> bool {
u32::from(self.0) == u32::from(*other)
}
}
pub const NUL_16: Char16 = unsafe { Char16::from_u16_unchecked(0) };
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_char8_from_char() {
assert_eq!(Char8::try_from('A').unwrap(), Char8(0x41));
}
#[test]
fn test_char16_from_char() {
assert_eq!(Char16::try_from('A').unwrap(), Char16(0x41));
assert_eq!(Char16::try_from('ꋃ').unwrap(), Char16(0xa2c3));
}
#[test]
fn test_char_eq() {
let primitive_char: char = 'A';
assert_eq!(Char8(0x41), primitive_char);
assert_eq!(Char16(0x41), primitive_char);
}
}