object-rainbow 0.0.0-a.67

distributed object model
Documentation
use typenum::{B0, B1, U1};

use crate::{enumkind::UsizeTag, *};

impl ToOutput for char {
    fn to_output(&self, output: &mut impl Output) {
        if output.is_real() {
            let mut buf = [0; 4];
            self.encode_utf8(&mut buf).to_output(output);
        }
    }
}

impl<I: ParseInput> Parse<I> for char {
    fn parse(input: I) -> crate::Result<Self> {
        Self::parse_as_inline(input)
    }
}

impl<I: ParseInput> ParseInline<I> for char {
    fn parse_inline(input: &mut I) -> crate::Result<Self> {
        let c0 = input.parse_inline::<u8>()?;
        if c0 & 0b11000000 != 0b11000000 {
            let c = str::from_utf8(&[c0])
                .map_err(crate::Error::parse)?
                .chars()
                .next()
                .expect("this string is not empty");
            return Ok(c);
        }
        let c1 = input.parse_inline::<u8>()?;
        if c0 & 0b00100000 == 0 {
            let c = str::from_utf8(&[c0, c1])
                .map_err(crate::Error::parse)?
                .chars()
                .next()
                .expect("this string is not empty");
            return Ok(c);
        }
        let c2 = input.parse_inline::<u8>()?;
        if c0 & 0b00010000 == 0 {
            let c = str::from_utf8(&[c0, c1, c2])
                .map_err(crate::Error::parse)?
                .chars()
                .next()
                .expect("this string is not empty");
            return Ok(c);
        }
        let c3 = input.parse_inline::<u8>()?;
        let c = str::from_utf8(&[c0, c1, c2, c3])
            .map_err(crate::Error::parse)?
            .chars()
            .next()
            .expect("this string is not empty");
        Ok(c)
    }
}

impl InlineOutput for char {}
impl Tagged for char {}
impl ListHashes for char {}
impl Topological for char {}

pub struct CharNiche;
pub struct CharNicheNext;

impl Niche for CharNiche {
    type NeedsTag = B0;
    type Cut = B1;
    type N = U1;
    fn niche() -> GenericArray<u8, Self::N> {
        [0xffu8].into()
    }
    type Next = CharNicheNext;
}

impl MaybeHasNiche for char {
    type MnArray = SomeNiche<CharNiche>;
}

impl ByteOrd for char {
    fn bytes_cmp(&self, other: &Self) -> Ordering {
        self.cmp(other)
    }
}

impl SignificantLength for char {}

impl UsizeTag for char {
    fn from_usize(n: usize) -> Self {
        Self::from_u32(n.try_into().unwrap()).unwrap()
    }

    fn to_usize(&self) -> usize {
        *self as _
    }

    fn try_to_usize(&self) -> Option<usize> {
        Some(self.to_usize())
    }
}

#[test]
fn reparse() -> crate::Result<()> {
    assert_eq!('x'.reparse()?, 'x');
    assert_eq!('ч'.reparse()?, 'ч');
    Ok(())
}