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

153 lines
4.6 KiB
Rust

use serde::ser::{SerializeStruct, Serializer};
use serde::Serialize;
use super::*;
use crate::documents::BuildXML;
use crate::types::*;
use crate::xml_builder::*;
#[derive(Serialize, Debug, Clone, PartialEq)]
#[serde(rename_all = "camelCase")]
pub struct TableCell {
pub children: Vec<TableCellContent>,
pub property: TableCellProperty,
pub has_numbering: bool,
}
#[derive(Debug, Clone, PartialEq)]
pub enum TableCellContent {
Paragraph(Paragraph),
}
impl Serialize for TableCellContent {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
match *self {
TableCellContent::Paragraph(ref s) => {
let mut t = serializer.serialize_struct("Paragraph", 2)?;
t.serialize_field("type", "paragraph")?;
t.serialize_field("data", s)?;
t.end()
}
}
}
}
impl TableCell {
pub fn new() -> TableCell {
Default::default()
}
pub fn add_paragraph(mut self, p: Paragraph) -> TableCell {
if p.has_numbering {
self.has_numbering = true
}
self.children.push(TableCellContent::Paragraph(p));
self
}
pub fn vertical_merge(mut self, t: VMergeType) -> TableCell {
self.property = self.property.vertical_merge(t);
self
}
pub fn vertical_align(mut self, t: VAlignType) -> TableCell {
self.property = self.property.vertical_align(t);
self
}
pub fn grid_span(mut self, v: usize) -> TableCell {
self.property = self.property.grid_span(v);
self
}
pub fn width(mut self, v: usize, t: WidthType) -> TableCell {
self.property = self.property.width(v, t);
self
}
pub fn set_border(mut self, border: TableCellBorder) -> Self {
self.property = self.property.set_border(border);
self
}
pub fn set_borders(mut self, borders: TableCellBorders) -> Self {
self.property = self.property.set_borders(borders);
self
}
pub fn clear_border(mut self, position: BorderPosition) -> Self {
self.property = self.property.clear_border(position);
self
}
pub fn clear_all_border(mut self) -> Self {
self.property = self.property.clear_all_border();
self
}
}
impl Default for TableCell {
fn default() -> Self {
let property = TableCellProperty::new();
let children = vec![];
Self {
property,
children,
has_numbering: false,
}
}
}
impl BuildXML for TableCell {
fn build(&self) -> Vec<u8> {
let b = XMLBuilder::new();
let mut b = b.open_table_cell().add_child(&self.property);
for c in &self.children {
match c {
TableCellContent::Paragraph(p) => b = b.add_child(p),
}
}
b.close().build()
}
}
#[cfg(test)]
mod tests {
use super::*;
#[cfg(test)]
use pretty_assertions::assert_eq;
use std::str;
#[test]
fn test_cell() {
let b = TableCell::new().build();
assert_eq!(str::from_utf8(&b).unwrap(), r#"<w:tc><w:tcPr /></w:tc>"#);
}
#[test]
fn test_cell_add_p() {
let b = TableCell::new()
.add_paragraph(Paragraph::new().add_run(Run::new().add_text("Hello")))
.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
r#"<w:tc><w:tcPr /><w:p w14:paraId="12345678"><w:pPr><w:rPr /></w:pPr><w:r><w:rPr /><w:t xml:space="preserve">Hello</w:t></w:r></w:p></w:tc>"#
);
}
#[test]
fn test_cell_json() {
let c = TableCell::new()
.add_paragraph(Paragraph::new().add_run(Run::new().add_text("Hello")))
.grid_span(2);
assert_eq!(
serde_json::to_string(&c).unwrap(),
r#"{"children":[{"type":"paragraph","data":{"id":"12345678","children":[{"type":"run","data":{"runProperty":{"sz":null,"szCs":null,"color":null,"highlight":null,"underline":null,"bold":null,"boldCs":null,"italic":null,"italicCs":null,"vanish":null,"spacing":null,"fonts":null},"children":[{"type":"text","data":{"preserveSpace":true,"text":"Hello"}}]}}],"property":{"runProperty":{"sz":null,"szCs":null,"color":null,"highlight":null,"underline":null,"bold":null,"boldCs":null,"italic":null,"italicCs":null,"vanish":null,"spacing":null,"fonts":null},"style":null,"numberingProperty":null,"alignment":null,"indent":null,"lineHeight":null},"hasNumbering":false}}],"property":{"width":null,"borders":null,"gridSpan":2,"verticalMerge":null,"verticalAlign":null},"hasNumbering":false}"#
);
}
}