Down the Emacs Rabbit Hole

17 May 2019

Over the years I have followed the holy war between vi and Emacs with some interest. As a university student and a GIS Analyst using various versions of Unix, I was handed vi and taught to use it. vi was omnipresent and could be learned to a passable level in a short amount of time. So, I was a vi user. Emacs was alluring though. The old joke, that Emacs is “a great operating system, lacking only a decent editor”[1], was both a draw to and a caution against it’s use. I tried it out a few times, but the requirement to learn so many commands with multiple meta keys was too much bother. I would give up after an hour or two, when I had to look up open a file for the eighth time.

At the same time, I was experimenting with various ways of keeping track of my huge backlog of “to do” items. Often hundred’s of items long. Tracking it in a Word document, tracking it in an excel spreadsheet. Using tabs in steno note book. When I finally gave up the fight and switched from paper notes to Microsoft OneNote, I started keeping the list in there. It did not work that well. Also, I have an ever growing fear that when Microsoft decides to move from OneNote to the next great thing, all my notes will be locked inside its proprietary file format. I’ve been burned with that problem in the past.

I became aware of Org Mode. It was built within the platform that is Emacs. It uses plain text files. It has fantastic miraculous sounding capabilities. It can be version controlled with git. I can store it in the cloud. I’ve become familiar with using Markdown and Jekyll to publish my blogs on GitHub and I love the plain text files being used for markup. So I took a serious stab at using Emacs for the first time.

My initial attempt is partially documented below in First Install - Aquamacs. While at the same time using it on my Windows 8.1 computer at work. That became frustrating because of the differences associated with Aquamacs. So I’ve set out to document fully how I got Emacs with Org Mode and all its other wonders in Fresh Install - Homebrew.

This post is a living document:

I’m using this post as a test-bed for the Magit portion of the content.

