Skip to content

Commit e665505

Browse files
authored
Merge pull request #1775 from ehuss/rule-ref-link
Add the ability for rules to be specified in link definitions
2 parents d1e0e0a + f37173d commit e665505

File tree

2 files changed

+35
-1
lines changed

2 files changed

+35
-1
lines changed

mdbook-spec/src/lib.rs

+33
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,10 @@ static ADMONITION_RE: Lazy<Regex> = Lazy::new(|| {
2424
Regex::new(r"(?m)^ *> \[!(?<admon>[^]]+)\]\n(?<blockquote>(?: *>.*\n)+)").unwrap()
2525
});
2626

27+
/// A primitive regex to find link reference definitions.
28+
static MD_LINK_REFERENCE_DEFINITION: Lazy<Regex> =
29+
Lazy::new(|| Regex::new(r"(?m)^\[(?<label>[^]]+)]: (?<dest>.*)").unwrap());
30+
2731
pub fn handle_preprocessing() -> Result<(), Error> {
2832
let pre = Spec::new(None)?;
2933
let (ctx, book) = CmdPreprocessor::parse_input(io::stdin())?;
@@ -114,6 +118,34 @@ impl Spec {
114118
Ok(Spec { rust_root })
115119
}
116120

121+
/// Converts link reference definitions that point to a rule to the correct link.
122+
///
123+
/// For example:
124+
/// ```markdown
125+
/// See [this rule].
126+
///
127+
/// [this rule]: expr.array
128+
/// ```
129+
///
130+
/// This will convert the `[this rule]` definition to point to the actual link.
131+
fn rule_link_references(&self, chapter: &Chapter, rules: &Rules) -> String {
132+
let current_path = chapter.path.as_ref().unwrap().parent().unwrap();
133+
MD_LINK_REFERENCE_DEFINITION
134+
.replace_all(&chapter.content, |caps: &Captures<'_>| {
135+
let dest = &caps["dest"];
136+
if let Some((_source_path, path)) = rules.def_paths.get(dest) {
137+
let label = &caps["label"];
138+
let relative = pathdiff::diff_paths(path, current_path).unwrap();
139+
// Adjust paths for Windows.
140+
let relative = relative.display().to_string().replace('\\', "/");
141+
format!("[{label}]: {relative}#r-{dest}")
142+
} else {
143+
caps.get(0).unwrap().as_str().to_string()
144+
}
145+
})
146+
.to_string()
147+
}
148+
117149
/// Generates link references to all rules on all pages, so you can easily
118150
/// refer to rules anywhere in the book.
119151
fn auto_link_references(&self, chapter: &Chapter, rules: &Rules) -> String {
@@ -255,6 +287,7 @@ impl Preprocessor for Spec {
255287
return;
256288
}
257289
ch.content = self.admonitions(&ch, &mut diag);
290+
ch.content = self.rule_link_references(&ch, &rules);
258291
ch.content = self.auto_link_references(&ch, &rules);
259292
ch.content = self.render_rule_definitions(&ch.content, &tests, &git_ref);
260293
if ch.name == "Test summary" {

src/introduction.md

+2-1
Original file line numberDiff line numberDiff line change
@@ -116,7 +116,7 @@ These conventions are documented here.
116116
See [Notation] for more detail.
117117

118118
r[example.rule.label]
119-
* Rule identifiers appear before each language rule enclosed in square brackets. These identifiers provide a way to refer to a specific rule in the language. The rule identifier uses periods to separate sections from most general to most specific ([destructors.scope.nesting.function-body] for example). On narrow screens, the rule name will collapse to display `[*]`.
119+
* Rule identifiers appear before each language rule enclosed in square brackets. These identifiers provide a way to refer to and link to a specific rule in the language ([e.g.][example rule]). The rule identifier uses periods to separate sections from most general to most specific ([destructors.scope.nesting.function-body] for example). On narrow screens, the rule name will collapse to display `[*]`.
120120

121121
The rule name can be clicked to link to that rule.
122122

@@ -144,6 +144,7 @@ We also want the reference to be as normative as possible, so if you see anythin
144144
[_Expression_]: expressions.md
145145
[cargo book]: ../cargo/index.html
146146
[cargo reference]: ../cargo/reference/index.html
147+
[example rule]: example.rule.label
147148
[expressions chapter]: expressions.html
148149
[file an issue]: https://github.com/rust-lang/reference/issues
149150
[lifetime of temporaries]: expressions.html#temporaries

0 commit comments

Comments
 (0)