156 lines
9.5 KiB
HTML
156 lines
9.5 KiB
HTML
<!DOCTYPE html><html lang="en"><head><meta charset="utf-8"/><meta author="Alex Mikhailov"/><meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no"/><meta name="color-scheme" content="light dark"/><meta http-equiv="content-language" content="en-us"/><meta name="description" content="Add post preview for OpenGraph cards"/><meta property="og:description" content="Add post preview for OpenGraph cards"/><meta property="og:image" content="https://fidonode.me/resources/images/preview/posts/posts_preview.org.png"/><meta property="og:title" content="Posts preview"/><meta name="twitter:description" content="Add post preview for OpenGraph cards"/><meta name="twitter:title" content="Posts preview"/><meta name="twitter:image" content="https://fidonode.me/resources/images/preview/posts/posts_preview.org.png"/><meta name="twitter:card" content="summary_large_image"/><link rel="icon" type="image/x-icon" href="/resources/favicon.ico"/><link rel="stylesheet" type="text/css" href="/resources/css/pico.sand.min.css"/><script defer="true" src="https://umami.dokutsu.xyz/script.js" data-website-id="d52d9af1-0c7d-4531-84c6-0b9c2850011f"></script><title>Posts preview</title><link id="highlight-theme" rel="stylesheet" type="text/css"/><script src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/highlight.min.js"></script><script src="https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.9.0/build/languages/bash.min.js"></script><script src="https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.9.0/build/languages/lisp.min.js"></script><script src="/resources/js/theme-selector.js"></script></head><body><header class="header"><div class="container"><nav><ul><li><strong>Alex Mikhailov</strong></li></ul><ul><li><a href="/index.html">About</a></li><li><a href="/posts.html">Blog</a></li><li><a href="/rss.xml">RSS</a></li></ul></nav></div></header><main class="container blog-post"><hgroup><h1>Posts preview</h1><p>Add post preview for OpenGraph cards</p><nav><ul><li>Tags:</li><li><mark><a href="/tags/@org-mode.html" class="secondary">@org-mode</a></mark></li><li><mark><a href="/tags/@elisp.html" class="secondary">@elisp</a></mark></li><li><mark><a href="/tags/@imagemagick.html" class="secondary">@imagemagick</a></mark></li></ul></nav></hgroup><div id="table-of-contents" role="doc-toc">
|
|
<h2>Table of Contents</h2>
|
|
<div id="text-table-of-contents" role="doc-toc">
|
|
<ul>
|
|
<li><a href="#org746ed86">What is imagemagick</a></li>
|
|
<li><a href="#org844ff86">Basic idea of this process</a></li>
|
|
<li><a href="#org1f6582b">Integrate into build</a></li>
|
|
<li><a href="#orgfa0e61f">Whats next?</a>
|
|
<ul>
|
|
<li><a href="#orgea7d693">Tags</a></li>
|
|
<li><a href="#org649cc65">Post series</a></li>
|
|
<li><a href="#org75908b7">Adopt/fix htmlize.el</a></li>
|
|
<li><a href="#orgaf1ec94">Show more meta on posts index page.</a></li>
|
|
</ul>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
</div>
|
|
<div id="outline-container-org746ed86" class="outline-2">
|
|
<h2 id="org746ed86">What is imagemagick</h2>
|
|
<div class="outline-text-2" id="text-org746ed86">
|
|
<p>
|
|
<a href="https://imagemagick.org">Imagemagick</a> is a ffmpeg of the image world. You can do a lot of fun things with it. For example you can take a picture, cut corners on it, place it on top of another image, add some text and get final result. So it is looks like a good tool for making previews from code.
|
|
</p>
|
|
</div>
|
|
</div>
|
|
<div id="outline-container-org844ff86" class="outline-2">
|
|
<h2 id="org844ff86">Basic idea of this process</h2>
|
|
<div class="outline-text-2" id="text-org844ff86">
|
|
<p>
|
|
I want to hook a process of rendering post. Since this function called on each run of publishing and for each post it is a good idea to cache resulting images. I gonna simply check presence of preview image and use it as guard for running image generation. After that I gonna extract <code>#+TITLE</code> and <code>#+DESCRIPTION</code> properties from Org file. Each Org file I have, has next header:
|
|
</p>
|
|
<pre><code class="language-plaintext">#+TITLE: Posts preview
|
|
#+DATE: &lt;2024-06-28 Fri&gt;
|
|
#+DESCRIPTION: Add post preview for OpenGraph cards
|
|
#+TAGS: @blogging @org-mode @elisp @imagemagick
|
|
</code></pre>
|
|
<p>
|
|
So I with all these data I can generate my previews with simple script:
|
|
</p>
|
|
<pre><code class="language-bash"> #!/bin/bash
|
|
magick $4/with_profile.png \
|
|
\( -size 850x150 \-background none -font Roboto-Regular -fill '#FEFEFE' label:"$1" -trim -gravity center -extent 850x150 \) \
|
|
-gravity northeast -geometry +50+75 -composite /tmp/text_main.png
|
|
magick /tmp/text_main.png \( -size 1100x300 -background none -font Roboto-Regular -pointsize 50 -fill '#FFFFFF' pango:"$2" -trim -gravity center -extent 1100x300 \) -gravity east -geometry +40+140 -composite "$3"
|
|
#
|
|
</code></pre>
|
|
</div>
|
|
</div>
|
|
|
|
<div id="outline-container-org1f6582b" class="outline-2">
|
|
<h2 id="org1f6582b">Integrate into build</h2>
|
|
<div class="outline-text-2" id="text-org1f6582b">
|
|
<p>
|
|
As I already mention I gonna skip file generation when file already here. Here is the whole function. Pretty simple. Just prepare pathes, check some dependencies, create pathes and execute script which calls imagemagick.
|
|
</p>
|
|
<pre><code class="language-lisp">(defun my/render-preview (file-name title description)
|
|
(let* ((has-imagemagick (executable-find "magick"))
|
|
(full-file-path (file-truename(format "%s%s/resources/images/preview/%s.png" script-directory my/blog-src-path file-name )))
|
|
(file-dir (file-name-directory full-file-path))
|
|
(has-dir (file-directory-p file-dir))
|
|
(has-file (file-exists-p full-file-path))
|
|
(path-to-script-root (format "%shelpers" script-directory))
|
|
(path-to-script (format "%s/og_image_gen.sh" path-to-script-root)))
|
|
(if (and has-imagemagick
|
|
(and description (not (string= description ""))))
|
|
(progn
|
|
(when (not has-file)
|
|
(progn
|
|
(when (not has-dir)
|
|
(make-directory file-dir t))
|
|
(shell-command (format "bash '%s' '%s' '%s' '%s' '%s'" path-to-script title description full-file-path path-to-script-root))
|
|
)
|
|
))
|
|
(message "Imagemagick is not installed. Preview generation skipped.")
|
|
)))
|
|
</code></pre>
|
|
|
|
<p>
|
|
And here is the part of <code>my/template</code> function related to OpenGraph meta tags.
|
|
</p>
|
|
<pre><code class="language-lisp">(defun my/template (contents info)
|
|
|
|
(let* ((title-str (org-export-data (plist-get info :title) info))
|
|
(description-str (org-export-data (plist-get info :description) info))
|
|
(file-path-str (org-export-data (plist-get info :input-file) info))
|
|
(base-directory-str (org-export-data (plist-get info :base-directory) info))
|
|
(file-name-str (file-relative-name file-path-str (format "%s/%s" script-directory base-directory-str)))
|
|
(img-link-str (format "%s/resources/images/preview/%s.png" my/url file-name-str))
|
|
|
|
(my/render-preview file-name-str title-str description-str)
|
|
|
|
(set-text-properties 0 (length title-str) nil title-str)
|
|
(set-text-properties 0 (length description-str) nil description-str)
|
|
(set-text-properties 0 (length img-link-str) nil img-link-str)
|
|
...
|
|
;; OG block
|
|
(meta ((name . "description") (content . ,description-str)))
|
|
(meta ((name . "og:description") (content . ,description-str)))
|
|
(meta ((name . "twitter:description") (content . ,description-str)))
|
|
|
|
(meta ((name . "og:image") (content . ,img-link-str)))
|
|
(meta ((name . "twitter:image") (content . ,img-link-str)))
|
|
|
|
(meta ((name . "og:title") (content . ,title-str)))
|
|
(meta ((name . "twitter:title") (content . ,title-str)))
|
|
|
|
(meta ((name . "twitter:card") (content . "summary_large_image")))
|
|
|
|
</code></pre>
|
|
<p>
|
|
You can check whole function in previous post <a href="./improve_code_blocks.html">Improve code blocks.</a>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
<div id="outline-container-orgfa0e61f" class="outline-2">
|
|
<h2 id="orgfa0e61f">Whats next?</h2>
|
|
<div class="outline-text-2" id="text-orgfa0e61f">
|
|
</div>
|
|
<div id="outline-container-orgea7d693" class="outline-3">
|
|
<h3 id="orgea7d693">Tags</h3>
|
|
<div class="outline-text-3" id="text-orgea7d693">
|
|
<p>
|
|
Show tags, show posts by tag.
|
|
<a href="./blog_index_and_tags_automation.html">Blog index and tags automation</a>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
<div id="outline-container-org649cc65" class="outline-3">
|
|
<h3 id="org649cc65">Post series</h3>
|
|
<div class="outline-text-3" id="text-org649cc65">
|
|
<p>
|
|
Dunno how, but I'll figure out something.
|
|
</p>
|
|
</div>
|
|
</div>
|
|
<div id="outline-container-org75908b7" class="outline-3">
|
|
<h3 id="org75908b7">Adopt/fix htmlize.el</h3>
|
|
<div class="outline-text-3" id="text-org75908b7">
|
|
<p>
|
|
I want to highlight code during publishing step.
|
|
</p>
|
|
</div>
|
|
</div>
|
|
<div id="outline-container-orgaf1ec94" class="outline-3">
|
|
<h3 id="orgaf1ec94">Show more meta on posts index page.</h3>
|
|
<div class="outline-text-3" id="text-orgaf1ec94">
|
|
<p>
|
|
Creation date, preview, tags, whatever.
|
|
<a href="./blog_index_and_tags_automation.html">Blog index and tags automation</a>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</main><footer class="footer"><div class="container"><hr/><small><p>Alex Mikhailov</p><p>Built with: <a href="https://www.gnu.org/software/emacs/">GNU Emacs</a> <a href="https://orgmode.org/">Org Mode</a> <a href="https://picocss.com/">picocss</a></p></small></div></footer></body></html>
|