Skip to content

Commit aed7ca4

Browse files
committed
feat: allow keeping templates as they are
In some cases the <template> tags should not be replaced, but kept as they are. This allows Closes: trunk-rs/trunk#742
1 parent 54e044e commit aed7ca4

File tree

4 files changed

+103
-9
lines changed

4 files changed

+103
-9
lines changed

.gitignore

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
/target
22
**/*.rs.bk
33
Cargo.lock
4-
*.go
4+
*.go
5+
/.idea

src/document.rs

+55-8
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,21 @@ use std::collections::HashSet;
1515
use tendril::StrTendril;
1616
use tendril::TendrilSink;
1717

18+
#[derive(Clone, Debug, Default)]
19+
#[non_exhaustive]
20+
/// Parser options
21+
pub struct ParseOptions {
22+
/// Keep the content of `<template>` tags in place.
23+
pub keep_templates: bool,
24+
}
25+
26+
impl ParseOptions {
27+
pub fn keep_templates(mut self, keep_templates: bool) -> Self {
28+
self.keep_templates = keep_templates;
29+
self
30+
}
31+
}
32+
1833
/// Document represents an HTML document to be manipulated.
1934
pub struct Document {
2035
/// The document's dom tree.
@@ -25,6 +40,9 @@ pub struct Document {
2540

2641
/// The document's quirks mode.
2742
pub quirks_mode: QuirksMode,
43+
44+
/// Keep content of templates
45+
pub keep_templates: bool,
2846
}
2947

3048
impl Default for Document {
@@ -33,6 +51,7 @@ impl Default for Document {
3351
tree: Tree::new(NodeData::Document),
3452
errors: vec![],
3553
quirks_mode: tree_builder::NoQuirks,
54+
keep_templates: false,
3655
}
3756
}
3857
}
@@ -60,6 +79,23 @@ impl Document {
6079
pub fn root(&self) -> NodeRef<NodeData> {
6180
self.tree.root()
6281
}
82+
83+
/// Parse a document allowing to provide additional options
84+
pub fn parse<S>(options: ParseOptions, html: S) -> Self
85+
where
86+
S: AsRef<str>,
87+
{
88+
let ParseOptions { keep_templates } = options;
89+
90+
parse_document(
91+
Document {
92+
keep_templates,
93+
..Document::default()
94+
},
95+
Default::default(),
96+
)
97+
.one(html.as_ref())
98+
}
6399
}
64100

65101
impl TreeSink for Document {
@@ -87,14 +123,19 @@ impl TreeSink for Document {
87123

88124
// Get a handle to a template's template contents. The tree builder promises this will never be called with
89125
// something else than a template element.
126+
// If templates are kept in place, return the actual node id.
90127
fn get_template_contents(&mut self, target: &NodeId) -> NodeId {
91-
self.tree.query_node(target, |node| match node.data {
92-
NodeData::Element(Element {
93-
template_contents: Some(ref contents),
94-
..
95-
}) => contents.clone(),
96-
_ => panic!("not a template element!"),
97-
})
128+
if self.keep_templates {
129+
target.clone()
130+
} else {
131+
self.tree.query_node(target, |node| match node.data {
132+
NodeData::Element(Element {
133+
template_contents: Some(ref contents),
134+
..
135+
}) => contents.clone(),
136+
_ => panic!("not a template element!"),
137+
})
138+
}
98139
}
99140

100141
// Set the document's quirks mode.
@@ -126,7 +167,7 @@ impl TreeSink for Document {
126167
attrs: Vec<Attribute>,
127168
flags: ElementFlags,
128169
) -> NodeId {
129-
let template_contents = if flags.template {
170+
let template_contents = if !self.keep_templates && flags.template {
130171
Some(self.tree.create_node(NodeData::Document))
131172
} else {
132173
None
@@ -282,6 +323,12 @@ impl TreeSink for Document {
282323
}
283324
}
284325

326+
impl Document {
327+
pub fn get_node(&self, id: &NodeId) -> NodeData {
328+
self.tree.query_node(id, |node| node.data.clone())
329+
}
330+
}
331+
285332
#[cfg(test)]
286333
mod tests {
287334
use super::*;

src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,7 @@ mod selection;
3838
mod traversal;
3939

4040
pub use document::Document;
41+
pub use document::ParseOptions;
4142
pub use dom_tree::Node;
4243
#[doc(hidden)]
4344
pub use dom_tree::NodeId;

tests/templates.rs

+45
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
use markup5ever::interface::TreeSink;
2+
use nipper::{Document, ParseOptions};
3+
4+
#[test]
5+
fn test_templates() {
6+
let mut doc = Document::parse(
7+
ParseOptions {
8+
keep_templates: true,
9+
},
10+
r#"
11+
<html>
12+
<body>
13+
<template>
14+
<p>Hello world!</p>
15+
</template>
16+
</body>
17+
</html>
18+
"#,
19+
);
20+
21+
let nodes = doc
22+
.select("template")
23+
.nodes()
24+
.iter()
25+
.map(|n| n.id)
26+
.collect::<Vec<_>>();
27+
for node in nodes {
28+
let c = doc.get_template_contents(&node);
29+
println!("Node: {c:?}");
30+
let data = doc.get_node(&node);
31+
println!("Data: {data:?}");
32+
}
33+
34+
let result = doc.html().to_string();
35+
assert_eq!(
36+
result,
37+
r#"<html><head></head><body>
38+
<template>
39+
<p>Hello world!</p>
40+
</template>
41+
42+
43+
</body></html>"#
44+
);
45+
}

0 commit comments

Comments
 (0)