Skip to content

Commit 6b89f5d

Browse files
authored
Merge pull request #2327 from ehuss/smart-punctuation
Rename curly-quotes to smart-punctuation.
2 parents 504900d + d28cf53 commit 6b89f5d

File tree

6 files changed

+89
-32
lines changed

6 files changed

+89
-32
lines changed

guide/src/format/configuration/renderers.md

+5-3
Original file line numberDiff line numberDiff line change
@@ -97,7 +97,7 @@ description = "The example book covers examples."
9797
theme = "my-theme"
9898
default-theme = "light"
9999
preferred-dark-theme = "navy"
100-
curly-quotes = true
100+
smart-punctuation = true
101101
mathjax-support = false
102102
copy-fonts = true
103103
additional-css = ["custom.css", "custom2.css"]
@@ -122,8 +122,10 @@ The following configuration options are available:
122122
the browser requests the dark version of the site via the
123123
['prefers-color-scheme'](https://developer.mozilla.org/en-US/docs/Web/CSS/@media/prefers-color-scheme)
124124
CSS media query. Defaults to `navy`.
125-
- **curly-quotes:** Convert straight quotes to curly quotes, except for those
126-
that occur in code blocks and code spans. Defaults to `false`.
125+
- **smart-punctuation:** Converts quotes to curly quotes, `...` to ``, `--` to en-dash, and `---` to em-dash.
126+
See [Smart Punctuation].
127+
Defaults to `false`.
128+
- **curly-quotes:** Deprecated alias for `smart-punctuation`.
127129
- **mathjax-support:** Adds support for [MathJax](../mathjax.md). Defaults to
128130
`false`.
129131
- **copy-fonts:** (**Deprecated**) If `true` (the default), mdBook uses its built-in fonts which are copied to the output directory.

guide/src/format/markdown.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -214,12 +214,12 @@ characters:
214214
So, no need to manually enter those Unicode characters!
215215

216216
This feature is disabled by default.
217-
To enable it, see the [`output.html.curly-quotes`] config option.
217+
To enable it, see the [`output.html.smart-punctuation`] config option.
218218

219219
[strikethrough]: https://github.github.com/gfm/#strikethrough-extension-
220220
[tables]: https://github.github.com/gfm/#tables-extension-
221221
[task list extension]: https://github.github.com/gfm/#task-list-items-extension-
222-
[`output.html.curly-quotes`]: configuration/renderers.md#html-renderer-options
222+
[`output.html.smart-punctuation`]: configuration/renderers.md#html-renderer-options
223223

224224
### Heading attributes
225225

src/book/mod.rs

+18-12
Original file line numberDiff line numberDiff line change
@@ -70,18 +70,24 @@ impl MDBook {
7070

7171
config.update_from_env();
7272

73-
if config
74-
.html_config()
75-
.map_or(false, |html| html.google_analytics.is_some())
76-
{
77-
warn!(
78-
"The output.html.google-analytics field has been deprecated; \
79-
it will be removed in a future release.\n\
80-
Consider placing the appropriate site tag code into the \
81-
theme/head.hbs file instead.\n\
82-
The tracking code may be found in the Google Analytics Admin page.\n\
83-
"
84-
);
73+
if let Some(html_config) = config.html_config() {
74+
if html_config.google_analytics.is_some() {
75+
warn!(
76+
"The output.html.google-analytics field has been deprecated; \
77+
it will be removed in a future release.\n\
78+
Consider placing the appropriate site tag code into the \
79+
theme/head.hbs file instead.\n\
80+
The tracking code may be found in the Google Analytics Admin page.\n\
81+
"
82+
);
83+
}
84+
if html_config.curly_quotes {
85+
warn!(
86+
"The output.html.curly-quotes field has been renamed to \
87+
output.html.smart-punctuation.\n\
88+
Use the new name in book.toml to remove this warning."
89+
);
90+
}
8591
}
8692

8793
if log_enabled!(log::Level::Trace) {

src/config.rs

+46-5
Original file line numberDiff line numberDiff line change
@@ -526,7 +526,9 @@ pub struct HtmlConfig {
526526
/// The theme to use if the browser requests the dark version of the site.
527527
/// Defaults to 'navy'.
528528
pub preferred_dark_theme: Option<String>,
529-
/// Use "smart quotes" instead of the usual `"` character.
529+
/// Supports smart quotes, apostrophes, ellipsis, en-dash, and em-dash.
530+
pub smart_punctuation: bool,
531+
/// Deprecated alias for `smart_punctuation`.
530532
pub curly_quotes: bool,
531533
/// Should mathjax be enabled?
532534
pub mathjax_support: bool,
@@ -590,6 +592,7 @@ impl Default for HtmlConfig {
590592
theme: None,
591593
default_theme: None,
592594
preferred_dark_theme: None,
595+
smart_punctuation: false,
593596
curly_quotes: false,
594597
mathjax_support: false,
595598
copy_fonts: true,
@@ -623,6 +626,11 @@ impl HtmlConfig {
623626
None => root.join("theme"),
624627
}
625628
}
629+
630+
/// Returns `true` if smart punctuation is enabled.
631+
pub fn smart_punctuation(&self) -> bool {
632+
self.smart_punctuation || self.curly_quotes
633+
}
626634
}
627635

628636
/// Configuration for how to render the print icon, print.html, and print.css.
@@ -798,7 +806,7 @@ mod tests {
798806
[output.html]
799807
theme = "./themedir"
800808
default-theme = "rust"
801-
curly-quotes = true
809+
smart-punctuation = true
802810
google-analytics = "123456"
803811
additional-css = ["./foo/bar/baz.css"]
804812
git-repository-url = "https://foo.com/"
@@ -845,7 +853,7 @@ mod tests {
845853
runnable: true,
846854
};
847855
let html_should_be = HtmlConfig {
848-
curly_quotes: true,
856+
smart_punctuation: true,
849857
google_analytics: Some(String::from("123456")),
850858
additional_css: vec![PathBuf::from("./foo/bar/baz.css")],
851859
theme: Some(PathBuf::from("./themedir")),
@@ -1025,7 +1033,7 @@ mod tests {
10251033
[output.html]
10261034
destination = "my-book" # the output files will be generated in `root/my-book` instead of `root/book`
10271035
theme = "my-theme"
1028-
curly-quotes = true
1036+
smart-punctuation = true
10291037
google-analytics = "123456"
10301038
additional-css = ["custom.css", "custom2.css"]
10311039
additional-js = ["custom.js"]
@@ -1050,7 +1058,7 @@ mod tests {
10501058

10511059
let html_should_be = HtmlConfig {
10521060
theme: Some(PathBuf::from("my-theme")),
1053-
curly_quotes: true,
1061+
smart_punctuation: true,
10541062
google_analytics: Some(String::from("123456")),
10551063
additional_css: vec![PathBuf::from("custom.css"), PathBuf::from("custom2.css")],
10561064
additional_js: vec![PathBuf::from("custom.js")],
@@ -1320,4 +1328,37 @@ mod tests {
13201328
assert!(html_config.print.enable);
13211329
assert!(!html_config.print.page_break);
13221330
}
1331+
1332+
#[test]
1333+
fn curly_quotes_or_smart_punctuation() {
1334+
let src = r#"
1335+
[book]
1336+
title = "mdBook Documentation"
1337+
1338+
[output.html]
1339+
smart-punctuation = true
1340+
"#;
1341+
let config = Config::from_str(src).unwrap();
1342+
assert_eq!(config.html_config().unwrap().smart_punctuation(), true);
1343+
1344+
let src = r#"
1345+
[book]
1346+
title = "mdBook Documentation"
1347+
1348+
[output.html]
1349+
curly-quotes = true
1350+
"#;
1351+
let config = Config::from_str(src).unwrap();
1352+
assert_eq!(config.html_config().unwrap().smart_punctuation(), true);
1353+
1354+
let src = r#"
1355+
[book]
1356+
title = "mdBook Documentation"
1357+
"#;
1358+
let config = Config::from_str(src).unwrap();
1359+
assert_eq!(
1360+
config.html_config().unwrap_or_default().smart_punctuation(),
1361+
false
1362+
);
1363+
}
13231364
}

src/renderer/html_handlebars/hbs_renderer.rs

+8-4
Original file line numberDiff line numberDiff line change
@@ -54,10 +54,13 @@ impl HtmlHandlebars {
5454
.insert("git_repository_edit_url".to_owned(), json!(edit_url));
5555
}
5656

57-
let content = utils::render_markdown(&ch.content, ctx.html_config.curly_quotes);
57+
let content = utils::render_markdown(&ch.content, ctx.html_config.smart_punctuation());
5858

59-
let fixed_content =
60-
utils::render_markdown_with_path(&ch.content, ctx.html_config.curly_quotes, Some(path));
59+
let fixed_content = utils::render_markdown_with_path(
60+
&ch.content,
61+
ctx.html_config.smart_punctuation(),
62+
Some(path),
63+
);
6164
if !ctx.is_index && ctx.html_config.print.page_break {
6265
// Add page break between chapters
6366
// See https://developer.mozilla.org/en-US/docs/Web/CSS/break-before and https://developer.mozilla.org/en-US/docs/Web/CSS/page-break-before
@@ -164,7 +167,8 @@ impl HtmlHandlebars {
164167
.to_string()
165168
}
166169
};
167-
let html_content_404 = utils::render_markdown(&content_404, html_config.curly_quotes);
170+
let html_content_404 =
171+
utils::render_markdown(&content_404, html_config.smart_punctuation());
168172

169173
let mut data_404 = data.clone();
170174
let base_url = if let Some(site_url) = &html_config.site_url {

src/utils/mod.rs

+10-6
Original file line numberDiff line numberDiff line change
@@ -190,26 +190,30 @@ fn adjust_links<'a>(event: Event<'a>, path: Option<&Path>) -> Event<'a> {
190190
}
191191

192192
/// Wrapper around the pulldown-cmark parser for rendering markdown to HTML.
193-
pub fn render_markdown(text: &str, curly_quotes: bool) -> String {
194-
render_markdown_with_path(text, curly_quotes, None)
193+
pub fn render_markdown(text: &str, smart_punctuation: bool) -> String {
194+
render_markdown_with_path(text, smart_punctuation, None)
195195
}
196196

197-
pub fn new_cmark_parser(text: &str, curly_quotes: bool) -> Parser<'_> {
197+
pub fn new_cmark_parser(text: &str, smart_punctuation: bool) -> Parser<'_> {
198198
let mut opts = Options::empty();
199199
opts.insert(Options::ENABLE_TABLES);
200200
opts.insert(Options::ENABLE_FOOTNOTES);
201201
opts.insert(Options::ENABLE_STRIKETHROUGH);
202202
opts.insert(Options::ENABLE_TASKLISTS);
203203
opts.insert(Options::ENABLE_HEADING_ATTRIBUTES);
204-
if curly_quotes {
204+
if smart_punctuation {
205205
opts.insert(Options::ENABLE_SMART_PUNCTUATION);
206206
}
207207
Parser::new_ext(text, opts)
208208
}
209209

210-
pub fn render_markdown_with_path(text: &str, curly_quotes: bool, path: Option<&Path>) -> String {
210+
pub fn render_markdown_with_path(
211+
text: &str,
212+
smart_punctuation: bool,
213+
path: Option<&Path>,
214+
) -> String {
211215
let mut s = String::with_capacity(text.len() * 3 / 2);
212-
let p = new_cmark_parser(text, curly_quotes);
216+
let p = new_cmark_parser(text, smart_punctuation);
213217
let events = p
214218
.map(clean_codeblock_headers)
215219
.map(|event| adjust_links(event, path))

0 commit comments

Comments
 (0)