From 7d7efb0709d82554e0d88c2b3d92838d9b0f6521 Mon Sep 17 00:00:00 2001 From: Nobuyoshi Nakada Date: Wed, 18 Dec 2024 06:48:27 +0900 Subject: [PATCH] Enable cross reference in code (#1240) Some people like to mark up method names in MarkDown style block quotes, like this: ruby/ruby#12333. Currently, no links are created in the code in the RDoc, but such words most likely refer to methods. This PR makes a word a code cross-reference if the whole word can be resolved as a reference. --- lib/rdoc/markup/formatter.rb | 31 +++++++++------ lib/rdoc/markup/to_html_crossref.rb | 39 +++++++++++++++++++ .../rdoc/test_rdoc_markup_to_html_crossref.rb | 12 ++++++ 3 files changed, 70 insertions(+), 12 deletions(-) diff --git a/lib/rdoc/markup/formatter.rb b/lib/rdoc/markup/formatter.rb index 9daffaabb8..fb382df680 100644 --- a/lib/rdoc/markup/formatter.rb +++ b/lib/rdoc/markup/formatter.rb @@ -195,18 +195,20 @@ def in_tt? @in_tt > 0 end + def tt_tag? attr_mask, reverse = false + each_attr_tag(attr_mask, reverse) do |tag| + return true if tt? tag + end + false + end + ## # Turns on tags for +item+ on +res+ def on_tags res, item - attr_mask = item.turn_on - return if attr_mask.zero? - - @attr_tags.each do |tag| - if attr_mask & tag.bit != 0 then - res << annotate(tag.on) - @in_tt += 1 if tt? tag - end + each_attr_tag(item.turn_on) do |tag| + res << annotate(tag.on) + @in_tt += 1 if tt? tag end end @@ -214,13 +216,18 @@ def on_tags res, item # Turns off tags for +item+ on +res+ def off_tags res, item - attr_mask = item.turn_off + each_attr_tag(item.turn_off, true) do |tag| + @in_tt -= 1 if tt? tag + res << annotate(tag.off) + end + end + + def each_attr_tag attr_mask, reverse = false return if attr_mask.zero? - @attr_tags.reverse_each do |tag| + @attr_tags.public_send(reverse ? :reverse_each : :each) do |tag| if attr_mask & tag.bit != 0 then - @in_tt -= 1 if tt? tag - res << annotate(tag.off) + yield tag end end end diff --git a/lib/rdoc/markup/to_html_crossref.rb b/lib/rdoc/markup/to_html_crossref.rb index 172518c7f9..e6661b621e 100644 --- a/lib/rdoc/markup/to_html_crossref.rb +++ b/lib/rdoc/markup/to_html_crossref.rb @@ -182,4 +182,43 @@ def link name, text, code = true, rdoc_ref: false end end + def convert_flow(flow) + res = [] + + i = 0 + while i < flow.size + item = flow[i] + i += 1 + case item + when RDoc::Markup::AttrChanger then + # Make "+Class#method+" a cross reference + if tt_tag?(item.turn_on) and + String === (str = flow[i]) and + RDoc::Markup::AttrChanger === flow[i+1] and + tt_tag?(flow[i+1].turn_off, true) and + (@options.hyperlink_all ? ALL_CROSSREF_REGEXP : CROSSREF_REGEXP).match?(str) and + (text = cross_reference str) != str + then + text = yield text, res if defined?(yield) + res << text + i += 2 + next + end + off_tags res, item + on_tags res, item + when String then + text = convert_string(item) + text = yield text, res if defined?(yield) + res << text + when RDoc::Markup::RegexpHandling then + text = convert_regexp_handling(item) + text = yield text, res if defined?(yield) + res << text + else + raise "Unknown flow element: #{item.inspect}" + end + end + + res.join('') + end end diff --git a/test/rdoc/test_rdoc_markup_to_html_crossref.rb b/test/rdoc/test_rdoc_markup_to_html_crossref.rb index d5817560e2..fb0aca2909 100644 --- a/test/rdoc/test_rdoc_markup_to_html_crossref.rb +++ b/test/rdoc/test_rdoc_markup_to_html_crossref.rb @@ -16,6 +16,18 @@ def test_convert_CROSSREF result = @to.convert 'C1' assert_equal para("C1"), result + + result = @to.convert '+C1+' + assert_equal para("C1"), result + + result = @to.convert 'FOO' + assert_equal para("FOO"), result + + result = @to.convert '+FOO+' + assert_equal para("FOO"), result + + result = @to.convert '# :stopdoc::' + assert_equal para("# :stopdoc::"), result end def test_convert_CROSSREF_method