use serde::Serialize; use super::*; use crate::documents::BuildXML; use crate::types::{AlignmentType, SpecialIndentType}; use crate::xml_builder::*; #[derive(Serialize, Debug, Clone, PartialEq)] #[serde(rename_all = "camelCase")] pub struct ParagraphProperty { pub run_property: RunProperty, pub style: Option, pub numbering_property: Option, pub alignment: Option, pub indent: Option, pub line_spacing: Option, pub keep_next: bool, pub keep_lines: bool, pub page_break_before: bool, pub window_control: bool, pub outline_lvl: Option, // read only pub(crate) div_id: Option, } impl Default for ParagraphProperty { fn default() -> Self { ParagraphProperty { run_property: RunProperty::new(), style: None, numbering_property: None, alignment: None, indent: None, line_spacing: None, keep_next: false, keep_lines: false, page_break_before: false, window_control: false, outline_lvl: None, div_id: None, } } } // 17.3.1.26 // pPr (Paragraph Properties) // This element specifies a set of paragraph properties which shall be applied to the contents of the parent // paragraph after all style/numbering/table properties have been applied to the text. These properties are defined // as direct formatting, since they are directly applied to the paragraph and supersede any formatting from styles. impl ParagraphProperty { pub fn new() -> ParagraphProperty { Default::default() } pub fn align(mut self, alignment_type: AlignmentType) -> Self { self.alignment = Some(Justification::new(alignment_type.to_string())); self } pub fn style(mut self, style_id: &str) -> Self { self.style = Some(ParagraphStyle::new(Some(style_id))); self } pub fn indent( mut self, left: Option, special_indent: Option, end: Option, start_chars: Option, ) -> Self { self.indent = Some(Indent::new(left, special_indent, end, start_chars)); self } pub fn numbering(mut self, id: NumberingId, level: IndentLevel) -> Self { self.numbering_property = Some(NumberingProperty::new().add_num(id, level)); self } pub fn line_spacing(mut self, spacing: LineSpacing) -> Self { self.line_spacing = Some(spacing); self } pub fn keep_next(mut self, v: bool) -> Self { self.keep_next = v; self } pub fn keep_lines(mut self, v: bool) -> Self { self.keep_lines = v; self } pub fn outline_lvl(mut self, v: usize) -> Self { if v >= 10 { return self; } self.outline_lvl = Some(OutlineLvl::new(v)); self } pub fn page_break_before(mut self, v: bool) -> Self { self.page_break_before = v; self } pub fn window_control(mut self, v: bool) -> Self { self.window_control = v; self } pub(crate) fn hanging_chars(mut self, chars: i32) -> Self { if let Some(indent) = self.indent { self.indent = Some(indent.hanging_chars(chars)); } self } pub(crate) fn first_line_chars(mut self, chars: i32) -> Self { if let Some(indent) = self.indent { self.indent = Some(indent.first_line_chars(chars)); } self } } impl BuildXML for ParagraphProperty { fn build(&self) -> Vec { let mut b = XMLBuilder::new() .open_paragraph_property() .add_child(&self.run_property) .add_optional_child(&self.style) .add_optional_child(&self.numbering_property) .add_optional_child(&self.alignment) .add_optional_child(&self.indent) .add_optional_child(&self.line_spacing) .add_optional_child(&self.outline_lvl); if self.keep_next { b = b.keep_next() } if self.keep_lines { b = b.keep_lines() } if self.page_break_before { b = b.page_break_before() } if self.window_control { b = b.window_control() } b.close().build() } } #[cfg(test)] mod tests { use super::*; use crate::types::LineSpacingType; #[cfg(test)] use pretty_assertions::assert_eq; use std::str; #[test] fn test_default() { let c = ParagraphProperty::new(); let b = c.build(); assert_eq!(str::from_utf8(&b).unwrap(), r#""#); } #[test] fn test_alignment() { let c = ParagraphProperty::new(); let b = c.align(AlignmentType::Right).build(); assert_eq!( str::from_utf8(&b).unwrap(), r#""# ); } #[test] fn test_indent() { let c = ParagraphProperty::new(); let b = c.indent(Some(20), None, None, None).build(); assert_eq!( str::from_utf8(&b).unwrap(), r#""# ); } #[test] fn test_keep_next() { let c = ParagraphProperty::new(); let b = c.keep_next(true).build(); assert_eq!( str::from_utf8(&b).unwrap(), r#" "# ); } #[test] fn test_outline_lvl() { let props = ParagraphProperty::new(); let bytes = props.outline_lvl(1).build(); assert_eq!( str::from_utf8(&bytes).unwrap(), r#""# ) } #[test] fn test_indent_json() { let c = ParagraphProperty::new(); let b = c.indent(Some(20), Some(SpecialIndentType::FirstLine(10)), None, None); assert_eq!( serde_json::to_string(&b).unwrap(), r#"{"runProperty":{"sz":null,"szCs":null,"color":null,"highlight":null,"vertAlign":null,"underline":null,"bold":null,"boldCs":null,"italic":null,"italicCs":null,"vanish":null,"characterSpacing":null,"fonts":null,"textBorder":null,"del":null,"ins":null},"style":null,"numberingProperty":null,"alignment":null,"indent":{"start":20,"startChars":null,"end":null,"specialIndent":{"type":"firstLine","val":10},"hangingChars":null,"firstLineChars":null},"lineSpacing":null,"keepNext":false,"keepLines":false,"pageBreakBefore":false,"windowControl":false,"outlineLvl":null,"divId":null}"# ); } #[test] fn test_line_spacing() { let props = ParagraphProperty::new(); let spacing = LineSpacing::new() .line_rule(LineSpacingType::AtLeast) .line(100); let bytes = props.line_spacing(spacing).build(); assert_eq!( str::from_utf8(&bytes).unwrap(), r#""# ) } }