UP | HOME

Holymacs

Table of Contents

Specs

It's a literate configuration but I'm not good at writing, this means that it will not be so literate in the end 😂 If you're reading this and you're new to the emacs world I recommend to view this document using Emacs: if you're reading the org version (from source repository) or from the internet using the Emacs browser eww, you can C-h o (or C-h p for packages) to have more information from Emacs documentation.

Holymacs

Package Manager

(add-to-list 'load-path "~/.emacs.d/vendor/") ; that's your sandbox
(require 'package)

(setq package-archives '(("melpa" . "https://melpa.org/packages/")
                           ("elpa" . "https://elpa.gnu.org/packages/")
                           ("nongnu" . "https://elpa.nongnu.org/nongnu/")))

(require 'use-package)
(require 'use-package-ensure)
(setq use-package-always-ensure t
        package-enable-at-startup t)
(package-initialize)

Main configuration

User data

Remember to set your name and email address! Place here your real data, or use the standard Emacs customization (customize-variable 'user-full-name).

(setq user-full-name "Your username"
      user-mail-address "username@domain.net")

Tab key, backup files and enabled major modes

(setq tab-always-indent 'complete
        enable-recursive-minibuffers t
        make-backup-files nil)

Just C-h o over those functions for more information.

(ffap-bindings)
(savehist-mode 1)
(recentf-mode 1)

Bookmarks

  • eww-bookmark-jump

    Here add a custom bookmark handler relying on browse-url and browse-url-handlers, so even if we store a bookmark via eww, we're going to use the proper browser when opening it.

    (defun browse-url-bookmark-jump (bookmark)
      "Bookmark jump handler: relies on =browse-url-handler= to pick the right browser.
      Define a custom jumper avoid to open always on EWW in case the bookmark was placed with it"
      (browse-url (bookmark-prop-get bookmark 'location)))
    (require 'eww)
    (defalias #'eww-bookmark-jump #'browse-url-bookmark-jump)
    

Dired

(require 'dired)
(setq make-backup-files nil
        tab-always-indent 'complete
        dired-dwim-target t
        dired-listing-switches "-alth"
        dired-omit-files "\\`[.]?#\\|\\`[.][.]?")
(define-key dired-mode-map (kbd "W") #'wdired-change-to-wdired-mode)
(define-key dired-mode-map (kbd "H") #'dired-omit-mode)
(define-key dired-mode-map (kbd "M-p") (lambda () (interactive) (dired-previous-line 1) (dired-display-file)))
(define-key dired-mode-map (kbd "M-n") (lambda () (interactive) (dired-next-line 1) (dired-display-file)))
(add-hook 'dired-mode-hook #'dired-hide-details-mode)
(add-hook 'dired-mode-hook #'dired-omit-mode)

Appearance

(menu-bar-mode -1)
(tool-bar-mode -1)
(scroll-bar-mode -1)
(setq ring-bell-function 'ignore
        visible-bell t)

Window and Tabs management

No configuration till now. If you want to learn something about this topic there this fantastic article, or place the cursor on the function and C-h o.

(winner-mode 1)
(defun holymacs-new-tab (name)
  "Create a new tab with NAME. When called interactively asks for the name in the minibuffer."
  (interactive "MTab: ")
  (tab-new)
  (tab-rename name))

(global-set-key (kbd "M-o w") #'holymacs-new-tab)

Orderless

(use-package orderless
  :custom
  (completion-styles '(orderless partial-completion)))

Minibuffer and Completion framework

  • Minibuffer
    (setq enable-recursive-minibuffers t)
    (add-hook 'minibuffer-setup-hook (lambda () (local-set-key (kbd "C-o") #'switch-to-completions)))
    (define-key completion-list-mode-map (kbd "e") #'switch-to-minibuffer)
    
  • Vertico
    (use-package vertico
      :init
      (vertico-mode t)
      (setq vertico-count 10
              vertico-resize t
              vertico-cycle t))
    

Auto-completion

(use-package corfu
  :hook
  (eshell-mode . (lambda ()
                     (setq-local corfu-auto nil)
                     (corfu-mode)))
  :custom
  (corfu-cycle t)
  :config
  (global-corfu-mode))
(use-package cape
  :init
  (add-hook 'completion-at-point-functions #'cape-dabbrev)
  (add-hook 'completion-at-point-functions #'cape-file)
  (add-hook 'completion-at-point-functions #'cape-elisp-block)
  (add-hook 'completion-at-point-functions #'cape-history)
  ;;(add-hook 'completion-at-point-functions #'cape-keyword)
  ;;(add-hook 'completion-at-point-functions #'cape-tex)
  ;;(add-hook 'completion-at-point-functions #'cape-sgml)
  ;;(add-hook 'completion-at-point-functions #'cape-rfc1345)
  (add-hook 'completion-at-point-functions #'cape-abbrev)
  (add-hook 'completion-at-point-functions #'cape-dict)
  (add-hook 'completion-at-point-functions #'cape-elisp-symbol)
  (add-hook 'completion-at-point-functions #'cape-line))

Embark

(use-package embark
  :config
  (add-to-list 'god-exempt-major-modes 'embark-mode)
  :bind*
  ("C-," . embark-act)
  (:map embark-general-map
          ("C-w" . browse-url))
  (:map embark-identifier-map
          ("R" . query-replace)
          ("O" . multi-occur-in-matching-buffers)
          ("c" . browse-url-chrome)
          ("f" . browse-url-firefox))
  (:map embark-url-map
          ("c" . browse-url-chrome)
          ("f" . browse-url-firefox))
  (:map  embark-file-map
           ("p" . project-find-file)
           ("b" . project-switch-to-buffer)
           ("," . append-to-previous-buffer)))

Eshell

(add-to-list 'org-babel-load-languages '(eshell . t))

An example of eshell aliases.

alias clear clear -1
alias vsnap ffmpeg -i $1 -ss 10 -frames:v 1 $2
alias econf view-file "~/minasmazar.github.io/org/holymacs.org"
alias t holymacs-tmux/open
alias q bury-buffer
alias g magit-status
alias customize (find-file custom-file)
alias o view-file $1
alias d dired-jump t $1
alias aliases (view-file eshell-aliases-file)
alias agenda (org-agenda "t")
alias x async-shell-command $1
alias l ls -lh $*
alias $* what $*
alias yt (elfeed-search "@6-months-ago +unread +news")
alias reload eshell-read-aliases-list
alias v eshell-exec-visual $1
alias www eww $1

Projects

(require 'project)
(add-to-list 'project-switch-commands '(project-shell "Shell"))

Keybindings

  • Custom keybindings

    In vanilla Emacs the M-o keybinding is free: I'll use it as sort of an entrypoint for commands I often use, as for org commands. I'll use M-o M-o to map major-mode related commands.

    (global-set-key (kbd "M-o a") #'org-agenda)
    (global-set-key (kbd "M-o k") #'org-capture)
    (global-set-key (kbd "M-o 8") #'emoji-search)
    (global-set-key (kbd "M-o e") #'eshell)
    (global-set-key (kbd "M-o b") #'ibuffer)
    (global-set-key (kbd "M-o E") #'project-eshell)
    (global-set-key (kbd "M-o h") #'info-display-manual)
    
    (define-key dired-mode-map (kbd "M-o M-o h") #'dired-omit-mode)
    
  • For not graphical client (i.e. emacs -nw)

    Sometimes when you run Emacs in old terminal emulators, you don't have access to the alt or super key. For this reason I redefine important Emacs standard keybindings to M-x and the god-mode toggle.

    (global-set-key (kbd "C-k") nil)
    (global-set-key (kbd "C-k") #'execute-extended-command)
    (global-set-key (kbd "C-l") nil)
    (global-set-key (kbd "C-l") 'god-local-mode)
    
  • Ease Isearch navigation

    Uses normal movements keys and TAB key to move through isearch.

    (define-key isearch-mode-map (kbd "C-p") 'isearch-repeat-backward)
    (define-key isearch-mode-map (kbd "C-n") 'isearch-repeat-forward)
    (define-key isearch-mode-map (kbd "<tab>") 'isearch-repeat-forward)
    (define-key isearch-mode-map (kbd "<S-tab>") 'isearch-repeat-backward)
    
  • Translations keys
    (define-key key-translation-map (kbd "s-k") (kbd "M-x"))
    (define-key key-translation-map (kbd "s-m") (kbd "C-x"))
    (define-key key-translation-map (kbd "s-M") (kbd "C-c"))
    (define-key key-translation-map (kbd "s-o") (kbd "M-o"))
    (define-key key-translation-map (kbd "s-O") (kbd "M-o M-o"))
    
  • Super-key
    (global-set-key (kbd "s-0") (kbd "C-x 0"))
    (global-set-key (kbd "s-1") (kbd "C-x 1"))
    (global-set-key (kbd "s-2") (kbd "C-x 2"))
    (global-set-key (kbd "s-3") (kbd "C-x 3"))
    (global-set-key (kbd "s-w") (kbd "C-x o"))
    (global-set-key (kbd "s-r") (kbd "C-x z"))
    (global-set-key (kbd "s-i") (kbd "M-g i"))
    (global-set-key (kbd "s-N") (kbd "C-x C-j"))
    (global-set-key (kbd "s-[") (kbd "C-x <left>"))
    (global-set-key (kbd "s-]") (kbd "C-x <right>"))
    (global-set-key (kbd "s-,") #'switch-to-completions) ;; useful if you don't use ido, vertico or similar
    (global-set-key (kbd "s-b") #'ibuffer)
    (global-set-key (kbd "s-j") #'switch-to-buffer)
    (global-set-key (kbd "s-)") #'kill-current-buffer)
    (global-set-key (kbd "s--") #'bookmark-jump)
    (global-set-key (kbd "s-_") #'bookmark-set)
    (global-set-key (kbd "s-q") #'tab-switch)
    (global-set-key (kbd "s-\\") #'other-frame)
    (global-set-key (kbd "s-t") #'find-name-dired)
    (global-set-key (kbd "s-e") #'dabbrev-expand)
    (global-set-key (kbd "s-r") #'repeat)
    (global-set-key (kbd "s-u") #'revert-buffer)
    (global-set-key (kbd "s-p") #'project-switch-to-buffer)
    (global-set-key (kbd "s-P") #'project-find-file)
    
  • God mode
    (use-package god-mode
      :config
      (defun minemacs-god-mode-disabled ()
        (setq cursor-type 'bar))
    
      (defun minemacs-god-mode-enabled ()
        (setq cursor-type 'box))
    
      (add-hook 'god-mode-enabled-hook 'minemacs-god-mode-enabled)
      (add-hook 'god-mode-disabled-hook 'minemacs-god-mode-disabled)
      (global-set-key (kbd "s-n") #'god-local-mode)
      (define-key god-local-mode-map (kbd "i") #'god-local-mode)
      (define-key god-local-mode-map (kbd "I") (kbd "C-c '"))
      (define-key god-local-mode-map (kbd "u") #'undo)
      (define-key god-local-mode-map (kbd "U") #'undo-redo)
      (define-key god-local-mode-map (kbd ">") #'end-of-buffer)
      (define-key god-local-mode-map (kbd "<") #'beginning-of-buffer)
      (define-key god-local-mode-map (kbd "[") #'backward-paragraph)
      (define-key god-local-mode-map (kbd "]") #'forward-paragraph)
      (define-key god-local-mode-map (kbd "{") #'backward-sexp)
      (define-key god-local-mode-map (kbd "}") #'forward-sexp)
      (define-key god-local-mode-map (kbd "H") #'describe-symbol)
      (mapcar (lambda (exempted-major-mode)
                (add-to-list 'god-exempt-major-modes exempted-major-mode))
              '(compilation-mode
                org-capture-mode
                org-agenda-mode
                elfeed-search-mode
                elfeed-show-mode))
      (god-mode))
    
  • which-key
    (use-package which-key
      :config
      (which-key-mode)
      (which-key-enable-god-mode-support))
    

Shared configuration

Here's I share my most used configuration files across systems by simply usiung a shared directory.

(if (file-exists-p "~/Dropbox/emacs/")
    (setq bookmark-file "~/Dropbox/emacs/bookmarks"
            eshell-aliases-file "~/Dropbox/emacs/eshell-aliases"
            custom-file "~/.emacs.d/custom.el"
            org-agenda-files '("~/Dropbox/org/")
            org-roam-directory "~/Dropbox/org-roam/"))
(load custom-file 'noerror)

Modules

Compilation and loading

Re-tangle all modules. I don't think a specific command is needed, unless in the future I want to achieve something more complex. When you want to sync the literate files to the actual Emacs configuration, open this org file and eval the next code block (i.e. C-c if you're reading this in org-mode (experimenting with eww and auto-activate org-mode.

(defun holymacs-compile ()
  "Babel-compile all '*.org' literate configuration files in './holymacs' to generate emacs modules files. Load them with 'holymacs-modules-load'"
  (interactive)
  (message "holymacs: compilation..")
  (mapcar #'org-babel-tangle-file
          (let ((default-directory "~/minasmazar.github.io/org/"))
               (directory-files "holymacs" t "\\.org$" directory-files-no-dot-files-regexp))))

Here we define a mechanism to easily extend your Emacs experience with some custom defined modules, a set of packages and configuration that can be grouped for any reason (they're the equivalent of the layers in Doom Emacs), but I'm not creating a whole mechanism to handle this: just place an .el file in the ~/.emacs.d/modules/ directory, and it will be autoloaded.

(defun holymacs-modules-load ()
  "Load all modules '*.el' files in './.emacs.d/modules/' babel-generated with 'holymacs-compile'"
  (interactive)
  (let* ((modules-dir (concat user-emacs-directory "modules/"))
         (modules (if (file-exists-p modules-dir)
                      (directory-files modules-dir))))
    (if modules (mapcar (lambda (module)
                          (load (concat modules-dir module) 'noerror))
                        modules))))

Load modules!

(holymacs-modules-load)

Helpers (🔬experimental)

(defvar holymacs--private-data-pathname
  "~/Dropbox/emacs/private.el"
  "Holymacs store here some data; its purpose is to keep some data separate and almost secure.")

(defmacro holymacs-with-private-data (&rest body)
  `(progn
     (load holymacs--private-data-pathname)
     ,@body))

(defun holymacs-kill (name)
  (interactive "Bbuffer:")
  (when (get-buffer name)
    (progn
      (holymacs-inspect (format "killing %s" name))
      (kill-buffer name))))

(defun holymacs-async (cmd &optional name)
  (interactive (list (read-from-minibuffer "Cmd: ")
                     (read-from-minibuffer "buffer: " "*holymacs-aync*")))
  (holymacs-inspect cmd)
  (save-window-excursion
    (async-shell-command cmd name)))

(defun holymacs-async (cmd &body)
  (holymacs-inspect cmd)
  (save-window-excursion
    (async-shell-command cmd "*holymacs-async*")))

(defun holymacs-inspect (obj)
  (message (format "holymacs-inspect: %s" obj))
  obj)

Blorg

The Blorg publishing functions described in README.org will tangle its code in the blorg.el module.

IDE

On hand documentation

Other Commands, functions, variables, keybindings

Just press C-h o to get help on those symbols.

  • add-file-local-variable (append at bottom of the file) and add-file-local-variable-prop-line (prepend in the first line)
  • align-regexp
  • auto-mode-alist
  • browse-url-of-buffer
  • command-switch-alist
  • completions-max-height
  • completion-auto-help
  • compilation-scroll-output
  • current-time-string
  • format-time-string
  • describe-text-properties
  • face-remap-add-relative and face-remap-remove-relative to easily modify faces in current buffer
  • find-name-dired
  • image-crop
  • imenu-generic-expression
  • json-pretty-print-buffer
  • mm-discouraged-alternatives
  • org-agenda-include-diary
  • org-agenda-custom-commands
  • org-archive-subtree-default
  • org-babel-load-file
  • org-todo-keywords
  • pp-eval-expression
  • proced
  • shr-inhibit-images
  • string-match
  • C-x @

Org code-blocks

#+BEGIN_SRC sh :var filename=(buffer-file-name) :exports both
  wc -w $filename
#+END_SRC

String and Symbols

(symbol-name 'some-symbol)
; => "some-symbol"
(intern "some-symbol")
; => some-symbol

Macros

(defmacro ++ (x)
  (list 'setq x (list '1+ x)))

Vectors

Get an element from a vector

;; get a element from vector
(aref ["a" "b" "c"] 0) ; ⇒ "a"
;; get a element from vector
(elt ["a" "b" "c"] 0) ; ⇒ "a"

What's the difference between aref and elt? elt is more general. It works on vector and list. But if you know it's vector, you should use aref, because it's precise and faster.

Handle List

To find an element

  • member
  • contains
  • seq-find
(seq-find (lambda (val) (= 3 val)) '(1 2 3))

(interactive) functions

Refs to this SO post try elisp:(completing-read%20%22Prompt:%20%22%20'(i1%20i2%20i3)) and this post; try elisp:(read-from-minibuffer%20%22Prompt:%20%22%20%22initial%22). Take a look at minemacs/demo-function, it can help if you want to write functions with bit more complex interactive prompts.

(defun minemacs/demo-function (url &optional handler)
  "Example function."
  (interactive (list
		  (read-from-minibuffer "URL: ")
		  (completing-read "handler: " '(browse-url-firefox browse-url-chrome))))
   (message (format "url: %s, handler: %s" url handler )))

Regular Expression Syntax (regexp)

You can use the wonderful tool regexp-builder. Use string-match to run a regexp and match-string to extract the captures (remember to specify the 2nd arugment as the matched string itself). Example:

(let ((match-data (string-match "\\(\\ca\\{11\\}\\)\$" "https://www.youtube.com/watch?v=JvBfP62HTAM")))
  (match-string 1 "https://www.youtube.com/watch?v=JvBfP62HTAM"))

Here is the syntax used by Emacs for regular expressions. Any character matches itself, except for the list below.

The following characters are special : . * + ? ^ $ \ [

Between brackets [], the following are special : ] - ^

Many characters are special when they follow a backslash – see below.

. any character (but newline)

  • previous character or group, repeated 0 or more time
  • previous character or group, repeated 1 or more time

? previous character or group, repeated 0 or 1 time ^ start of line $ end of line […] any character between brackets [^..] any character not in the brackets [a-z] any character between a and z \ prevents interpretation of following special char \| or \w word constituent \b word boundary \sc character with c syntax (e.g. \s- for whitespace char) \( \) start\end of group \< \> start\end of word \_< \_> start\end of symbol \` \' start\end of buffer\string \1 string matched by the first group \n string matched by the nth group \{3\} previous character or group, repeated 3 times \{3,\} previous character or group, repeated 3 or more times \{3,6\} previous character or group, repeated 3 to 6 times \= match succeeds if it is located at point *?, +?, and ?? are non-greedy versions of *, +, and ? – see NonGreedyRegexp. Also, \W, \B, and \Sc match any character that does not match \w, \b, and \sc.

Characters are organized by category. Use C-u C-x = to display the category of the character under the cursor.

\ca ascii character \Ca non-ascii character (newline included) \cl latin character \cg greek character Here are some syntax classes that can be used between brackets, e.g. [[:upper:]\|[:digit:]\.].

[:digit:] a digit, same as [0-9] [:alpha:] a letter (an alphabetic character) [:alnum:] a letter or adigit (an alphanumeric character () [:upper:] a letter in uppercase [:space:] a whitespace character, as defined by the syntax table [:xdigit:] an hexadecimal digit [:cntrl:] a control character [:ascii:] an ascii character Syntax classes:

\s- whitespace character \s/ character quote character \sw word constituent \s$ paired delimiter \s_ symbol constituent \s' expression prefix \s. punctuation character \s< comment starter \s( open delimiter character \s> comment ender \s) close delimiter character \s! generic comment delimiter \s" string quote character \s| generic string delimiter \s\ escape character

Timers

You can run specified function at specified time and with a specific timeout. Check out the doc for more information.

(run-at-time "11 min" t #'elfeed-update)
(run-at-time "5 hours" t (lambda () (load-theme (seq-random-elt custom-known-themes))))

The list-timers command lists all the currently active timers. There’s only one command available in the buffer displayed: c (timer-list-cancel) that will cancel the timer on the line under point.

Insert colorize text programmatically

(insert (propertize "Red Text" 'font-lock-face '(:foreground "red")))

Buffer management

Code stolen from stackoverflow.

(defun minemacs-clear-buffer ()
  "clear whole buffer add contents to the kill ring"
  (interactive)
  (kill-region (point-min) (point-max)))

(defun minemacs-clear-buffer-permenantly ()
  "clear whole buffer, contents is not added to the kill ring"
  (interactive)
  (delete-region (point-min) (point-max)))

Faces and fonts

If you'd like to have a smaller font on compialtion buffers

(add-hook 'compilation-mode-hook (lambda () (text-scale-decrease 1)))

Useful function if you want to quick inspect the face under cursor

(defun what-face (pos)
  (interactive "d")
  (let ((face (or (get-char-property pos 'read-face-name)
                  (get-char-property pos 'face))))
    (if face (message "Face: %s" face) (message "No face at %d" pos))))

Get the list of font families (font-family-list); here's the code to set a random font family in the current frame.

(set-face-font 'default (seq-random-elt (font-family-list)))

Libaries

Useful topics

Credits

I would like to thanks:

Date: 2023-07-15 Sat 00:00

Emacs 30.1 (Org mode 9.7.11)

Validate