use crate::optional_data_line; use std::fmt; #[derive(Clone, Debug, Eq, PartialEq)] pub struct Dialogue { pub id: String, pub contents: String, pub name: Option<String>, } impl Dialogue { pub fn from_str(str: &str) -> Result<Dialogue, crate::Error> { let mut lines: Vec<&str> = str.lines().collect(); if lines.is_empty() || !lines[0].starts_with("DLG ") { return Err(crate::Error::Dialogue); } let id = lines[0].replace("DLG ", ""); let last_line = lines.pop().unwrap(); let name = if last_line.starts_with("NAME ") { Some(last_line.replace("NAME ", "")) } else { lines.push(last_line); None }; let contents = lines[1..].join("\n"); Ok(Dialogue { id, contents, name }) } } impl fmt::Display for Dialogue { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { write!( f, "DLG {}\n{}{}", self.id, self.contents, optional_data_line("NAME", self.name.as_ref()) ) } } #[cfg(test)] mod test { use crate::Dialogue; #[test] fn dialogue_from_str() { let output = Dialogue::from_str( "DLG h\nhello\nNAME not a dialogue name\nNAME a dialogue name" ).unwrap(); let expected = Dialogue { id: "h".to_string(), contents: "hello\nNAME not a dialogue name".to_string(), name: Some("a dialogue name".to_string()) }; assert_eq!(output, expected); } #[test] fn dialogue_to_string() { let output = Dialogue { id: "y".to_string(), contents: "This is a bit of dialogue,\nblah blah\nblah blah".to_string(), name: Some("a dialogue name".to_string()) }.to_string(); let expected = "DLG y\nThis is a bit of dialogue,\nblah blah\nblah blah\nNAME a dialogue name"; assert_eq!(output, expected); } }