Fresh Install - Homebrew

  1. Install Emacs
    1. Install Homebrew
    2. At terminal prompt type brew cask install emacs
    3. For MacOS 10.14 (Mojave) it is necessary to allow Emacs to control your computer. It will prompt you when it tries to do something that is not yet permitted. This is a setting in the Security & Privacy panel in System Preferences under the Privacy tab with Accessibility selected.
  2. Configure Interface - add items to init.el
    1. Line Numbers:
      • create file ~/.emacs.d/init.el and add:
        (global-linum-mode t)
    2. Recent Files
      • Add to init.el:
        (recentf-mode 1)
        (setq recentf-max-menu-items 25)
        (global-set-key "\C-x\ \C-r" 'recentf-open-files)
    3. Visual Line Mode - wraps lines on word breaks
      • Add to init.el: (global-visual-line-mode t)
    4. Turn on Parenthesis Matching
      • Add (show-paren-mode 1) to init.el
    5. Configure MELPA repository for packages
      • Add to init.el
        (require 'package)
        (add-to-list 'package-archives
          '("melpa-stable" . ""))
  3. Markdown Mode for Emacs
    1. Install package.
      • M-x package-install [RET] markdown-mode [RET]
    2. Pandoc
      • At Terminal Prompt type brew install Pandoc
    3. Add to init.el inside the (custom-set-variables section before the final ):
      • '(markdown-command "/usr/local/bin/pandoc")
  4. Org Mode
    1. Add to init.el:
      (require 'org)
      (define-key global-map "\C-cl" 'org-store-link)
      (define-key global-map "\C-ca" 'org-agenda)
      (setq org-log-done 'time)
    2. For each .org file you want to participate in the Agenda open the file and issue command C-c [

    3. Create a Custom Agenda Command
      • NOTE you can use: M-x customize-variable [RET] org-agenda-custom-commands
      • Otherwise, add the following text to init.el inside the (custom-set-variables section. The “n” option was included already I added the “L” section to do a 2 week agenda with log mode turned on.
          (("L" "Time (L)ine - Biweekly Report" agenda ""
                  (org-agenda-start-day "-14d")
                  (org-agenda-span 14)
                  (org-agenda-start-on-weekday nil)
                  (org-agenda-start-with-log-mode '(closed clock state))
              ("n" "Agenda and all TODOs"
                  ((agenda "" nil)
                  (alltodo "" nil))\
    4. Org files in the cloud. I have started experimenting with Orgzly and keeping an org file synced on my Android phone, my laptop, and my desktop. It’s working OK so far.
  5. Auto-Complete
    1. Install package.
      • M-x package-install [RET] auto-complete [RET]
    2. Add to init.el:
      ;; Set up configuration for auto complete
      ;; Among other things the first line pops-up doc strings for functions
      ;;  highlighted in AC's suggestions list, while in Emacs-Lisp mode.
      (global-auto-complete-mode t)
      (add-to-list 'ac-modes 'markdown-mode)
      (add-to-list 'ac-modes 'org-mode)
  6. Spell Checking
    1. At Terminal Prompt
      brew install aspell --with-all-langs
      which aspell
    2. Copy the response from last command. Similar to: /usr/local/bin/aspell
    3. Run Customize… from Edit->Spell menu
    4. Set ispell-program-name variable to /usr/local/bin/aspell or other return from earlier prompt command.
    5. Run Spell-Check Buffer from Edit->Spell menu
    6. NOTE Flyspell can work but needs a functional mouse-2 click which requires hackery for macOS track-pad.
  7. Magit: git version control
    1. M-x package-refresh-contents [RET]
    2. M-x package-install [RET] magit [RET]
    3. Test version numbers: M-x magit-version [RET]
      Magit 2.90.1, Git 2.17.2 (Apple Git-113), Emacs 26.1, darwin
    4. Add a global key binding for magit-status to init.el
      ;; Configure magit
      (global-set-key (kbd "C-x g") 'magit-status)
    5. NOTE Regarding push-to-deploy One of the things I hoped to accomplish with Magit in emacs is a simple solution for publishing my static hand-crafted personal website. I was struggling because default behavior requires using two branches and then automating the merge to the master branch using git hooks, which was on my list to figure out, but since I don’t ever edit the files on the server, push-to-deploy is more straight-forward. In the remote repository on the web server you only need to configure git with git config receive.denyCurrentBranch updateInstead
    6. NOTE I have not yet learned how to clone a repository with Magit
  8. Time to tryout a different look: Change Default Font and Theme I elected to use the handy GUI interface for this as I did it originally on my work computer (Windows) quickly while on break.
    1. Download the hipster coding font of your choice. I ended up downloading Fira Mono and Ubuntu Mono from Google Fonts. Install them in the usual fashion for your operating system.
    2. Click on Set Default Font… on the Options menu and find the font you chose on the list.
    3. Click on Custom Themes on the Customize Emacs sub-menu on the Options menu and browse the themes listed there. I selected leuven because of how it renders an org mode document. After you decide which one to keep click the Save Theme Settings button. Just kidding! That worked on my Windows 10 Computer at work, but not on my Mac. I found some directions on Stack Overflow, which were slightly dated.
      org mode with leuven theme
    4. These Directions worked on MacOS for me:
      1. Type: M-x customize-face [RET]
      2. At “Customize face (…):” prompt type: default [RET]
      3. Edit Font Family value to match what you see in the Fonts dialog. In my case “Fira Mono”.
        _Fonts_ Dialog
      4. Uncheck all of the other options. If you do not those settings will override the settings in any subsequent theme that you choose in the Custom Themes buffer.
      5. Click the _Apply and Save__ button. This will make a change in your init.el file like this:
        ;; custom-set-faces was added by Custom.
        ;; If you edit it by hand, you could mess it up, so be careful.
        ;; Your init file should contain only one such instance.
        ;; If there is more than one, they won't work right.
        '(default ((t (:family "Fira Mono")))))
      6. Exit the buffer using the graphic button in he tool bar at the top of the screen.
  9. Topics to explore in the future:
    1. CSV-Mode - Allows column alignment and sorting by column. Also transposing columns and rows.

    2. Saving Sessions This looks very helpful to get figured out, so I don’t have to keep loading the same files over and configuring my windows.

    3. GNUS - research for use as e-mail client. Rejected Mu4e as too complicated.

    4. Elfeed - RSS Feed Reader

    5. Bookmark Plus

    6. Buffer Menu

    7. Abbrev Mode

First Install - Aquamacs

Things installed or configured

Config customization

/Users/paulmccombs/Library/Preferences/Aquamacs Emacs/Preferences.el

;; This is the Aquamacs Preferences file.
;; Add Emacs-Lisp code here that should be executed whenever
;; you start Aquamacs Emacs. If errors occur, Aquamacs will stop
;; evaluating this file and print errors in the *Messages* buffer.
;; Use this file in place of ~/.emacs (which is loaded as well.)
(require 'org)
(define-key global-map "\C-cl" 'org-store-link)
(define-key global-map "\C-ca" 'org-agenda)
(setq org-log-done t)
(setq org-agenda-files (list "~/org/work_org_mode/"))

/Users/paulmccombs/Library/Preferences/Aquamacs Emacs/customizations.el

;; Added by Package.el.  This must come before configurations of
;; installed packages.  Don't delete this line.  If you don't want it,
;; just comment it out by adding a semicolon to the start of the line.
;; You may delete these explanatory comments.

 ;; custom-set-variables was added by Custom.
 ;; If you edit it by hand, you could mess it up, so be careful.
 ;; Your init file should contain only one such instance.
 ;; If there is more than one, they won't work right.
    (emacs-lisp-mode lisp-mode lisp-interaction-mode slime-repl-mode nim-mode c-mode cc-mode c++-mode objc-mode swift-mode go-mode java-mode malabar-mode clojure-mode clojurescript-mode scala-mode scheme-mode ocaml-mode tuareg-mode coq-mode haskell-mode agda-mode agda2-mode perl-mode cperl-mode python-mode ruby-mode lua-mode tcl-mode ecmascript-mode javascript-mode js-mode js-jsx-mode js2-mode js2-jsx-mode coffee-mode php-mode css-mode scss-mode less-css-mode elixir-mode makefile-mode sh-mode fortran-mode f90-mode ada-mode xml-mode sgml-mode web-mode ts-mode sclang-mode verilog-mode qml-mode apples-mode markdown-mode gfm-mode org-mode)))
 '(aquamacs-additional-fontsets nil t)
 '(aquamacs-customization-version-id 308 t)
 '(aquamacs-tool-bar-user-customization nil t)
 '(global-auto-complete-mode t)
 '(global-linum-mode t)
 '(global-visual-line-mode t)
 '(markdown-command "pandoc")
 '(ns-tool-bar-display-mode (quote both) t)
 '(ns-tool-bar-size-mode (quote regular) t)
 '(package-selected-packages (quote (auto-complete auto-correct markdown-mode)))
 '(text-mode-hook nil)
 '(visual-line-mode nil t))
 ;; custom-set-faces was added by Custom.
 ;; If you edit it by hand, you could mess it up, so be careful.
 ;; Your init file should contain only one such instance.
 ;; If there is more than one, they won't work right.

Foot Notes

[1] Editor war, Wikipedia, The Free Encyclopedia (last visited Oct. 11, 2018).