Skip to content

Commit

Permalink
plugins: add clickable id tags (anchors) to list items
Browse files Browse the repository at this point in the history
- each text block is now referenceable
- direct linking to each snippet from within backlinks
  • Loading branch information
kouloumos committed Mar 24, 2023
1 parent e714d3a commit bd2e924
Show file tree
Hide file tree
Showing 3 changed files with 87 additions and 1 deletion.
2 changes: 1 addition & 1 deletion _includes/backlinks.html
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ <h3 style="margin-bottom: 1em; color: #828282">{{ page.backlinks.size }} Linked
{% for snippet in backlink.snippets %}
<div class="backlink-box">
<div class="backlink-header">{{ snippet.header }}</div>
{{ snippet.paragraph | markdownify }}
{{ snippet.paragraph | link_to_anchor: backlink.doc.url | markdownify }}
</div>
{% endfor %}
</div>
Expand Down
61 changes: 61 additions & 0 deletions _plugins/auto-anchor.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
# frozen_string_literal: true
# This file is based on code from https://github.com/bitcoinops/bitcoinops.github.io

# Automatically adds id tags (anchors) to list items

require 'digest/md5'

def generate_slug(text)
# Remove double-quotes from titles before attempting to slugify
text.gsub!('"', '')
# Remove whitespace character from the end and use Liquid/Jekyll slugify filter
slug_text = "{{ \"#{text.rstrip}\" | slugify: 'latin' }}"
# use the digest library to create deterministic ids based on text
id = Digest::MD5.hexdigest(slug_text)[0...7]
slug = "\#b#{id}" # prefix with 'b', ids cannot start with a number
slug
end

def generate_anchor_list_link(anchor_link)
# custom clickable bullet linking to an anchor
"<a href=\"#{anchor_link}\" class=\"anchor-list-link\">●</a>"
end

def auto_anchor(content)
# finds “bulleted” list items that start with hyphen (-) or asterisk (*)
# adds anchor and clickable bullet
content.gsub!(/^ *[\*-] .*?(?:\n\n|\z)/m) do |bulleted_paragraph|
slug = generate_slug(bulleted_paragraph)
bullet_character = bulleted_paragraph.match(/^ *([\*-])/)[1] # bullet can be '-' or '*'
id_prefix = "#{bullet_character} {:#{slug} .anchor-list} #{generate_anchor_list_link(slug)}"
bulleted_paragraph.sub!(/#{Regexp.quote(bullet_character)}/, id_prefix)
end
# finds “numbered” list items that start with number (1.)
# adds anchor only
content.gsub!(/^ *\d+\. .*?(?:\n\n|\z)/m) do |numbered_paragraph|
slug = generate_slug(numbered_paragraph)
id_prefix = "1. {:#{slug} .anchor-list .anchor-numbered}"
numbered_paragraph.sub!(/\d+\./, id_prefix)
end
end

## Run automatically on all documents
Jekyll::Hooks.register :documents, :pre_render do |post|
## Don't process documents if YAML headers say: "auto_id: false"
unless post.data["auto_id"] == false
auto_anchor(post.content)
end
end

module TextFilter
# This is a custom filter used in backlinks.html to
# add anchor links to each backlink snippet
def link_to_anchor(text, url)
slug = generate_slug(text)
id_prefix = generate_anchor_list_link("#{url}#{slug}")
text.sub!(/(?:-|\*|\d+\.)/, id_prefix) # this targets both “bulleted” and “numbered” list items
text
end
end

Liquid::Template.register_filter(TextFilter)
25 changes: 25 additions & 0 deletions assets/css/all.sass
Original file line number Diff line number Diff line change
Expand Up @@ -201,3 +201,28 @@ a
font-size: 0.85em
color: #828282

@keyframes highlightfade
from
background: var(--target-color)
to
background: transparent

.anchor-list
list-style: none
&:target
background: transparent
animation-name: highlightfade
animation-duration: 7s

.anchor-numbered
list-style: decimal

.anchor-list-link
color: var(--text-color)
margin-left: -14px
padding-right: 3px
font-size: 0.45em
display: inline-block
position: relative
top: -0.4em /* position the dot vertically in the middle of the container element */
text-decoration: none

0 comments on commit bd2e924

Please sign in to comment.