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

216 lines
6.3 KiB
Rust
Raw Normal View History

2019-12-05 08:44:18 +02:00
use super::*;
2019-11-07 09:08:59 +02:00
use crate::documents::BuildXML;
2019-11-11 06:05:07 +02:00
use crate::types::*;
2019-11-07 09:08:59 +02:00
use crate::xml_builder::*;
2019-11-12 11:57:16 +02:00
#[derive(Debug, Clone)]
2019-11-15 11:15:43 +02:00
pub struct Paragraph<'a> {
2019-12-05 08:44:18 +02:00
pub(crate) children: Vec<ParagraphChild<'a>>,
2019-12-06 19:15:21 +02:00
property: ParagraphProperty,
2019-11-13 10:52:02 +02:00
attrs: Vec<(String, String)>,
2019-11-07 09:08:59 +02:00
}
2019-11-15 11:15:43 +02:00
impl<'a> Default for Paragraph<'a> {
2019-11-07 09:08:59 +02:00
fn default() -> Self {
2019-11-11 06:05:07 +02:00
Self {
2019-11-15 11:15:43 +02:00
children: Vec::new(),
2019-11-11 06:05:07 +02:00
property: ParagraphProperty::new(),
2019-11-13 10:52:02 +02:00
attrs: Vec::new(),
2019-11-11 06:05:07 +02:00
}
2019-11-07 09:08:59 +02:00
}
}
2019-11-15 11:15:43 +02:00
#[derive(Debug, Clone)]
pub enum ParagraphChild<'a> {
2019-12-04 11:26:09 +02:00
Run(Run<'a>),
2019-11-15 11:15:43 +02:00
Insert(Insert<'a>),
Delete(Delete<'a>),
2019-12-04 09:55:03 +02:00
BookmarkStart(BookmarkStart<'a>),
BookmarkEnd(BookmarkEnd<'a>),
2019-12-05 08:44:18 +02:00
CommentStart(CommentRangeStart<'a>),
CommentEnd(CommentRangeEnd<'a>),
2019-11-15 11:15:43 +02:00
}
impl<'a> BuildXML for ParagraphChild<'a> {
fn build(&self) -> Vec<u8> {
match self {
ParagraphChild::Run(v) => v.build(),
ParagraphChild::Insert(v) => v.build(),
ParagraphChild::Delete(v) => v.build(),
2019-12-04 09:55:03 +02:00
ParagraphChild::BookmarkStart(v) => v.build(),
ParagraphChild::BookmarkEnd(v) => v.build(),
2019-12-05 08:44:18 +02:00
ParagraphChild::CommentStart(v) => v.build(),
ParagraphChild::CommentEnd(v) => v.build(),
2019-11-15 11:15:43 +02:00
}
}
}
impl<'a> Paragraph<'a> {
pub fn new() -> Paragraph<'a> {
2019-11-07 09:08:59 +02:00
Default::default()
}
2019-12-05 08:44:18 +02:00
pub fn children(&self) -> &Vec<ParagraphChild> {
&self.children
}
2019-12-04 11:26:09 +02:00
pub fn add_run(mut self, run: Run<'a>) -> Paragraph<'a> {
2019-11-15 11:15:43 +02:00
self.children.push(ParagraphChild::Run(run));
2019-11-07 09:08:59 +02:00
self
}
2019-11-11 06:05:07 +02:00
2019-11-15 11:15:43 +02:00
pub fn add_insert(mut self, insert: Insert<'a>) -> Paragraph<'a> {
self.children.push(ParagraphChild::Insert(insert));
2019-11-13 10:52:02 +02:00
self
}
2019-11-15 11:15:43 +02:00
pub fn add_delete(mut self, delete: Delete<'a>) -> Paragraph<'a> {
self.children.push(ParagraphChild::Delete(delete));
2019-11-11 06:05:07 +02:00
self
}
2019-11-11 08:36:26 +02:00
2019-11-15 11:15:43 +02:00
pub fn add_attr(mut self, key: impl Into<String>, val: impl Into<String>) -> Paragraph<'a> {
self.attrs.push((key.into(), val.into()));
2019-11-11 08:36:26 +02:00
self
}
2019-12-04 09:55:03 +02:00
pub fn add_bookmark_start(mut self, id: &'a str, name: &'a str) -> Paragraph<'a> {
self.children
.push(ParagraphChild::BookmarkStart(BookmarkStart::new(id, name)));
self
}
pub fn add_bookmark_end(mut self, id: &'a str) -> Paragraph<'a> {
self.children
.push(ParagraphChild::BookmarkEnd(BookmarkEnd::new(id)));
self
}
2019-12-05 08:44:18 +02:00
pub fn add_comment_start(mut self, comment: Comment<'a>) -> Paragraph<'a> {
self.children
.push(ParagraphChild::CommentStart(CommentRangeStart::new(
comment,
)));
self
}
pub fn add_comment_end(mut self, id: &'a str) -> Paragraph<'a> {
self.children
.push(ParagraphChild::CommentEnd(CommentRangeEnd::new(id)));
self
}
2019-11-15 11:15:43 +02:00
pub fn align(mut self, alignment_type: AlignmentType) -> Paragraph<'a> {
self.property = self.property.align(alignment_type);
self
}
pub fn style(mut self, style_id: &str) -> Paragraph<'a> {
2019-11-11 08:36:26 +02:00
self.property = self.property.style(style_id);
self
}
2019-11-11 09:48:28 +02:00
2019-11-15 11:15:43 +02:00
pub fn indent(
mut self,
left: usize,
special_indent: Option<SpecialIndentType>,
) -> Paragraph<'a> {
2019-11-11 09:48:28 +02:00
self.property = self.property.indent(left, special_indent);
self
}
2019-12-06 12:18:48 +02:00
2019-12-06 19:15:21 +02:00
pub fn numbering(mut self, id: NumberingId, level: IndentLevel) -> Self {
2019-12-06 12:18:48 +02:00
self.property = self.property.numbering(id, level);
self
}
2019-11-07 09:08:59 +02:00
}
2019-11-15 11:15:43 +02:00
impl<'a> BuildXML for Paragraph<'a> {
2019-11-07 09:08:59 +02:00
fn build(&self) -> Vec<u8> {
2019-11-07 10:31:04 +02:00
XMLBuilder::new()
2019-11-13 10:52:02 +02:00
.open_paragraph(&self.attrs)
2019-11-11 06:05:07 +02:00
.add_child(&self.property)
2019-11-15 11:15:43 +02:00
.add_children(&self.children)
2019-11-07 10:31:04 +02:00
.close()
.build()
2019-11-07 09:08:59 +02:00
}
}
#[cfg(test)]
mod tests {
use super::*;
#[cfg(test)]
use pretty_assertions::assert_eq;
use std::str;
#[test]
fn test_paragraph() {
2019-11-13 09:51:58 +02:00
let b = Paragraph::new()
.add_run(Run::new().add_text("Hello"))
.build();
2019-11-07 09:08:59 +02:00
assert_eq!(
str::from_utf8(&b).unwrap(),
2019-11-12 06:33:45 +02:00
r#"<w:p><w:pPr><w:pStyle w:val="Normal" /><w:rPr /></w:pPr><w:r><w:rPr /><w:t xml:space="preserve">Hello</w:t></w:r></w:p>"#
2019-11-07 09:08:59 +02:00
);
}
2019-11-11 08:36:26 +02:00
2019-11-13 10:52:02 +02:00
#[test]
fn test_custom_attr() {
let b = Paragraph::new()
.add_run(Run::new().add_text("Hello"))
.add_attr("customId", "abcd-1234-567890")
.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
r#"<w:p customId="abcd-1234-567890"><w:pPr><w:pStyle w:val="Normal" /><w:rPr /></w:pPr><w:r><w:rPr /><w:t xml:space="preserve">Hello</w:t></w:r></w:p>"#
);
}
2019-12-04 09:55:03 +02:00
#[test]
fn test_bookmark() {
let b = Paragraph::new()
.add_bookmark_start("1234-5678", "article")
.add_run(Run::new().add_text("Hello"))
.add_bookmark_end("1234-5678")
.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
r#"<w:p><w:pPr><w:pStyle w:val="Normal" /><w:rPr /></w:pPr><w:bookmarkStart w:id="1234-5678" w:name="article" /><w:r><w:rPr /><w:t xml:space="preserve">Hello</w:t></w:r><w:bookmarkEnd w:id="1234-5678" /></w:p>"#
);
}
2019-12-05 08:44:18 +02:00
#[test]
fn test_comment() {
let b = Paragraph::new()
.add_comment_start(
Comment::new("1234-5678"), // .paragraph(Paragraph::new().add_run(Run::new().add_text("Comment"))),
)
.add_run(Run::new().add_text("Hello"))
.add_comment_end("1234-5678")
.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
2019-12-06 12:18:48 +02:00
r#"<w:p><w:pPr><w:pStyle w:val="Normal" /><w:rPr /></w:pPr><w:commentRangeStart w:id="1234-5678" /><w:r><w:rPr /><w:t xml:space="preserve">Hello</w:t></w:r><w:r>
<w:rPr />
</w:r>
<w:commentRangeEnd w:id="1234-5678" />
2019-12-05 08:44:18 +02:00
<w:r>
<w:commentReference w:id="1234-5678" />
</w:r></w:p>"#
);
}
2019-12-06 12:18:48 +02:00
#[test]
fn test_numbering() {
let b = Paragraph::new()
.add_run(Run::new().add_text("Hello"))
2019-12-06 19:15:21 +02:00
.numbering(NumberingId::new(0), IndentLevel::new(1))
2019-12-06 12:18:48 +02:00
.build();
assert_eq!(
str::from_utf8(&b).unwrap(),
2019-12-06 19:15:21 +02:00
r#"<w:p><w:pPr><w:pStyle w:val="Normal" /><w:rPr /><w:numPr><w:numId w:val="0" /><w:ilvl w:val="1" /></w:numPr></w:pPr><w:r><w:rPr /><w:t xml:space="preserve">Hello</w:t></w:r></w:p>"#
2019-12-06 12:18:48 +02:00
);
}
2019-11-07 09:08:59 +02:00
}