Org-Mode

Org-mode might the emacs killer app. This file holds some extra settings which were not set in the use-package statement in init.el.

General settings

Some general settings for org. TODO: Clean up the use-package definition

(use-package org
  :init ; loaded before package is initialized
  ;; The background color of my org source code blocks needs to be defined before
  ;; org is loaded.
  (defface org-block-begin-line
    '((t (:foreground "#99968b" :background "#303030" :box (:style released-button) :extend t)))
    "Face used for the line delimiting the begin of source blocks.")
  (defface org-block-end-line
    '((t (:foreground "#99968b" :background "#303030" :box (:style released-button) :extend t)))
    "Face used for the line delimiting the end of source blocks.")
  :config
  (require 'org-tempo)
  (setq-default
   org-return-follows-link t
   org-image-actual-width '(800)
   org-highlight-latex-and-related '(latex script entities))
  (add-hook 'org-mode-hook #'org-indent-mode)
  (add-hook 'org-mode-hook #'visual-line-mode)
  (add-hook 'org-mode-hook #'flyspell-mode)
  (set-face-attribute 'org-drawer nil
    :inherit 'font-lock-keyword-face)

  ;; change some keys to be a little more like LaTeX
  (defun do.org/insert-math ()
    (interactive)
    (insert "\\(\\)")
    (backward-char 2))

  (defun do.org/insert-today-inactive ()
    "Insert today's date as inactive time stamp."
    (interactive)
    (org-insert-time-stamp (org-current-time) nil t))

  (defun do.org/time-stamp-inactive ()
    "Insert an inactive timestamp of the user's choosing."
    (interactive)
    (org-time-stamp nil t))

  (defalias 'ts #'do.org/time-stamp-inactive)

  ;; non-breaking spaces
  <<do.org/non-breaking-spaces>>

  ;; fix `electric-pair-mode'
  <<do.org/fix-electric-pair>>

  :bind
  (("C-c l" . #'org-store-link)
   ("C-c t" . #'do.org/insert-today-inactive)
   :map org-mode-map
   ("$" . do.org/insert-math)
   ("~" . do.org/insert-nbsp)))

(use-package org-contrib
  :ensure t)

fixing `electric-pair-mode'

(defun do.org/fix-electric-pair-fn ()
    (setq-local electric-pair-inhibit-predicate
                `(lambda (c)
                   (if (char-equal c ?<) t (,electric-pair-inhibit-predicate c)))))
(add-hook 'org-mode-hook #'do.org/fix-electric-pair-fn)

Syntax highlighting in Code Blocks

Very important if your config file is a .org document… Also, add native <tab> behavior in source blocks.

(setq
 org-src-fontify-natively t
 org-src-tab-acts-natively t)

We also want this in LaTeX output!

(setq org-latex-listings 'minted)

HTML Export

For HTML output, we want to be able to set a custom stylesheet.

(setq org-html-htmlize-output-type 'css)

We need the htmlize package for that

(use-package htmlize
  :ensure t
  :commands (htmlize-buffer
       htmlize-file
       htmlize-many-files
       htmlize-many-files-dired
       htmlize-region))

Static site generation for my website

The code to build my website is in a self-contained .el file in my website's git repo. This makes rebuilding the page using emacs --batch a little easier and does not require me to load the entire config. Whenever I edit the website, I use emacs-async to spawn an external emacs process to rebuild the site. A few commands and keybindings to make this easier follow.

First, I define the paths to my website's base directory and the static site generator script.

(defvar do.website/base-dir nil
  "Path to the base directory of the website.")
(defconst do.website/project-script
  (file-name-concat do.website/base-dir "lisp/project.el")
  "Path to the project.el script containing my static site generator.")

Next I define some helper functions to let me easily publish the website or republish only a single article asynchronously. Note that these commands never actually deploy the website to the web server, unless that was specifically requested

(defun do.website/generate-website (arg)
  "Rebuild the org-publish project only. If prefix ARG is set, force rebuild everything."
  (interactive "P")
  (async-start
   `(lambda ()
      (load-file ,do.website/project-script)
      (generate-website ,arg))
   (lambda (result)
     (beep)
     (message "generate-website: %s" (prin1-to-string result)))))

(defun do.website/make-website (arg)
  "Run the full \"make\" command for the website asynchronously. If prefix ARG is set, deploy the website"
  (interactive "P")
  (let ((buf-name "*Website Build Output*")
  (deploy (and arg (y-or-n-p "You requested to deploy the website after build. Are you sure?"))))
    (async-shell-command
     (concat "cd " do.website/base-dir " && " (if deploy "make deploy" "make"))
     buf-name)))

(defun do.website/deploy-website ()
  (interactive)
  (make-website t))

(defun do.website/republish-current-file ()
  "Re-publish the currently edited file asynchronously."
  (interactive)
  (save-buffer)
  (async-start
   `(lambda ()
      (load-file ,do.website/project-script)
      (org-publish-file ,(buffer-file-name)))
   (lambda (result)
     (beep)
     (message "republish-current-file: Done"))))

If the website actually exists, we add some local keybindings to org-mode to the above functions.

(when (file-exists-p do.website/project-script)
  (add-hook 'org-mode-hook (lambda ()
                             (local-set-key (kbd "C-c c") #'do.website/republish-current-file)
                             (local-set-key (kbd "C-c m") #'do.website/make-website)
                             (local-set-key (kbd "C-c d") #'do.website/deploy-website)
                             (local-set-key (kbd "C-c g") #'do.website/generate-website))))

For convenience, I define an org-tempo template for a new blog post.

(require 'org-tempo)
(tempo-define-template "blog-header" ; just some name for the template
     '("#+title: ?" n
       "#+AUTHOR: Dennis Ogbe" n
       "#+EMAIL: [email protected]" n
       "#+DATE:" n
       "#+STARTUP: showall" n
       "#+STARTUP: inlineimages" n
       "#+BEGIN_PREVIEW" n p n
       "#+END_PREVIEW")
     "<b"
     "Insert blog header" ; documentation
     'org-tempo-tags)

Finally, an org-tempo template for to insert a collapsible <details> block into a webpage.

(tempo-define-template "html-details" ; just some name for the template
                       '("#+HTML: <details><summary><b>" p
                         "</b></summary>" n p n
                         "#+HTML: </details>" n)
                       "<d"
                       "Insert collapsible block" ; documentation
                       'org-tempo-tags)

org-latex settings

For some reason, explained here, we need to run pdflatex with the -shell-escape flag. This is getting too complicated, so we're just going to run latexmk from org.

(setq org-latex-pdf-process (list "latexmk -pdflatex='pdflatex -interaction nonstopmode' -shell-escape -pdf -bibtex -f %f"))
(setq org-latex-with-hyperref
"\\hypersetup{
colorlinks=true,
citecolor=red,
filecolor=blue,
linkcolor=red,
urlcolor=blue,
pdfauthor={%a},
pdftitle={%t},
pdfkeywords={%k},
pdfsubject={%d},
pdfcreator={%c},
pdflang={%L}}\n")
(setq org-latex-prefer-user-labels t)

If we want to export to a documentclass other than article, we need to add its structure to the following list. To use that class, we can then put the following in the org-mode file header: #+LATEX_CLASS: <class>. To add options to the class, we add the following: #+LATEX_CLASS_OPTIONS: [op1, op2,...].

(require 'ox-latex)
(add-to-list 'org-latex-classes
             '("IEEEtran"
               "\\documentclass{IEEEtran}"
               ("\\section{%s}" . "\\section*{%s}")
               ("\\subsection{%s}" . "\\subsection*{%s}")
               ("\\subsubsection{%s}" . "\\subsubsection*{%s}")
               ("\\paragraph{%s}" . "\\paragraph*{%s}")))

We add some templates for some common math environments

(add-to-list 'org-structure-template-alist
       '("prop"  "#+begin_proposition ?\n\n#+end_proposition"))
(add-to-list 'org-structure-template-alist
       '("thm"  "#+begin_theorem ?\n\n#+end_theorem"))
(add-to-list 'org-structure-template-alist
       '("lem"  "#+begin_lemma ?\n\n#+end_lemma"))
(add-to-list 'org-structure-template-alist
       '("pf"  "#+begin_proof ?\n\n#+end_proof"))

Non-breaking spaces

Entering "~" in org-mode buffers enters a literal tilde character and not the non-breaking space that I usually want. To get around this, we can insert "non-breaking-space" org entity. Unfortunately, having \nbsp{} littered around org-mode buffers is super ugly, so we use prettify-symbols-mode to correct this eyesore.

This is straightforward, but the default behavior of prettify-symbols-mode does not match occurrences like as\nbsp{}cite:citation, where the space is butting up to other strings on both sides. So we have to fix that too.

(defun do.org/prettify-symbols-compose-predicate (start end _match)
  "Explicitly allow any occurrence of the non-breaking space to be composed."
  (let ((result (prettify-symbols-default-compose-p start end _match)))
    (or result (string-equal (buffer-substring start end) "\\nbsp{}"))))

(defun do.org/prettify-symbols-setup ()
  "Set up `prettify-symbols-mode' for `org-mode' buffers."
  (make-variable-buffer-local 'prettify-symbols-unprettify-at-point)
  (setq prettify-symbols-unprettify-at-point 'right-edge)
  (setq prettify-symbols-compose-predicate #'do.org/prettify-symbols-compose-predicate)
  (setq prettify-symbols-alist `(("\\nbsp{}" . ,(string-to-char "~"))))
  (prettify-symbols-mode 1))

(add-hook 'org-mode-hook #'do.org/prettify-symbols-setup)

(defun do.org/insert-nbsp ()
  "Bind this to the ~ (tilde) character to insert non-breaking
spaces on `org-mode' for LaTeX export."
  (interactive)
  (insert "\\nbsp{}"))

Default Applications

Org opens PDFs by default in gv… change that to evince. Also open HTML in Chrome.

(let ((pdf-reader (if (boundp 'do.minimal/pdf-reader)
    do.minimal/pdf-reader
        "evince")))
(setq org-file-apps `((auto-mode . emacs)
    ("\\.x?html?\\'" . "firefox %s")
    ("\\.pdf\\'" . ,(concat pdf-reader " \"%s\""))
    ("\\.pdf::\\([0-9]+\\)\\'" . ,(concat pdf-reader " \"%s\" -p %1"))
    ("\\.pdf.xoj" . "xournal %s"))))

org-babel

Babel lets the user run code inside an org-mode document. I feel like this can come in handy one day.

Active Babel languages

(org-babel-do-load-languages
 'org-babel-load-languages
 '((python . t)
   (emacs-lisp . t)
   (matlab . t)
   (octave . t)
   (latex . t)
   (js . t)
   (shell . t)
   (C . t)
   (ditaa . t)))

org-babel settings

(setq org-confirm-babel-evaluate nil)

ditaa

Ditaa is a small tool that takes ASCII art and renders it as neat diagrams. Combined with org, I can sketch out ideas directly in an org buffer using the emacs Artist mode.

(setq org-ditaa-jar-path (concat initel-directory "random/ditaa/ditaa0_9.jar"))

Windmove and org-mode clashes

Make windmove (and framemove) play nice with org.

(add-hook 'org-shiftup-final-hook 'windmove-up)
(add-hook 'org-shiftleft-final-hook 'windmove-left)
(add-hook 'org-shiftdown-final-hook 'windmove-down)
(add-hook 'org-shiftright-final-hook 'windmove-right)

YaSnippet and org

backtab is already used in org-mode. Let's use C-<tab> instead.

(define-key org-mode-map (kbd "C-<tab>") 'yas-expand)

org-download

Incredibly useful for inserting images and screenshots from the clipboard into my org-mode notes files.

(use-package org-download
  :ensure t
  :init
  (setq org-download-screenshot-method "xclip -selection clipboard -t image/png -o > %s")
  (setq org-download-method 'directory)
  (defalias 'insert-screenshot 'org-download-screenshot)
  :config
  (add-hook 'dired-mode-hook 'org-download-enable))

Execute a code block on startup

Sometimes I want to execute a code block in an org file as soon as it is started. This can be done as follows. We first need the following function.

(defun do.org/execute-startup-block ()
  "Go to the startup block and execute it. It must be named 'do-org-startup-code'."
  (interactive)
  ;; only run this once. some modes (looking at you, org-ref) run
  ;; `hack-local-variables' again. we want our code to be executed only once on
  ;; buffer creation.
  (unless (bound-and-true-p do.org/startup-was-run-p)
    (org-babel-goto-named-src-block "do-org-startup-code")
    (let ((org-babel-after-execute-hook nil))
      (org-babel-execute-src-block))
    (org-global-cycle)
    (setq-local do.org/startup-was-run-p t)))

We can identify the code block to execute on startup by giving it the name do-org-startup-code, i.e., we add a #+NAME: do-org-startup-code in the line before #+begin_src.

Then to execute, we need to set the :eval local variable to call our function, which visits the block and runs it. This might look like this:

FILENAME -*- mode: org; eval: (do.org/execute-startup-block) -*-

or the corresponding multiline comment form at the end of a file.