use crate::datacell::{BlockChildType::*, CellTrait::Cell, Datacell::*, ElementCell::*};
pub struct ElementText {
    pub text: String,
    rules: Vec<DelimeterRules>,
}
#[derive(Debug)]
struct DelimeterRules {
    symbol: &'static str,
    end_symbol: &'static str,
    left_replacement: &'static str,
    right_replacement: &'static str,
    no_break: bool,
    keep_delimiter: bool,
    ignore_nested_delimeters: bool,
    ignore_when_before: Vec<char>,
    ignore_when_after: Vec<char>,
}
impl Default for DelimeterRules {
    fn default() -> Self {
        Self {
            symbol: "",
            end_symbol: "",
            left_replacement: "",
            right_replacement: "",
            no_break: false,
            keep_delimiter: false,
            ignore_nested_delimeters: false,
            ignore_when_before: Vec::new(),
            ignore_when_after: Vec::new(),
        }
    }
}
impl ElementText {
    pub fn new(text: &str) -> ElementText {
        ElementText {
            text: text.to_string(),
            rules: vec![
                DelimeterRules {
                    symbol: "*",
                    end_symbol: "*",
                    left_replacement: "<Span bold=true>",
                    right_replacement: "</Span>",
                    ignore_when_after: vec!['(', '[', '{', '*', ' '],
                    ignore_when_before: vec![')', ']', '}', '*', ' '],
                    ..Default::default()
                },
                DelimeterRules {
                    symbol: "__",
                    end_symbol: "__",
                    left_replacement: "<Span italic=true align=Align::Center>",
                    right_replacement: "</Span>",
                    ..Default::default()
                },
                DelimeterRules {
                    symbol: "_|",
                    end_symbol: "|_",
                    left_replacement: "<Span align=Align::Center>",
                    right_replacement: "</Span>",
                    ..Default::default()
                },
                DelimeterRules {
                    symbol: "_",
                    end_symbol: "_",
                    left_replacement: "<Span italic=true>",
                    right_replacement: "</Span>",
                    ignore_when_after: vec!['(', '[', '{', ' '],
                    ignore_when_before: vec![')', ']', '}', ' '],
                    ..Default::default()
                },
                DelimeterRules {
                    symbol: "$$",
                    end_symbol: "$$",
                    left_replacement: "<MathBlock>",
                    right_replacement: "</MathBlock>",
                    no_break: true,
                    keep_delimiter: true,
                    ignore_nested_delimeters: true,
                    ..Default::default()
                },
                DelimeterRules {
                    symbol: "$",
                    end_symbol: "$",
                    left_replacement: "<Math>",
                    right_replacement: "</Math>",
                    no_break: true,
                    keep_delimiter: true,
                    ignore_nested_delimeters: true,
                    ..Default::default()
                },
            ],
        }
    }
    pub fn handle_delimeters(self) -> String {
        let mut i = 0;
        let mut output = String::new();
        while i <= self.text.len() {
            let (del, skips, text) = &self.find_next_delimeter(i, false);
            output.push_str(text);
            if del.is_none() {
                break;
            }
            i = *skips;
            if i <= self.text.len() {
                let (found, closing_index, del_content) =
                    &self.find_closing_delimeter(i, &del.unwrap(), false);
                if !found {
                    output.push_str(&del.unwrap().symbol);
                    let nested_content = ElementText::new(&del_content).handle_delimeters();
                    output.push_str(nested_content.as_str());
                    i = *closing_index + 1;
                    continue;
                }
                let nested_content = if !del.unwrap().ignore_nested_delimeters {
                    ElementText::new(&del_content).handle_delimeters()
                } else {
                    del_content.to_string()
                };
                if i <= self.text.len() {
                    i = closing_index + del.unwrap().end_symbol.len();
                    let mut char_after_closing_del = "";
                    if i <= self.text.len() && del.unwrap().no_break {
                        char_after_closing_del = &self.get_slice(i, i + 1).unwrap_or("");
                    }
                    if char_after_closing_del != " "
                        && char_after_closing_del != ""
                        && del.unwrap().no_break
                    {
                        let mut removed = String::new();
                        while output.len() > 0 && output.chars().last().unwrap() != ' ' {
                            removed = format!("{}{}", output.pop().unwrap(), removed);
                        }
                        output.push_str("\"#<span class=\"nobreak\">r#\"");
                        output.push_str(removed.as_str());
                        output.push_str("\"#");
                    } else {
                        output.push_str("\"#");
                    }
                    output.push_str(del.unwrap().left_replacement);
                    output.push_str("r#\"");
                    if del.unwrap().keep_delimiter {
                        output.push_str(&del.unwrap().symbol);
                    }
                    output.push_str(&nested_content);
                    if del.unwrap().keep_delimiter {
                        output.push_str(&del.unwrap().end_symbol);
                    }
                    output.push_str("\"#");
                    output.push_str(del.unwrap().right_replacement);
                    if del.unwrap().no_break
                        && char_after_closing_del != " "
                        && char_after_closing_del != ""
                    {
                        output.push_str("r#\"");
                        let mut string = "".to_string();
                        while i < self.text.len()
                            && self.get_char(i) != " "
                            && self.get_char(i) != ""
                        {
                            string.push_str(self.get_char(i).as_str());
                            i += 1;
                        }
                        let handled_string = self::ElementText::new(&string).handle_delimeters();
                        i += 1;
                        output.push_str(&handled_string);
                        output.push_str("\"#</span>r#\"");
                    } else {
                        output.push_str("r#\"");
                        i += 1;
                    }
                }
            }
        }
        output
    }
    pub fn split_text(self) -> Vec<BlockChildType> {
        let mut i = 0;
        let mut output = Vec::<BlockChildType>::new();
        while i <= self.text.len() {
            let (del, skips, text) = &self.find_next_delimeter(i, true);
            output.push(BlockChildType::Text(TextCell {
                content: text.to_string(),
                wrapped_with: None,
            }));
            if !del.is_some() {
                break;
            }
            i = *skips;
            if i <= self.text.len() {
                let (found, closing_index, del_content) =
                    &self.find_closing_delimeter(i, &del.unwrap(), true);
                if !found {
                    let last_child = output.pop().unwrap();
                    match last_child {
                        BlockChildType::Text(mut t) => {
                            t.content
                                .push_str(&format!("{}{}", &del.unwrap().symbol, del_content));
                            output.push(BlockChildType::Text(t))
                        }
                        _ => (),
                    }
                    i = *closing_index + 1;
                    continue;
                }
                if i <= self.text.len() {
                    i = closing_index + del.unwrap().end_symbol.len();
                    output.push(BlockChildType::Delimited(DelimitedCell {
                        terminal: del_content.to_owned(),
                        open_delimeter: del.unwrap().symbol.to_string(),
                        close_delimeter: del.unwrap().end_symbol.to_string(),
                        display_type: if del.unwrap().symbol.len() > 1 {
                            DelimitedDisplayType::BLOCK
                        } else {
                            DelimitedDisplayType::INLINE
                        },
                        wrapped_with: None,
                    }));
                    i += 1;
                }
            }
        }
        output
    }
    fn find_next_delimeter(
        &self,
        mut i: usize,
        keep_escape_char: bool,
    ) -> (Option<&DelimeterRules>, usize, String) {
        let mut found_symbol = "";
        let mut del: Option<&DelimeterRules> = None;
        let mut text = "".to_string();
        let symbols: Vec<&str> = self.rules.iter().map(|d| d.symbol).collect();
        let mut has_multi_char = false;
        while i <= self.text.len() {
            let _del = self.rules.iter().find(|d| {
                if i.checked_sub(d.symbol.len()).is_some() {
                    d.symbol
                        == &self
                            .text
                            .chars()
                            .take(i)
                            .skip(i - d.symbol.len())
                            .collect::<String>()
                        && self.text.chars().nth(i - d.symbol.len() + 1).is_some()
                        && !d
                            .ignore_when_before
                            .contains(&self.text.chars().nth(i - d.symbol.len() + 1).unwrap())
                } else {
                    false
                }
            });
            if _del.is_some() {
                if symbols.contains(
                    &self
                        .text
                        .chars()
                        .take(i + 1)
                        .skip(i - _del.unwrap().symbol.len())
                        .collect::<String>()
                        .as_str(),
                ) {
                    i += 1;
                    has_multi_char = true;
                    continue;
                }
                del = _del;
                found_symbol = _del.unwrap().symbol;
                if !has_multi_char {}
                if self.is_escaped(i - found_symbol.len()) {
                    let mut nth = text.chars().nth(i);
                    while nth.is_some() && nth.unwrap() != '\\' {
                        text.pop();
                        nth = text.chars().nth(i);
                    }
                    if !keep_escape_char {
                        text.pop();
                    }
                    text.push_str(found_symbol); i += 1;
                    continue;
                }
                break;
            }
            if i.checked_sub(1).is_some() && !(i == 1 && &self.get_char(i - 1) == " ") {
                text.push_str(&self.get_char(i - 1));
            }
            i += 1;
        }
        (del, i, text)
    }
    fn find_closing_delimeter(
        &self,
        mut i: usize,
        found_del: &DelimeterRules,
        keep_escape_char: bool,
    ) -> (bool, usize, String) {
        let end_symbol = found_del.end_symbol;
        let mut found = false;
        let mut del_content = "".to_string();
        let symbols: Vec<&str> = self.rules.iter().map(|d| d.symbol).collect();
        while i < self.text.len() {
            let is_found = end_symbol
                == &self
                    .text
                    .chars()
                    .take(i + end_symbol.len())
                    .skip(i)
                    .collect::<String>()
                && !found_del
                    .ignore_when_after
                    .contains(&self.text.chars().nth(i - 1).unwrap());
            if !is_found {
                if end_symbol
                    != &self
                        .text
                        .chars()
                        .take(i + end_symbol.len())
                        .skip(i)
                        .collect::<String>()
                    || found_del
                        .ignore_when_after
                        .contains(&self.text.chars().nth(i - 1).unwrap())
                {
                    del_content.push_str(&self.get_char(i));
                }
                i += 1;
                continue;
            }
            let next_char = &self.text.chars().take(i + 2).skip(i).collect::<String>();
            if symbols.contains(
                &next_char.as_str(),
            ) && next_char != &end_symbol
            {
                i += 2;
                del_content.push_str(next_char);
                continue;
            }
            found = true;
            if self.is_escaped(i) {
                if keep_escape_char {
                    del_content.push_str("\\");
                }
                del_content.push_str(end_symbol);
                found = false;
                i += end_symbol.len();
                continue;
            }
            break;
        }
        (found, i, del_content)
    }
    fn get_char(&self, i: usize) -> String {
        if let Some(_char) = self.text.chars().nth(i) {
            _char.to_string()
        } else {
            "".to_string()
        }
    }
    fn is_escaped(&self, i: usize) -> bool {
        i > 0 && self.get_char(i - 1) == "\\"
    }
    fn get_slice(&self, start: usize, end: usize) -> Option<&str> {
        assert!(end >= start);
        let s = &self.text;
        let mut iter = s
            .char_indices()
            .map(|(pos, _)| pos)
            .chain(Some(s.len()))
            .skip(start)
            .peekable();
        let start_pos = *iter.peek()?;
        for _ in start..end {
            iter.next();
        }
        Some(&s[start_pos..*iter.peek()?])
    }
}