docx-rs/docx-core/src/documents/elements/run_property.rs

360 lines
10 KiB
Rust
Raw Normal View History

use serde::Serialize;
use std::io::Write;
use super::*;
2019-11-06 12:17:49 +02:00
use crate::documents::BuildXML;
use crate::types::*;
2019-11-06 12:17:49 +02:00
use crate::xml_builder::*;
#[derive(Debug, Clone, Serialize, PartialEq, Default)]
#[serde(rename_all = "camelCase")]
2019-11-06 12:17:49 +02:00
pub struct RunProperty {
#[serde(skip_serializing_if = "Option::is_none")]
pub style: Option<RunStyle>,
#[serde(skip_serializing_if = "Option::is_none")]
pub sz: Option<Sz>,
#[serde(skip_serializing_if = "Option::is_none")]
pub sz_cs: Option<SzCs>,
#[serde(skip_serializing_if = "Option::is_none")]
pub color: Option<Color>,
#[serde(skip_serializing_if = "Option::is_none")]
pub highlight: Option<Highlight>,
#[serde(skip_serializing_if = "Option::is_none")]
pub vert_align: Option<VertAlign>,
#[serde(skip_serializing_if = "Option::is_none")]
pub underline: Option<Underline>,
#[serde(skip_serializing_if = "Option::is_none")]
pub bold: Option<Bold>,
#[serde(skip_serializing_if = "Option::is_none")]
pub bold_cs: Option<BoldCs>,
#[serde(skip_serializing_if = "Option::is_none")]
pub caps: Option<Caps>,
#[serde(skip_serializing_if = "Option::is_none")]
pub italic: Option<Italic>,
#[serde(skip_serializing_if = "Option::is_none")]
pub italic_cs: Option<ItalicCs>,
#[serde(skip_serializing_if = "Option::is_none")]
pub vanish: Option<Vanish>,
#[serde(skip_serializing_if = "Option::is_none")]
2022-11-01 08:10:06 +02:00
pub spec_vanish: Option<SpecVanish>,
#[serde(skip_serializing_if = "Option::is_none")]
pub character_spacing: Option<CharacterSpacing>,
#[serde(skip_serializing_if = "Option::is_none")]
pub fonts: Option<RunFonts>,
#[serde(skip_serializing_if = "Option::is_none")]
pub text_border: Option<TextBorder>,
#[serde(skip_serializing_if = "Option::is_none")]
pub del: Option<Delete>,
#[serde(skip_serializing_if = "Option::is_none")]
pub ins: Option<Insert>,
#[serde(skip_serializing_if = "Option::is_none")]
pub strike: Option<Strike>,
#[serde(skip_serializing_if = "Option::is_none")]
pub dstrike: Option<Dstrike>,
#[serde(skip_serializing_if = "Option::is_none")]
pub positional_tab: Option<PositionalTab>,
#[serde(skip_serializing_if = "Option::is_none")]
pub shading: Option<Shading>,
2019-11-06 12:17:49 +02:00
}
impl RunProperty {
pub fn new() -> RunProperty {
2019-11-07 06:57:58 +02:00
Default::default()
2019-11-06 12:17:49 +02:00
}
pub fn style(mut self, style_id: &str) -> Self {
self.style = Some(RunStyle::new(style_id));
self
}
2019-11-11 08:36:26 +02:00
pub fn size(mut self, size: usize) -> RunProperty {
self.sz = Some(Sz::new(size));
self.sz_cs = Some(SzCs::new(size));
2019-11-06 12:17:49 +02:00
self
}
2019-11-07 06:57:58 +02:00
pub fn spacing(mut self, spacing: i32) -> RunProperty {
self.character_spacing = Some(CharacterSpacing::new(spacing));
self
}
pub fn color(mut self, color: impl Into<String>) -> RunProperty {
2019-11-07 06:57:58 +02:00
self.color = Some(Color::new(color));
self
}
2019-11-13 09:08:25 +02:00
pub fn highlight(mut self, color: impl Into<String>) -> RunProperty {
2019-11-13 09:08:25 +02:00
self.highlight = Some(Highlight::new(color));
self
}
pub fn vert_align(mut self, a: VertAlignType) -> Self {
self.vert_align = Some(VertAlign::new(a));
self
}
2019-11-13 09:08:25 +02:00
pub fn bold(mut self) -> RunProperty {
self.bold = Some(Bold::new());
self.bold_cs = Some(BoldCs::new());
self
}
2021-03-16 10:41:36 +02:00
pub fn disable_bold(mut self) -> RunProperty {
self.bold = Some(Bold::new().disable());
self.bold_cs = Some(BoldCs::new().disable());
self
}
pub fn caps(mut self) -> RunProperty {
self.caps = Some(Caps::new());
self
}
2019-11-13 09:08:25 +02:00
pub fn italic(mut self) -> RunProperty {
self.italic = Some(Italic::new());
self.italic_cs = Some(ItalicCs::new());
self
}
2019-12-04 09:28:11 +02:00
pub fn strike(mut self) -> RunProperty {
self.strike = Some(Strike::new());
self.dstrike = None;
self
}
pub fn disable_strike(mut self) -> RunProperty {
self.strike = Some(Strike::new().disable());
self
}
pub fn dstrike(mut self) -> RunProperty {
self.dstrike = Some(Dstrike::new());
self.strike = None;
self
}
pub fn disable_dstrike(mut self) -> RunProperty {
self.dstrike = Some(Dstrike::new().disable());
self
}
2021-03-16 10:41:36 +02:00
pub fn disable_italic(mut self) -> RunProperty {
self.italic = Some(Italic::new().disable());
self.italic_cs = Some(ItalicCs::new().disable());
self
}
pub fn underline(mut self, line_type: impl Into<String>) -> RunProperty {
2019-12-04 09:28:11 +02:00
self.underline = Some(Underline::new(line_type));
self
}
pub fn vanish(mut self) -> RunProperty {
self.vanish = Some(Vanish::new());
self
}
2022-11-01 08:10:06 +02:00
pub fn spec_vanish(mut self) -> RunProperty {
self.spec_vanish = Some(SpecVanish::new());
self
}
pub fn fonts(mut self, font: RunFonts) -> RunProperty {
self.fonts = Some(font);
self
}
pub fn character_spacing(mut self, v: i32) -> RunProperty {
self.character_spacing = Some(CharacterSpacing::new(v));
self
}
pub fn text_border(mut self, b: TextBorder) -> Self {
self.text_border = Some(b);
self
}
pub fn delete(mut self, d: Delete) -> Self {
self.del = Some(d);
self
}
pub fn insert(mut self, i: Insert) -> Self {
self.ins = Some(i);
self
}
pub fn ptab(mut self, ptab: PositionalTab) -> Self {
self.positional_tab = Some(ptab);
self
}
pub fn shading(mut self, s: Shading) -> Self {
self.shading = Some(s);
self
}
2019-11-07 06:57:58 +02:00
}
2019-11-06 12:17:49 +02:00
impl BuildXML for RunProperty {
fn build_to<W: Write>(
&self,
stream: xml::writer::EventWriter<W>,
) -> xml::writer::Result<xml::writer::EventWriter<W>> {
XMLBuilder::from(stream)
.open_run_property()?
.add_optional_child(&self.sz)?
.add_optional_child(&self.sz_cs)?
.add_optional_child(&self.color)?
.add_optional_child(&self.bold)?
.add_optional_child(&self.bold_cs)?
.add_optional_child(&self.caps)?
.add_optional_child(&self.italic)?
.add_optional_child(&self.italic_cs)?
.add_optional_child(&self.strike)?
.add_optional_child(&self.dstrike)?
.add_optional_child(&self.highlight)?
.add_optional_child(&self.underline)?
.add_optional_child(&self.vanish)?
.add_optional_child(&self.spec_vanish)?
.add_optional_child(&self.fonts)?
.add_optional_child(&self.text_border)?
.add_optional_child(&self.ins)?
.add_optional_child(&self.del)?
.add_optional_child(&self.vert_align)?
.add_optional_child(&self.character_spacing)?
.add_optional_child(&self.style)?
.add_optional_child(&self.positional_tab)?
.add_optional_child(&self.shading)?
.close()?
.into_inner()
2019-11-06 12:17:49 +02:00
}
}
#[cfg(test)]
mod tests {
use super::*;
#[cfg(test)]
use pretty_assertions::assert_eq;
use std::str;
#[test]
2019-11-13 09:08:25 +02:00
fn test_size() {
2019-11-11 08:36:26 +02:00
let c = RunProperty::new().size(10).color("FFFFFF");
2019-11-06 12:17:49 +02:00
let b = c.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
2019-11-11 08:36:26 +02:00
r#"<w:rPr><w:sz w:val="10" /><w:szCs w:val="10" /><w:color w:val="FFFFFF" /></w:rPr>"#
2019-11-06 12:17:49 +02:00
);
}
2019-11-13 09:08:25 +02:00
#[test]
fn test_highlight() {
let c = RunProperty::new().highlight("FFFFFF");
let b = c.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
r#"<w:rPr><w:highlight w:val="FFFFFF" /></w:rPr>"#
);
}
#[test]
fn test_bold() {
let c = RunProperty::new().bold();
let b = c.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
r#"<w:rPr><w:b /><w:bCs /></w:rPr>"#
);
}
2019-12-04 09:28:11 +02:00
#[test]
fn test_strike() {
let c = RunProperty::new().strike();
let b = c.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
r#"<w:rPr><w:strike /></w:rPr>"#
);
}
#[test]
2019-12-04 09:28:11 +02:00
fn test_underline() {
let c = RunProperty::new().underline("single");
let b = c.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
r#"<w:rPr><w:u w:val="single" /></w:rPr>"#
);
}
#[test]
fn test_vanish() {
let c = RunProperty::new().vanish();
let b = c.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
r#"<w:rPr><w:vanish /></w:rPr>"#
);
}
#[test]
fn test_run_fonts() {
let c = RunProperty::new().fonts(RunFonts::new().east_asia("Hiragino"));
let b = c.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
r#"<w:rPr><w:rFonts w:eastAsia="Hiragino" /></w:rPr>"#
);
}
#[test]
fn test_character_spacing() {
let c = RunProperty::new().spacing(20);
let b = c.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
r#"<w:rPr><w:spacing w:val="20" /></w:rPr>"#
);
}
#[test]
fn test_ptab() {
let c = RunProperty::new().ptab(PositionalTab::new(
PositionalTabAlignmentType::Left,
PositionalTabRelativeTo::Margin,
TabLeaderType::None,
));
let b = c.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
r#"<w:rPr><w:ptab w:alignment="left" w:relativeTo="margin" w:leader="none" /></w:rPr>"#
);
}
#[test]
fn test_character_shading() {
let c = RunProperty::new().shading(
Shading::new()
.shd_type(ShdType::Clear)
.fill("FFFFFF")
.color("auto"),
);
let b = c.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
r#"<w:rPr><w:shd w:val="clear" w:color="auto" w:fill="FFFFFF" /></w:rPr>"#
);
}
#[test]
fn test_dstrike() {
let c = RunProperty::new().dstrike();
let b = c.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
r#"<w:rPr><w:dstrike /></w:rPr>"#
);
}
2019-11-06 12:17:49 +02:00
}