include!(concat!(env!("OUT_DIR"), "/space_separator.rs"));
use precis_core::Codepoints;
use precis_core::Error;
use std::borrow::Cow;
use unicode_normalization::UnicodeNormalization;
pub const SPACE: char = '\u{0020}';
pub fn is_space_separator(c: char) -> bool {
    let cp = c as u32;
    SPACE_SEPARATOR
        .binary_search_by(|cps| cps.partial_cmp(&cp).unwrap())
        .is_ok()
}
pub fn is_non_ascii_space(c: char) -> bool {
    c != SPACE && is_space_separator(c)
}
pub fn normalization_form_nfkc<'a, T>(s: T) -> Result<Cow<'a, str>, Error>
where
    T: Into<Cow<'a, str>>,
{
                let s = s.into();
    if unicode_normalization::is_nfkc(&s) {
        Ok(s)
    } else {
        Ok(s.nfkc().collect::<String>().into())
    }
}
pub fn normalization_form_nfc<'a, T>(s: T) -> Result<Cow<'a, str>, Error>
where
    T: Into<Cow<'a, str>>,
{
                let s = s.into();
    if unicode_normalization::is_nfc(&s) {
        Ok(s)
    } else {
        Ok(s.nfc().collect::<String>().into())
    }
}
pub fn case_mapping_rule<'a, T>(s: T) -> Result<Cow<'a, str>, Error>
where
    T: Into<Cow<'a, str>>,
{
    let s = s.into();
    match s.find(char::is_uppercase) {
        None => Ok(s),
        Some(pos) => {
            let mut res = String::from(&s[..pos]);
            res.reserve(s.len() - res.len());
            for c in s[pos..].chars() {
                if c.is_lowercase() {
                    res.push(c);
                } else {
                    c.to_lowercase().for_each(|x| res.push(x));
                }
            }
            Ok(res.into())
        }
    }
}
#[cfg(test)]
mod profile_rules {
    use crate::common::*;
    #[test]
    fn test_normalization_rule() {
        let res = normalization_form_nfc("");
        assert_eq!(res, Ok(Cow::from("")));
        let res = normalization_form_nfc("\u{212b}");
        assert_eq!(res, Ok(Cow::from("\u{00c5}")));
        let res = normalization_form_nfc("a\u{212b}");
        assert_eq!(res, Ok(Cow::from("a\u{00c5}")));
        let res = normalization_form_nfc("\u{212b}a");
        assert_eq!(res, Ok(Cow::from("\u{00c5}a")));
        let res = normalization_form_nfc("\u{212b}\u{2126}\u{1e0b}\u{0323}");
        assert_eq!(res, Ok(Cow::from("\u{00c5}\u{03a9}\u{1e0d}\u{0307}")));
    }
    #[test]
    fn test_case_mapping_rule() {
        let res = case_mapping_rule("");
        assert_eq!(res, Ok(Cow::from("")));
        let res = case_mapping_rule("T");
        assert_eq!(res, Ok(Cow::from("t")));
        let res = case_mapping_rule("aT");
        assert_eq!(res, Ok(Cow::from("at")));
        let res = case_mapping_rule("Ta");
        assert_eq!(res, Ok(Cow::from("ta")));
        let res = case_mapping_rule("TestUserName");
        assert_eq!(res, Ok(Cow::from("testusername")));
    }
    #[test]
    fn test_is_non_ascii_space() {
                assert!(!is_non_ascii_space(SPACE));
                assert!(!is_non_ascii_space('a'));
                assert!(is_non_ascii_space('\u{205f}'));
    }
}