<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom"><channel><title>Productivity on Sven Koschnicke</title><link>https://sven.guru/tags/productivity/</link><description>Recent content in Productivity on Sven Koschnicke</description><generator>Hugo</generator><language>en</language><lastBuildDate>Wed, 13 May 2026 00:00:00 +0000</lastBuildDate><atom:link href="https://sven.guru/tags/productivity/index.xml" rel="self" type="application/rss+xml"/><item><title>Inline Preview for Source File Links in Emacs</title><link>https://sven.guru/posts/emacs-file-link-preview/</link><pubDate>Wed, 13 May 2026 00:00:00 +0000</pubDate><guid>https://sven.guru/posts/emacs-file-link-preview/</guid><description>&lt;p&gt;While working, I constantly take notes using org-mode. Because I often work with code, I extensively use the ability of org-mode to link directly into code files when I need to reference where something can be found.
Something like &lt;code&gt;[[file:~/src/foo/bar.el::42][the bit that parses the header]]&lt;/code&gt;. I can easily follow this link and have the file open in emacs, but in the note itself the link is just some underlined text. To remind myself what is behind the link I have to jump there, look, and jump back. That gets annoying.&lt;/p&gt;
&lt;p&gt;What I really wanted was for org to show me the relevant lines &lt;em&gt;right below
the link&lt;/em&gt;, the way it already inlines images. Here is what the same link
looks like before and after:&lt;/p&gt;
&lt;figure&gt;&lt;img src="https://sven.guru/ox-hugo/emacs-link.png"&gt;
&lt;/figure&gt;
&lt;figure&gt;&lt;img src="https://sven.guru/ox-hugo/emacs-link-shown.png"&gt;
&lt;/figure&gt;
&lt;h2 id="the-hook-org-already-gives-us"&gt;The hook org already gives us&lt;/h2&gt;
&lt;p&gt;Org 9.7 ships &lt;code&gt;org-link-preview&lt;/code&gt; natively, but out of the box it only knows
how to preview images. The good news is that the mechanism is generic: every
link type can register a &lt;code&gt;:preview&lt;/code&gt; function via &lt;code&gt;org-link-set-parameters&lt;/code&gt;,
and org will call it with an overlay it has already placed on top of the
link. Whatever the function puts on that overlay is what the user sees.&lt;/p&gt;
&lt;p&gt;That is the entire trick. The rest of this post is just filling in the
preview function for &lt;code&gt;file:&lt;/code&gt; and &lt;code&gt;attachment:&lt;/code&gt; links so that, when the link
points at a specific line, we render a syntax-highlighted snippet around it.&lt;/p&gt;
&lt;h2 id="resolving-the-link-to-a-snippet"&gt;Resolving the link to a snippet&lt;/h2&gt;
&lt;p&gt;A file link in org can carry a &lt;em&gt;search option&lt;/em&gt; after &lt;code&gt;::&lt;/code&gt;. It can be a line
number (&lt;code&gt;foo.el::42&lt;/code&gt;) or a piece of text (&lt;code&gt;foo.el::defun my-thing&lt;/code&gt;) — org
uses it when following the link to jump to the right spot, and we want to
use it as the anchor for the preview.&lt;/p&gt;
&lt;p&gt;The helpers below do four things: turn a text search into a line number,
guess the source-block language from the file extension (so the snippet gets
the right fontification when re-inserted), apply font-lock to the extracted
text, and finally format the result with a 👉 marker on the target line so
it&amp;rsquo;s obvious which line the link actually points at.&lt;/p&gt;
&lt;div class="highlight"&gt;&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;"&gt;&lt;code class="language-emacs-lisp" data-lang="emacs-lisp"&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;(defcustom my/org-link-preview-context-lines &lt;span style="color:#ae81ff"&gt;5&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;Number of context lines to show before and after target line.&amp;#34;&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; :type &lt;span style="color:#e6db74"&gt;&amp;#39;integer&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; :group &lt;span style="color:#e6db74"&gt;&amp;#39;org-link&lt;/span&gt;)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;(defun my/org-link-find-line-by-text (file search-text)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;Find line number in FILE that contains SEARCH-TEXT.
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#e6db74"&gt;Returns line number or nil if not found.&amp;#34;&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (when (and file (&lt;span style="color:#a6e22e"&gt;file-exists-p&lt;/span&gt; file) search-text)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (with-temp-buffer
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;insert-file-contents&lt;/span&gt; file)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;goto-char&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;point-min&lt;/span&gt;))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (when (&lt;span style="color:#a6e22e"&gt;search-forward&lt;/span&gt; search-text &lt;span style="color:#66d9ef"&gt;nil&lt;/span&gt; &lt;span style="color:#66d9ef"&gt;t&lt;/span&gt;)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (line-number-at-pos)))))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;(defun my/org-link-get-language-from-extension (file)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;Get org-babel language identifier from FILE extension.&amp;#34;&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (let* ((mode (assoc-default file auto-mode-alist &lt;span style="color:#e6db74"&gt;&amp;#39;string-match&lt;/span&gt;))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (mode-name (and mode (&lt;span style="color:#a6e22e"&gt;symbol-name&lt;/span&gt; mode))))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (if mode-name
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (let ((lang (replace-regexp-in-string &lt;span style="color:#e6db74"&gt;&amp;#34;-mode$&amp;#34;&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;&amp;#34;&lt;/span&gt; mode-name)))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (cond
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; ((&lt;span style="color:#a6e22e"&gt;assoc&lt;/span&gt; lang org-src-lang-modes) lang)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; ((string= lang &lt;span style="color:#e6db74"&gt;&amp;#34;js&amp;#34;&lt;/span&gt;) &lt;span style="color:#e6db74"&gt;&amp;#34;javascript&amp;#34;&lt;/span&gt;)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#66d9ef"&gt;t&lt;/span&gt; lang)))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (or (file-name-extension file) &lt;span style="color:#e6db74"&gt;&amp;#34;text&amp;#34;&lt;/span&gt;))))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;(defun my/org-link-apply-syntax-highlighting (content file)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;Apply syntax highlighting to CONTENT based on FILE&amp;#39;s major mode.&amp;#34;&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (with-temp-buffer
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;insert&lt;/span&gt; content)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (let ((mode (assoc-default file auto-mode-alist &lt;span style="color:#e6db74"&gt;&amp;#39;string-match&lt;/span&gt;)))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (when (and mode (&lt;span style="color:#a6e22e"&gt;fboundp&lt;/span&gt; mode))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (ignore-errors
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;funcall&lt;/span&gt; mode)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (font-lock-ensure))))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;buffer-string&lt;/span&gt;)))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;(defun my/org-link-format-preview-content (content start-line target-line)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;Format CONTENT as preview with optional TARGET-LINE highlighting.&amp;#34;&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (with-temp-buffer
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;insert&lt;/span&gt; content)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;goto-char&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;point-min&lt;/span&gt;))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (let ((current-line start-line)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (result &lt;span style="color:#e6db74"&gt;&amp;#34;&amp;#34;&lt;/span&gt;))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (while (not (&lt;span style="color:#a6e22e"&gt;eobp&lt;/span&gt;))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (let* ((line-text (&lt;span style="color:#a6e22e"&gt;buffer-substring&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;point&lt;/span&gt;) (&lt;span style="color:#a6e22e"&gt;line-end-position&lt;/span&gt;)))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (is-target (and target-line (&lt;span style="color:#a6e22e"&gt;=&lt;/span&gt; current-line target-line))))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (setq result
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;concat&lt;/span&gt; result
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (if is-target
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;propertize&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;concat&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;👉 &amp;#34;&lt;/span&gt; line-text &lt;span style="color:#e6db74"&gt;&amp;#34;\n&amp;#34;&lt;/span&gt;) &lt;span style="color:#e6db74"&gt;&amp;#39;face&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#39;hl-line&lt;/span&gt;)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;concat&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34; &amp;#34;&lt;/span&gt; line-text &lt;span style="color:#e6db74"&gt;&amp;#34;\n&amp;#34;&lt;/span&gt;))))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (setq current-line (&lt;span style="color:#a6e22e"&gt;1+&lt;/span&gt; current-line))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;forward-line&lt;/span&gt; &lt;span style="color:#ae81ff"&gt;1&lt;/span&gt;)))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; result)))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;(defun my/org-link-get-file-preview (file target-line)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;Get preview text for FILE centered around TARGET-LINE.
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#e6db74"&gt;Returns propertized string formatted as an org source block, or nil.&amp;#34;&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (when (and file (&lt;span style="color:#a6e22e"&gt;file-exists-p&lt;/span&gt; file) target-line)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (with-temp-buffer
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;insert-file-contents&lt;/span&gt; file)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (let* ((language (my/org-link-get-language-from-extension file))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (start-line (&lt;span style="color:#a6e22e"&gt;max&lt;/span&gt; &lt;span style="color:#ae81ff"&gt;1&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;-&lt;/span&gt; target-line my/org-link-preview-context-lines)))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (end-line (&lt;span style="color:#a6e22e"&gt;+&lt;/span&gt; target-line my/org-link-preview-context-lines)))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;goto-char&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;point-min&lt;/span&gt;))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;forward-line&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;1-&lt;/span&gt; start-line))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (let* ((content-start (&lt;span style="color:#a6e22e"&gt;point&lt;/span&gt;))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (content-end (progn (&lt;span style="color:#a6e22e"&gt;forward-line&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;-&lt;/span&gt; end-line start-line &lt;span style="color:#ae81ff"&gt;-1&lt;/span&gt;)) (&lt;span style="color:#a6e22e"&gt;point&lt;/span&gt;)))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (content (&lt;span style="color:#a6e22e"&gt;buffer-substring&lt;/span&gt; content-start content-end))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (highlighted (my/org-link-apply-syntax-highlighting content file))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (formatted (my/org-link-format-preview-content
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; highlighted start-line target-line))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (line-info (&lt;span style="color:#a6e22e"&gt;format&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34; :file %s :line %d&amp;#34;&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;file-name-nondirectory&lt;/span&gt; file) target-line)))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;concat&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;propertize&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;format&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;#+begin_src %s%s\n&amp;#34;&lt;/span&gt; language line-info)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#e6db74"&gt;&amp;#39;face&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#39;org-block-begin-line&lt;/span&gt;)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; formatted
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;propertize&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;#+end_src&amp;#34;&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#39;face&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#39;org-block-end-line&lt;/span&gt;)))))))
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;There&amp;rsquo;s nothing clever in there — the only thing worth pointing at is that
the snippet is wrapped in a fake &lt;code&gt;#+begin_src&lt;/code&gt; / &lt;code&gt;#+end_src&lt;/code&gt; pair with the
filename and line baked into the header. That way the rendered overlay looks
exactly like a normal org source block, getting nice syntax highlighting in the process.&lt;/p&gt;
&lt;h2 id="wiring-it-into-org-link-preview"&gt;Wiring it into org-link-preview&lt;/h2&gt;
&lt;p&gt;A &lt;code&gt;:preview&lt;/code&gt; function receives &lt;code&gt;(OV PATH LINK)&lt;/code&gt;. The contract is: configure
the overlay &lt;code&gt;OV&lt;/code&gt; however you like and return non-nil to keep it, or return
nil to have org throw the overlay away. We render the snippet via
&lt;code&gt;after-string&lt;/code&gt; so it appears &lt;em&gt;below&lt;/em&gt; the link instead of replacing it —
keeping the original link visible matters, because there might also be
information in the link description.&lt;/p&gt;
&lt;p&gt;The function only kicks in when the link has a search option. If there
isn&amp;rsquo;t one, there&amp;rsquo;s nothing to anchor the preview to, so we fall through to
org&amp;rsquo;s built-in image previewer. That way PNGs and friends still inline as
before, and we register the same dispatcher for both &lt;code&gt;file:&lt;/code&gt; and
&lt;code&gt;attachment:&lt;/code&gt; links so org-attach works too.&lt;/p&gt;
&lt;div class="highlight"&gt;&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;"&gt;&lt;code class="language-emacs-lisp" data-lang="emacs-lisp"&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;(defun my/org-link-preview-source-file (ov path link)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;Preview a source-file link as a snippet over overlay OV.
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#e6db74"&gt;Handles file links with a numeric (::42) or text (::needle) search
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#e6db74"&gt;option. Returns non-nil on success, nil to let org fall back.&amp;#34;&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (let ((search-option (org-element-property :search-option link)))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (when search-option
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (let* ((file (&lt;span style="color:#a6e22e"&gt;expand-file-name&lt;/span&gt; path))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (line (if (&lt;span style="color:#a6e22e"&gt;string-match&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;\\`\\([0-9]+\\)\\&amp;#39;&amp;#34;&lt;/span&gt; search-option)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;string-to-number&lt;/span&gt; (match-string &lt;span style="color:#ae81ff"&gt;1&lt;/span&gt; search-option))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (my/org-link-find-line-by-text file search-option)))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (preview (my/org-link-get-file-preview file line)))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (when preview
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;overlay-put&lt;/span&gt; ov &lt;span style="color:#e6db74"&gt;&amp;#39;after-string&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;concat&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;\n&amp;#34;&lt;/span&gt; preview &lt;span style="color:#e6db74"&gt;&amp;#34;\n&amp;#34;&lt;/span&gt;))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;overlay-put&lt;/span&gt; ov &lt;span style="color:#e6db74"&gt;&amp;#39;face&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#39;default&lt;/span&gt;)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#66d9ef"&gt;t&lt;/span&gt;)))))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;(defun my/org-link-preview-file-dispatch (ov path link)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;Try source-file preview first, fall back to org&amp;#39;s image previewer.&amp;#34;&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (or (my/org-link-preview-source-file ov path link)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (org-link-preview-file ov path link)))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;(with-eval-after-load &lt;span style="color:#e6db74"&gt;&amp;#39;ol&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (org-link-set-parameters &lt;span style="color:#e6db74"&gt;&amp;#34;file&amp;#34;&lt;/span&gt; :preview &lt;span style="color:#a6e22e"&gt;#&amp;#39;&lt;/span&gt;my/org-link-preview-file-dispatch)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (org-link-set-parameters &lt;span style="color:#e6db74"&gt;&amp;#34;attachment&amp;#34;&lt;/span&gt; :preview &lt;span style="color:#a6e22e"&gt;#&amp;#39;&lt;/span&gt;my/org-link-preview-file-dispatch))
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;h2 id="one-small-papercut-links-with-descriptions"&gt;One small papercut: links with descriptions&lt;/h2&gt;
&lt;p&gt;There&amp;rsquo;s one last annoyance. By default &lt;code&gt;org-link-preview&lt;/code&gt; skips links that
have a description — which is, of course, most of the links I actually want
previewed, because I tend to write &lt;code&gt;[[file:foo.el::42][the parser]]&lt;/code&gt; rather
than dumping the raw path into the buffer. The fix is a one-line wrapper
that passes the &lt;code&gt;include-linked&lt;/code&gt; prefix argument so the link at point always
gets previewed, plus a Spacemacs binding under &lt;code&gt;SPC m l p&lt;/code&gt;. To clear a
preview, &lt;code&gt;C-u M-x org-link-preview&lt;/code&gt; still works as usual.&lt;/p&gt;
&lt;div class="highlight"&gt;&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;"&gt;&lt;code class="language-emacs-lisp" data-lang="emacs-lisp"&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;(defun my/org-link-preview-here ()
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;Preview the link at point, including links with descriptions.&amp;#34;&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (interactive)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (org-link-preview &lt;span style="color:#ae81ff"&gt;1&lt;/span&gt;))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;(with-eval-after-load &lt;span style="color:#e6db74"&gt;&amp;#39;org&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (spacemacs/set-leader-keys-for-major-mode &lt;span style="color:#e6db74"&gt;&amp;#39;org-mode&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;lp&amp;#34;&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#39;my/org-link-preview-here&lt;/span&gt;))
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;h2 id="another-step-towards-a-better-editor"&gt;Another step towards a better editor&lt;/h2&gt;
&lt;p&gt;There is room for improvement here. It doesn&amp;rsquo;t work if there are multiple
matching texts or no position in the link at all. But I can now read my notes
without context switching to the source file. It&amp;rsquo;s also satisfying to have used
an extension point that was designed for this kind of thing instead of creating
everything from scratch (which was what I had before). Just some small
customization making the editor I spend most of my day in a bit more at home.&lt;/p&gt;</description></item></channel></rss>