1 #+title: Literate Emacs Configuration of Amin Bandali
4 #+property: header-args :tangle yes
11 This org file is my literate configuration for GNU Emacs, and is
12 tangled to [[./init.el][init.el]]. Packages are installed and managed using
13 [[https://github.com/raxod502/straight.el][straight.el]]. Over the years, I've taken inspiration from
14 configurations of many different people. Some of the configurations
15 that I can remember off the top of my head are:
17 - [[https://github.com/dieggsy/dotfiles][dieggsy/dotfiles]]: literate Emacs and dotfiles configuration, uses
18 straight.el for managing packages
19 - [[https://github.com/dakra/dmacs][dakra/dmacs]]: literate Emacs configuration, using Borg for managing
21 - [[http://pages.sachachua.com/.emacs.d/Sacha.html][Sacha Chua's literate Emacs configuration]]
22 - [[https://github.com/dakrone/eos][dakrone/eos]]
23 - Ryan Rix's [[http://doc.rix.si/cce/cce.html][Complete Computing Environment]] ([[http://doc.rix.si/projects/fsem.html][about cce]])
24 - [[https://github.com/jwiegley/dot-emacs][jwiegley/dot-emacs]]: nix-based configuration
25 - [[https://github.com/wasamasa/dotemacs][wasamasa/dotemacs]]
26 - [[https://github.com/hlissner/doom-emacs][Doom Emacs]]
28 I'd like to have a fully reproducible Emacs setup (part of the reason
29 why I store my configuration in this repository) but unfortunately out
30 of the box, that's not achievable with =package.el=, not currently
31 anyway. So, I've opted to use =straight.el=. I also used Borg for a
32 few months, but decided to try =straight.el= which allows direct use
33 of the various package archives.
37 :CUSTOM_ID: installation
40 To use this config for your Emacs, first you need to clone this repo,
41 then tangle =init.org= into =init.el=, and optionally byte-compile
44 First, clone the repository and =cd= into it:
46 #+begin_src sh :tangle no
47 git clone https://git.sr.ht/~bandali/dotfiles ~/.emacs.d
51 Then, decide if you would like to use a byte-compiled init file, and
52 set the [[#byte-compiled-init][a/byte-compiled-init]] variable accordingly.
54 Now, first tangle =init.org=, and only if you chose to have a
55 byte-compiled init, build init as well:
57 #+begin_src sh :tangle no
62 If you'd like to use a byte-compiled init, it's important that it be
63 recompiled whenever =init.el= is generated from an updated =init.org=.
64 Not only does my setup automatically and asynchronously tangle
65 =init.org= to =init.el= every time you edit and save =init.org= in GNU
66 Emacs, it will also invoke =make build-init= if you set
67 =a/byte-compiled-init= to =t= above, so you wouldn't have to worry
68 about manually tangling and compiling your init file whenever you
69 change it. The output of the last byte-compilation in the current
70 session is kept in a =*compilation*= buffer, which will automatically
71 be displayed if compilation fails.
73 * Contents :toc_1:noexport:
77 - [[#initial-setup][Initial setup]]
79 - [[#borg-essentials][Borg's =layer/essentials=]]
80 - [[#editing][Editing]]
81 - [[#syntax-spell-checking][Syntax and spell checking]]
82 - [[#programming-modes][Programming modes]]
83 - [[#emacs-enhancements][Emacs enhancements]]
85 - [[#blogging][Blogging]]
86 - [[#post-initialization][Post initialization]]
96 #+begin_src emacs-lisp :comments none
97 ;;; init.el --- Amin Bandali's Emacs config -*- lexical-binding: t; eval: (view-mode 1) -*-
100 Enable =view-mode=, which both makes the file read-only (as a reminder
101 that =init.el= is an auto-generated file, not supposed to be edited),
102 and provides some convenient key bindings for browsing through the
107 #+begin_src emacs-lisp :comments none
108 ;; Copyright (C) 2018-2019 Amin Bandali <bandali@gnu.org>
110 ;; This program is free software: you can redistribute it and/or modify
111 ;; it under the terms of the GNU General Public License as published by
112 ;; the Free Software Foundation, either version 3 of the License, or
113 ;; (at your option) any later version.
115 ;; This program is distributed in the hope that it will be useful,
116 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
117 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
118 ;; GNU General Public License for more details.
120 ;; You should have received a copy of the GNU General Public License
121 ;; along with this program. If not, see <https://www.gnu.org/licenses/>.
126 #+begin_src emacs-lisp :comments none
129 ;; Emacs configuration of Amin Bandali, computer scientist, functional
130 ;; programmer, and free software advocate.
132 ;; THIS FILE IS AUTO-GENERATED FROM `init.org'.
137 :CUSTOM_ID: initial-setup
140 ** Byte-compiled init preference
142 :CUSTOM_ID: byte-compiled-init
145 If you would like a byte-compiled init file, set the following
146 variable to ~t~, otherwise set it to ~nil~.
148 #+begin_src emacs-lisp
149 (defvar a/byte-compiled-init t
150 "If non-nil, byte-(re)compile init.el on successful tangles.")
153 You can click on [[#installation][Installation]] to jump back up there if you like :)
155 ** Emacs initialization
157 I'd like to do a couple of measurements of Emacs' startup time. First,
158 let's see how long Emacs takes to start up, before even loading
159 =init.el=, i.e. =user-init-file=:
161 #+begin_src emacs-lisp
162 (defvar a/before-user-init-time (current-time)
163 "Value of `current-time' when Emacs begins loading `user-init-file'.")
164 (message "Loading Emacs...done (%.3fs)"
165 (float-time (time-subtract a/before-user-init-time
169 Also, temporarily increase ~gc-cons-threshhold~ and
170 ~gc-cons-percentage~ during startup to reduce garbage collection
171 frequency. Clearing the ~file-name-handler-alist~ seems to help reduce
172 startup time as well.
174 #+begin_src emacs-lisp
175 (defvar a/gc-cons-threshold gc-cons-threshold)
176 (defvar a/gc-cons-percentage gc-cons-percentage)
177 (defvar a/file-name-handler-alist file-name-handler-alist)
178 (setq gc-cons-threshold (* 400 1024 1024) ; 400 MiB
179 gc-cons-percentage 0.6
180 file-name-handler-alist nil
181 ;; sidesteps a bug when profiling with esup
182 esup-child-profile-require-level 0)
185 Of course, we'd like to set them back to their defaults once we're
188 #+begin_src emacs-lisp
192 (setq gc-cons-threshold a/gc-cons-threshold
193 gc-cons-percentage a/gc-cons-percentage
194 file-name-handler-alist a/file-name-handler-alist)))
197 Increase the number of lines kept in message logs (the =*Messages*=
200 #+begin_src emacs-lisp
201 (setq message-log-max 20000)
204 Optionally, we could suppress some byte compiler warnings like below,
205 but for now I've decided to keep them enabled. See documentation for
206 ~byte-compile-warnings~ for more details.
208 #+begin_src emacs-lisp
209 ;; (setq byte-compile-warnings
210 ;; '(not free-vars unresolved noruntime lexical make-local))
215 #+begin_src emacs-lisp
216 (setq user-full-name "Amin Bandali"
217 user-mail-address "amin@bndl.org")
220 ** Package management
224 I can do all my package management things with =straight.el=, and
225 don't need Emacs' built-in =package.el=. Emacs 27 lets us disable
226 =package.el= in the =early-init-file= (see [[https://git.savannah.gnu.org/cgit/emacs.git/commit/?id=24acb31c04b4048b85311d794e600ecd7ce60d3b][here]]).
228 #+begin_src emacs-lisp :tangle early-init.el
229 (setq package-enable-at-startup nil)
232 But since Emacs 27 isn't out yet (Emacs 26 is just around the corner
233 right now), and even when released it'll be long before most distros
234 ship in their repos, I'll still put the old workaround with the
235 commented call to ~package-initialize~ here anyway.
237 #+begin_src emacs-lisp :tangle no
238 (setq package-enable-at-startup nil)
239 ;; (package-initialize)
242 Update: the above is not necessary, since =straight.el= automatically
243 does that (and more). See =straight-package-neutering-mode=.
248 Next-generation, purely functional package manager for the Emacs
252 =straight.el= allows me to have a fully reproducible Emacs setup.
254 #+begin_src emacs-lisp
255 ;; Main engine start...
257 (setq straight-repository-branch "develop"
258 straight-check-for-modifications '(check-on-save find-when-checking))
260 (defun a/bootstrap-straight ()
261 (defvar bootstrap-version)
262 (let ((bootstrap-file
263 (expand-file-name "straight/repos/straight.el/bootstrap.el" user-emacs-directory))
264 (bootstrap-version 5))
265 (unless (file-exists-p bootstrap-file)
267 (url-retrieve-synchronously
268 "https://raw.githubusercontent.com/raxod502/straight.el/develop/install.el"
269 'silent 'inhibit-cookies)
270 (goto-char (point-max))
271 (eval-print-last-sexp)))
272 (load bootstrap-file nil 'nomessage)))
274 ;; Solid rocket booster ignition...
276 (defun a/build-init ()
277 (a/bootstrap-straight)
278 (byte-compile-file "init.el"))
280 (a/bootstrap-straight)
284 (setq straight-use-package-by-default t)
287 Since we enable =straight.el='s =straight-use-package-by-default=
288 integration, we will define a =use-feature= for plain ole
289 =use-package= without any of the =straight.el= stuff.
291 #+begin_src emacs-lisp
292 (defmacro use-feature (name &rest args)
293 "Like `use-package', but with `straight-use-package-by-default' disabled."
294 (declare (indent defun))
303 A use-package declaration for simplifying your .emacs
306 [[https://github.com/jwiegley/use-package][use-package]] is an awesome utility for managing and configuring
307 packages (in our case especially the latter) in a neatly organized way
308 and without compromising on performance.
310 #+begin_src emacs-lisp
311 (straight-use-package 'use-package)
312 (if nil ; set to t when need to debug init
314 (setq use-package-verbose t
315 use-package-expand-minimally nil
316 use-package-compute-statistics t
318 (require 'use-package))
319 (setq use-package-verbose nil
320 use-package-expand-minimally t))
322 (setq use-package-always-defer t)
329 Browse the Emacsmirror package database
332 Epkg provides access to a local copy of the [[https://emacsmirror.net][Emacsmirror]] package
333 database, low-level functions for querying the database, and a
334 =package.el=-like user interface for browsing the available packages.
336 #+begin_src emacs-lisp
338 :commands (epkg-list-packages epkg-describe-package)
340 (("C-c p e d" . epkg-describe-package)
341 ("C-c p e p" . epkg-list-packages))
343 (setq epkg-repository "~/.emacs.d/straight/repos/epkgs/")
344 (eval-when-compile (defvar ivy-initial-inputs-alist))
345 (with-eval-after-load 'ivy
347 'ivy-initial-inputs-alist '(epkg-describe-package . "^") t)))
350 ** No littering in =~/.emacs.d=
353 Help keeping ~/.emacs.d clean
356 By default, even for Emacs' built-in packages, the configuration files
357 and persistent data are all over the place. Use =no-littering= to help
360 #+begin_src emacs-lisp
361 (use-package no-littering
365 (add-to-list 'savehist-additional-variables 'kill-ring)
367 (setq auto-save-file-name-transforms
368 `((".*" ,(no-littering-expand-var-file-name "auto-save/") t))))
371 ** Custom file (=custom.el=)
373 I'm not planning on using the custom file much, but even so, I
374 definitely don't want it mixing with =init.el=. So, here; let's give
375 it it's own file. While at it, treat themes as safe.
377 #+begin_src emacs-lisp
381 (setq custom-file (no-littering-expand-etc-file-name "custom.el"))
382 (when (file-exists-p custom-file)
384 (setf custom-safe-themes t))
389 Load the secrets file if it exists, otherwise show a warning.
391 #+begin_src emacs-lisp
393 (load (no-littering-expand-etc-file-name "secrets")))
396 ** Better =$PATH= handling
398 Let's use [[https://github.com/purcell/exec-path-from-shell][exec-path-from-shell]] to make Emacs use the =$PATH= as set up
401 #+begin_src emacs-lisp
402 (use-package exec-path-from-shell
405 (setq exec-path-from-shell-arguments nil
406 exec-path-from-shell-check-startup-files nil)
408 (exec-path-from-shell-initialize)
409 ;; while we're at it, let's fix access to our running ssh-agent
410 (exec-path-from-shell-copy-env "SSH_AGENT_PID")
411 (exec-path-from-shell-copy-env "SSH_AUTH_SOCK"))
414 ** COMMENT Only one custom theme at a time
416 #+begin_src emacs-lisp
417 (defadvice load-theme (before clear-previous-themes activate)
418 "Clear existing theme settings instead of layering them"
419 (mapc #'disable-theme custom-enabled-themes))
424 Start server if not already running. Alternatively, can be done by
425 issuing =emacs --daemon= in the terminal, which can be automated with
426 a systemd service or using =brew services start emacs= on macOS. I use
427 Emacs as my window manager (via EXWM), so I always start Emacs on
428 login; so starting the server from inside Emacs is good enough for me.
430 See [[https://www.gnu.org/software/emacs/manual/html_node/emacs/Emacs-Server.html#Emacs-Server][Using Emacs as a Server]].
432 #+begin_src emacs-lisp
435 :config (or (server-running-p) (server-mode)))
438 ** COMMENT Unicode support
440 Font stack with better unicode support, around =Ubuntu Mono= and
443 #+begin_src emacs-lisp
444 (dolist (ft (fontset-list))
448 (font-spec :name "Source Code Pro" :size 14))
452 (font-spec :name "DejaVu Sans Mono")
459 ;; :name "Symbola monospacified for DejaVu Sans Mono")
465 ;; (font-spec :name "DejaVu Sans Mono")
471 (font-spec :name "DejaVu Sans Mono" :size 14)
476 ** Gentler font resizing
478 #+begin_src emacs-lisp
479 (setq text-scale-mode-step 1.05)
482 ** Focus follows mouse
484 I’d like focus to follow the mouse when I move the cursor from one
487 #+begin_src emacs-lisp
488 (setq mouse-autoselect-window t)
491 Let’s define a function to conveniently disable this for certain
492 buffers and/or modes.
494 #+begin_src emacs-lisp
495 (defun a/no-mouse-autoselect-window ()
496 (make-local-variable 'mouse-autoselect-window)
497 (setq mouse-autoselect-window nil))
500 ** Better scrolling (arguably)
502 #+begin_src emacs-lisp
503 (setq ;; scroll-margin 1
504 ;; scroll-conservatively 10000
506 scroll-conservatively 10
507 scroll-preserve-screen-position 1)
512 (setq mouse-wheel-scroll-amount '(1 ((shift) . 1)) ; one line at a time
513 mouse-wheel-progressive-speed nil ; don't accelerate scrolling
514 mouse-wheel-follow-mouse t)) ; scroll window under mouse
516 (use-feature pixel-scroll
518 :config (pixel-scroll-mode 1))
521 ** Ask for GPG passphrase in minibuffer
523 #+begin_src emacs-lisp
524 (setq epg-pinentry-mode 'loopback)
529 #+begin_src emacs-lisp
536 Convenience macro for =setq='ing multiple variables to the same value:
538 #+begin_src emacs-lisp
539 (defmacro a/setq-every (value &rest vars)
540 "Set all the variables from VARS to value VALUE."
541 (declare (indent defun) (debug t))
542 `(progn ,@(mapcar (lambda (x) (list 'setq x value)) vars)))
545 The following process-related stuff from [[https://github.com/alezost/emacs-config][alezost's emacs-config]].
547 #+begin_src emacs-lisp
548 (defun a/start-process (program &rest args)
549 "Same as `start-process', but doesn't bother about name and buffer."
550 (let ((process-name (concat program "_process"))
551 (buffer-name (generate-new-buffer-name
552 (concat program "_output"))))
553 (apply #'start-process
554 process-name buffer-name program args)))
556 (defun a/dired-start-process (program &optional args)
557 "Open current file with a PROGRAM."
558 ;; Shell command looks like this: "program [ARGS]... FILE" (ARGS can
559 ;; be nil, so remove it).
560 (apply #'a/start-process
562 (remove nil (list args (dired-get-file-for-visit)))))
572 *** Time and battery in mode-line
574 Enable displaying time and battery in the mode-line, since I'm not
575 using the Xfce panel anymore. Also, I don't need to see the load
576 average on a regular basis, so disable that.
578 Note: using =i3status= on sway at the moment, so disabling this.
580 #+begin_src emacs-lisp :tangle no
583 (setq display-time-default-load-average nil)
589 (display-battery-mode))
594 Might want to set the fringe to a smaller value, especially if using
595 EXWM. I'm fine with the default for now.
597 #+begin_src emacs-lisp
598 ;; (fringe-mode '(3 . 1))
602 *** Disable disabled commands
604 Emacs disables some commands by default that could persumably be
605 confusing for novice users. Let's disable that.
607 #+begin_src emacs-lisp
608 (setq disabled-command-function nil)
613 Save what I copy into clipboard from other applications into Emacs'
614 kill-ring, which would allow me to still be able to easily access it
615 in case I kill (cut or copy) something else inside Emacs before
616 yanking (pasting) what I'd originally intended to.
618 #+begin_src emacs-lisp
619 (setq save-interprogram-paste-before-kill t)
624 #+begin_src emacs-lisp
625 (setq enable-recursive-minibuffers t
626 resize-mini-windows t)
629 *** Lazy-person-friendly yes/no prompts
631 Lazy people would prefer to type fewer keystrokes, especially for yes
632 or no questions. I'm lazy.
634 #+begin_src emacs-lisp
635 (defalias 'yes-or-no-p #'y-or-n-p)
638 *** Startup screen and =*scratch*=
640 Firstly, let Emacs know that I'd like to have =*scratch*= as my
643 #+begin_src emacs-lisp
644 (setq initial-buffer-choice t)
647 Now let's customize the =*scratch*= buffer a bit. First off, I don't
648 need the default hint.
650 #+begin_src emacs-lisp
651 (setq initial-scratch-message nil)
654 Also, let's use Text mode as the major mode, in case I want to
655 customize it (=*scratch*='s default major mode, Fundamental mode,
656 can't really be customized).
658 #+begin_src emacs-lisp
659 (setq initial-major-mode 'text-mode)
662 Inhibit the buffer list when more than 2 files are loaded.
664 #+begin_src emacs-lisp
665 (setq inhibit-startup-buffer-menu t)
668 I don't really need to see the startup screen or echo area message
671 #+begin_src emacs-lisp
672 (advice-add #'display-startup-echo-area-message :override #'ignore)
673 (setq inhibit-startup-screen t
674 inhibit-startup-echo-area-message user-login-name)
677 *** More useful frame titles
679 Show either the file name or the buffer name (in case the buffer isn't
680 visiting a file). Borrowed from Emacs Prelude.
682 #+begin_src emacs-lisp
683 (setq frame-title-format
684 '("" invocation-name " - "
685 (:eval (if (buffer-file-name)
686 (abbreviate-file-name (buffer-file-name))
692 Emacs' default backup settings aren't that great. Let's use more
693 sensible options. See documentation for the ~make-backup-file~
696 #+begin_src emacs-lisp
697 (setq backup-by-copying t
699 delete-old-versions t)
704 Enable automatic reloading of changed buffers and files.
706 #+begin_src emacs-lisp
707 (global-auto-revert-mode 1)
708 (setq auto-revert-verbose nil
709 global-auto-revert-non-file-buffers nil)
712 *** Always use space for indentation
714 #+begin_src emacs-lisp
717 require-final-newline t
723 Enable =winner-mode=.
725 #+begin_src emacs-lisp
729 *** Don’t display =*compilation*= on success
731 Based on https://stackoverflow.com/a/17788551, with changes to use
732 =cl-letf= instead of the now obsolete =flet=.
734 #+begin_src emacs-lisp
735 (with-eval-after-load 'compile
736 (defun a/compilation-finish-function (buffer outstr)
737 (unless (string-match "finished" outstr)
738 (switch-to-buffer-other-window buffer))
741 (setq compilation-finish-functions #'a/compilation-finish-function)
745 (defadvice compilation-start
746 (around inhibit-display
747 (command &optional mode name-function highlight-regexp))
748 (if (not (string-match "^\\(find\\|grep\\)" command))
749 (cl-letf (((symbol-function 'display-buffer) #'ignore))
750 (save-window-excursion ad-do-it))
752 (ad-activate 'compilation-start))
755 *** Search for non-ASCII characters
757 I’d like non-ASCII characters such as ‘’“”«»‹›áⓐ𝒶 to be selected when
758 I search for their ASCII counterpart. Shoutout to [[http://endlessparentheses.com/new-in-emacs-25-1-easily-search-non-ascii-characters.html][endlessparentheses]]
761 #+begin_src emacs-lisp
762 (setq search-default-mode #'char-fold-to-regexp)
764 ;; uncomment to extend this behaviour to query-replace
765 ;; (setq replace-char-fold t)
770 #+begin_src emacs-lisp
771 (setq-default cursor-type 'bar)
774 *** Allow scrolling in Isearch
776 #+begin_src emacs-lisp
777 (setq isearch-allow-scroll t)
782 Some bindings for functions from built-in GNU Emacs packages:
784 #+begin_src emacs-lisp
788 ("C-c e b" . eval-buffer)
789 ("C-c e r" . eval-region)
791 ("C-c e i" . emacs-init-time)
792 ("C-c e u" . emacs-uptime)
794 ("C-c F m" . make-frame-command)
795 ("C-c F d" . delete-frame)
796 ("C-c F D" . delete-other-frames)
798 ("C-c o" . other-window)
800 ("C-S-h C" . describe-char)
801 ("C-S-h F" . describe-face)
803 ("C-x k" . kill-this-buffer)
804 ("C-x K" . kill-buffer)
806 ("s-p" . beginning-of-buffer)
807 ("s-n" . end-of-buffer))
809 (when (display-graphic-p)
810 (unbind-key "C-z" global-map))
813 While at it, let's bind a few for some =straight-*= functions too:
815 #+begin_src emacs-lisp
817 :prefix-map a/straight-prefix-map
819 ("u" . straight-use-package)
820 ("f" . straight-freeze-versions)
821 ("t" . straight-thaw-versions)
822 ("P" . straight-prune-build)
823 ("r" . straight-get-recipe)
824 ;; M-x ^straight-.*-all$
825 ("a c" . straight-check-all)
826 ("a f" . straight-fetch-all)
827 ("a m" . straight-merge-all)
828 ("a n" . straight-normalize-all)
829 ("a F" . straight-pull-all)
830 ("a P" . straight-push-all)
831 ("a r" . straight-rebuild-all)
832 ;; M-x ^straight-.*-package$
833 ("p c" . straight-check-package)
834 ("p f" . straight-fetch-package)
835 ("p m" . straight-merge-package)
836 ("p n" . straight-normalize-package)
837 ("p F" . straight-pull-package)
838 ("p P" . straight-push-package)
839 ("p r" . straight-rebuild-package))
844 The packages in this section are absolutely essential to my everyday
845 workflow, and they play key roles in how I do my computing. They
846 immensely enhance the Emacs experience for me; both using Emacs, and
849 *** [[https://github.com/emacscollective/auto-compile][auto-compile]]
851 #+begin_src emacs-lisp
852 (use-package auto-compile
855 (auto-compile-on-load-mode)
856 (auto-compile-on-save-mode)
857 (setq auto-compile-display-buffer nil
858 auto-compile-mode-line-counter t
859 auto-compile-source-recreate-deletes-dest t
860 auto-compile-toggle-deletes-nonlib-dest t
861 auto-compile-update-autoloads t)
862 (add-hook 'auto-compile-inhibit-compile-hook
863 'auto-compile-inhibit-compile-detached-git-head))
866 *** [[https://orgmode.org/][Org]]
869 Org mode is for keeping notes, maintaining TODO lists, planning
870 projects, and authoring documents with a fast and effective plain-text
874 In short, my favourite way of life.
876 We will use the =org-plus-contrib= package to get the whole deal:
878 #+begin_src emacs-lisp
879 (straight-use-package 'org-plus-contrib)
882 And here's where my actual Org configurations begin:
884 #+begin_src emacs-lisp
888 (setq org-src-tab-acts-natively t
889 org-src-preserve-indentation nil
890 org-edit-src-content-indentation 0
891 org-email-link-description-format "Email %c: %s" ; %.30s
892 org-highlight-latex-and-related '(entities)
893 org-use-speed-commands t
894 org-startup-folded 'content
895 org-catch-invisible-edits 'show-and-error
897 (add-to-list 'org-structure-template-alist '("L" . "src emacs-lisp") t)
898 :bind (:map org-mode-map ("M-L" . org-insert-last-stored-link))
899 :hook ((org-mode . org-indent-mode)
900 (org-mode . auto-fill-mode)
901 (org-mode . flyspell-mode))
903 (org-latex-packages-alist '(("" "listings") ("" "color")))
905 '(org-block-begin-line ((t (:foreground "#5a5b5a" :background "#1d1f21"))))
906 '(org-block ((t (:background "#1d1f21"))))
907 '(org-latex-and-related ((t (:foreground "#b294bb")))))
909 (use-feature ox-latex
912 (setq org-latex-listings 'listings
913 ;; org-latex-prefer-user-labels t
915 (add-to-list 'org-latex-classes
916 '("IEEEtran" "\\documentclass[11pt]{IEEEtran}"
917 ("\\section{%s}" . "\\section*{%s}")
918 ("\\subsection{%s}" . "\\subsection*{%s}")
919 ("\\subsubsection{%s}" . "\\subsubsection*{%s}")
920 ("\\paragraph{%s}" . "\\paragraph*{%s}")
921 ("\\subparagraph{%s}" . "\\subparagraph*{%s}"))
923 (require 'ox-beamer))
926 **** asynchronous tangle
928 =a/async-babel-tangle= is a function closely inspired by [[https://github.com/dieggsy/dotfiles/tree/cc10edf7701958eff1cd94d4081da544d882a28c/emacs.d#dotfiles][dieggsy's
929 d/async-babel-tangle]] which uses [[https://github.com/jwiegley/emacs-async][async]] to asynchronously tangle an org
932 #+begin_src emacs-lisp
933 (with-eval-after-load 'org
934 (defvar a/show-async-tangle-results nil
935 "Keep *emacs* async buffers around for later inspection.")
937 (defvar a/show-async-tangle-time nil
938 "Show the time spent tangling the file.")
940 (defvar a/async-tangle-post-compile
941 (when a/byte-compiled-init "make build-init")
942 "If non-nil, pass to `compile' after successful tangle.")
944 ;; TODO: look into why directly byte-compiling init.el causes a
945 ;; number of problems, including magit-status not loading (busy
947 (defvar a/async-tangle-byte-recompile nil
948 "If non-nil, byte-recompile the file on successful tangle.")
950 (defun a/async-babel-tangle ()
951 "Tangle org file asynchronously."
953 (let* ((file-tangle-start-time (current-time))
954 (file (buffer-file-name))
955 (file-nodir (file-name-nondirectory file))
956 ;; (async-quiet-switch "-q")
957 (file-noext (file-name-sans-extension file)))
961 (org-babel-tangle-file ,file))
962 (unless a/show-async-tangle-results
966 ;; (setq byte-compile-warnings '(not noruntime unresolved))
967 (message "Tangled %s%s"
969 (if a/show-async-tangle-time
971 (float-time (time-subtract (current-time)
972 ',file-tangle-start-time)))
974 (when a/async-tangle-post-compile
975 (compile a/async-tangle-post-compile))
976 (when a/async-tangle-byte-recompile
977 (byte-recompile-file (concat ,file-noext ".el"))))
978 (message "Tangling %s failed" ,file-nodir))))))))
981 'safe-local-variable-values
982 '(eval add-hook 'after-save-hook #'a/async-babel-tangle 'append 'local))
985 *** [[https://magit.vc/][Magit]]
988 It's Magit! A Git porcelain inside Emacs.
991 Not just how I do git, but /the/ way to do git.
993 #+begin_src emacs-lisp
996 :bind (("C-x g" . magit-status)
997 ("s-g s" . magit-status)
998 ("s-g l" . magit-log-buffer-file))
1000 (magit-add-section-hook 'magit-status-sections-hook
1001 'magit-insert-modules
1002 'magit-insert-stashes
1005 magit-repository-directories '(("~/.emacs.d/" . 0)
1006 ("~/src/git/" . 1)))
1007 (nconc magit-section-initial-visibility-alist
1008 '(([unpulled status] . show)
1009 ([unpushed status] . show)))
1010 :custom-face (magit-diff-file-heading ((t (:weight normal)))))
1015 Recently opened files.
1017 #+begin_src emacs-lisp
1018 (use-feature recentf
1021 (add-to-list 'recentf-exclude "^/\\(?:ssh\\|su\\|sudo\\)?:")
1022 (setq recentf-max-saved-items 40))
1028 A smart M-x enhancement for Emacs.
1031 Mostly because =counsel= needs it to remember history.
1033 #+begin_src emacs-lisp
1037 *** [[https://github.com/abo-abo/swiper][Ivy]] (and friends)
1040 Ivy - a generic completion frontend for Emacs, Swiper - isearch with
1041 an overview, and more. Oh, man!
1044 There's no way I could top that, so I won't attempt to.
1048 #+begin_src emacs-lisp
1052 (:map ivy-minibuffer-map
1053 ([escape] . keyboard-escape-quit)
1054 ([S-up] . ivy-previous-history-element)
1055 ([S-down] . ivy-next-history-element)
1056 ("DEL" . ivy-backward-delete-char))
1061 ;; (ivy-minibuffer-match-face-2 ((t (:background "#e99ce8" :weight semi-bold))))
1062 ;; (ivy-minibuffer-match-face-3 ((t (:background "#bbbbff" :weight semi-bold))))
1063 ;; (ivy-minibuffer-match-face-4 ((t (:background "#ffbbff" :weight semi-bold))))
1069 #+begin_src emacs-lisp
1072 :bind (("C-s" . swiper)
1074 ("C-S-s" . isearch-forward)))
1079 #+begin_src emacs-lisp
1080 (use-package counsel
1082 :bind (([remap execute-extended-command] . counsel-M-x)
1083 ([remap find-file] . counsel-find-file)
1084 ("s-r" . counsel-recentf)
1085 ("C-c x" . counsel-M-x)
1086 ("C-c f ." . counsel-find-file)
1087 :map minibuffer-local-map
1088 ("C-r" . counsel-minibuffer-history))
1091 (defalias 'locate #'counsel-locate))
1096 #+begin_src emacs-lisp
1100 :bind ("C-c a s e" . eshell)
1102 (eval-when-compile (defvar eshell-prompt-regexp))
1103 (defun a/eshell-quit-or-delete-char (arg)
1105 (if (and (eolp) (looking-back eshell-prompt-regexp nil))
1106 (eshell-life-is-too-much)
1109 (defun a/eshell-clear ()
1111 (let ((inhibit-read-only t))
1113 (eshell-send-input))
1115 (defun a/eshell-setup ()
1116 (make-local-variable 'company-idle-delay)
1117 (defvar company-idle-delay)
1118 (setq company-idle-delay nil)
1119 (bind-keys :map eshell-mode-map
1120 ("C-d" . a/eshell-quit-or-delete-char)
1121 ("C-S-l" . a/eshell-clear)
1122 ("M-r" . counsel-esh-history)
1123 ([tab] . company-complete)))
1125 :hook (eshell-mode . a/eshell-setup)
1127 (eshell-hist-ignoredups t)
1128 (eshell-input-filter 'eshell-input-filter-initial-space))
1133 #+begin_src emacs-lisp
1134 (use-feature ibuffer
1136 (("C-x C-b" . ibuffer-other-window)
1137 :map ibuffer-mode-map
1138 ("P" . ibuffer-backward-filter-group)
1139 ("N" . ibuffer-forward-filter-group)
1140 ("M-p" . ibuffer-do-print)
1141 ("M-n" . ibuffer-do-shell-command-pipe-replace))
1143 ;; Use human readable Size column instead of original one
1144 (define-ibuffer-column size-h
1145 (:name "Size" :inline t)
1147 ((> (buffer-size) 1000000) (format "%7.1fM" (/ (buffer-size) 1000000.0)))
1148 ((> (buffer-size) 100000) (format "%7.0fk" (/ (buffer-size) 1000.0)))
1149 ((> (buffer-size) 1000) (format "%7.1fk" (/ (buffer-size) 1000.0)))
1150 (t (format "%8d" (buffer-size)))))
1152 (ibuffer-saved-filter-groups
1154 ("dired" (mode . dired-mode))
1155 ("org" (mode . org-mode))
1158 (mode . gnus-group-mode)
1159 (mode . gnus-summary-mode)
1160 (mode . gnus-article-mode)
1161 ;; not really, but...
1162 (mode . message-mode)))
1171 (mode . eshell-mode)
1173 (mode . term-mode)))
1176 (mode . python-mode)
1180 (mode . emacs-lisp-mode)
1181 (mode . scheme-mode)
1182 (mode . haskell-mode)
1183 (mode . lean-mode)))
1186 (name . "^\\*scratch\\*$")
1187 (name . "^\\*Messages\\*$")))
1188 ("erc" (mode . erc-mode)))))
1190 '((mark modified read-only locked " "
1191 (name 18 18 :left :elide)
1193 (size-h 9 -1 :right)
1195 (mode 16 16 :left :elide)
1196 " " filename-and-process)
1200 :hook (ibuffer . (lambda () (ibuffer-switch-to-saved-filter-groups "default"))))
1205 #+begin_src emacs-lisp
1206 (use-feature outline
1207 :hook (prog-mode . outline-minor-mode)
1210 outline-minor-mode-map
1211 ("<s-tab>" . outline-toggle-children)
1212 ("M-p" . outline-previous-visible-heading)
1213 ("M-n" . outline-next-visible-heading)
1214 :prefix-map a/outline-prefix-map
1216 ("TAB" . outline-toggle-children)
1217 ("a" . outline-hide-body)
1218 ("H" . outline-hide-body)
1219 ("S" . outline-show-all)
1220 ("h" . outline-hide-subtree)
1221 ("s" . outline-show-subtree)))
1226 #+begin_src emacs-lisp
1227 (use-feature ls-lisp
1228 :custom (ls-lisp-dirs-first t))
1232 (setq dired-listing-switches "-alh"
1233 ls-lisp-use-insert-directory-program nil)
1235 ;; easily diff 2 marked files
1236 ;; https://oremacs.com/2017/03/18/dired-ediff/
1237 (defun dired-ediff-files ()
1239 (require 'dired-aux)
1240 (defvar ediff-after-quit-hook-internal)
1241 (let ((files (dired-get-marked-files))
1242 (wnd (current-window-configuration)))
1243 (if (<= (length files) 2)
1244 (let ((file1 (car files))
1245 (file2 (if (cdr files)
1249 (dired-dwim-target-directory)))))
1250 (if (file-newer-than-file-p file1 file2)
1251 (ediff-files file2 file1)
1252 (ediff-files file1 file2))
1253 (add-hook 'ediff-after-quit-hook-internal
1255 (setq ediff-after-quit-hook-internal nil)
1256 (set-window-configuration wnd))))
1257 (error "no more than 2 files should be marked"))))
1258 :bind (:map dired-mode-map
1259 ("b" . dired-up-directory)
1260 ("e" . dired-ediff-files)
1261 ("E" . dired-toggle-read-only)
1262 ("\\" . dired-hide-details-mode)
1265 (a/dired-start-process "zathura"))))
1266 :hook (dired-mode . dired-hide-details-mode))
1271 #+begin_src emacs-lisp
1274 (temp-buffer-resize-mode)
1275 (setq help-window-select t))
1280 #+begin_src emacs-lisp
1283 (add-to-list 'tramp-default-proxies-alist '(nil "\\`root\\'" "/ssh:%h:"))
1284 (add-to-list 'tramp-default-proxies-alist '("localhost" nil nil))
1285 (add-to-list 'tramp-default-proxies-alist
1286 (list (regexp-quote (system-name)) nil nil)))
1291 #+begin_src emacs-lisp
1293 :config (dash-enable-font-lock))
1303 Highlight uncommitted changes in the left fringe.
1305 #+begin_src emacs-lisp
1306 (use-package diff-hl
1308 (setq diff-hl-draw-borders nil)
1309 (global-diff-hl-mode)
1310 :hook (magit-post-refresh . diff-hl-magit-post-refresh))
1315 Display Lisp objects at point in the echo area.
1317 #+begin_src emacs-lisp
1319 :when (version< "25" emacs-version)
1320 :config (global-eldoc-mode))
1325 Highlight matching parens.
1327 #+begin_src emacs-lisp
1330 :config (show-paren-mode))
1333 ** simple (for column numbers)
1335 #+begin_src emacs-lisp
1337 :config (column-number-mode))
1342 Save minibuffer history.
1344 #+begin_src emacs-lisp
1345 (use-feature savehist
1346 :config (savehist-mode))
1351 Automatically save place in each file.
1353 #+begin_src emacs-lisp
1354 (use-feature saveplace
1355 :when (version< "25" emacs-version)
1356 :config (save-place-mode))
1361 #+begin_src emacs-lisp
1362 (use-feature prog-mode
1363 :config (global-prettify-symbols-mode)
1364 (defun indicate-buffer-boundaries-left ()
1365 (setq indicate-buffer-boundaries 'left))
1366 (add-hook 'prog-mode-hook #'indicate-buffer-boundaries-left))
1371 #+begin_src emacs-lisp
1372 (use-feature text-mode
1373 :hook ((text-mode . indicate-buffer-boundaries-left)
1374 (text-mode . abbrev-mode)))
1379 #+begin_src emacs-lisp
1380 (use-package company
1383 (:map company-active-map
1384 ([tab] . company-complete-common-or-cycle)
1385 ([escape] . company-abort))
1387 (company-minimum-prefix-length 1)
1388 (company-selection-wrap-around t)
1389 (company-dabbrev-char-regexp "\\sw\\|\\s_\\|[-_]")
1390 (company-dabbrev-downcase nil)
1391 (company-dabbrev-ignore-case nil)
1393 (global-company-mode t))
1398 #+begin_src emacs-lisp
1399 (use-package flycheck
1401 :hook (prog-mode . flycheck-mode)
1403 (:map flycheck-mode-map
1404 ("M-P" . flycheck-previous-error)
1405 ("M-N" . flycheck-next-error))
1407 ;; Use the load-path from running Emacs when checking elisp files
1408 (setq flycheck-emacs-lisp-load-path 'inherit)
1410 ;; Only flycheck when I actually save the buffer
1411 (setq flycheck-check-syntax-automatically '(mode-enabled save)))
1413 ;; http://endlessparentheses.com/ispell-and-apostrophes.html
1417 ;; ’ can be part of a word
1418 (setq ispell-local-dictionary-alist
1419 `((nil "[[:alpha:]]" "[^[:alpha:]]"
1420 "['\x2019]" nil ("-B") nil utf-8)))
1421 ;; don't send ’ to the subprocess
1422 (defun endless/replace-apostrophe (args)
1423 (cons (replace-regexp-in-string
1426 (advice-add #'ispell-send-string :filter-args
1427 #'endless/replace-apostrophe)
1429 ;; convert ' back to ’ from the subprocess
1430 (defun endless/replace-quote (args)
1431 (if (not (derived-mode-p 'org-mode))
1433 (cons (replace-regexp-in-string
1436 (advice-add #'ispell-parse-output :filter-args
1437 #'endless/replace-quote))
1442 :CUSTOM_ID: programming-modes
1447 #+begin_src emacs-lisp
1448 (use-feature lisp-mode
1450 (add-hook 'emacs-lisp-mode-hook 'outline-minor-mode)
1451 (add-hook 'emacs-lisp-mode-hook 'reveal-mode)
1452 (defun indent-spaces-mode ()
1453 (setq indent-tabs-mode nil))
1454 (add-hook 'lisp-interaction-mode-hook #'indent-spaces-mode))
1457 ** [[http://alloytools.org][Alloy]] (with [[https://github.com/dwwmmn/alloy-mode][alloy-mode]])
1459 #+begin_src emacs-lisp
1460 (use-package alloy-mode
1461 :straight (:host github :repo "dwwmmn/alloy-mode")
1462 :config (setq alloy-basic-offset 2))
1465 ** [[https://coq.inria.fr][Coq]] (with [[https://github.com/ProofGeneral/PG][Proof General]])
1467 #+begin_src emacs-lisp
1468 (use-package proof-site ; Proof General
1469 :straight proof-general)
1472 ** [[https://leanprover.github.io][Lean]] (with [[https://github.com/leanprover/lean-mode][lean-mode]])
1474 #+begin_src emacs-lisp
1475 (eval-when-compile (defvar lean-mode-map))
1476 (use-package lean-mode
1478 :bind (:map lean-mode-map
1479 ("S-SPC" . company-complete))
1481 (require 'lean-input)
1482 (setq default-input-method "Lean"
1483 lean-input-tweak-all '(lean-input-compose
1484 (lean-input-prepend "/")
1485 (lean-input-nonempty))
1486 lean-input-user-translations '(("/" "/")))
1492 *** [[https://github.com/haskell/haskell-mode][haskell-mode]]
1494 #+begin_src emacs-lisp
1495 (use-package haskell-mode
1497 (setq haskell-indentation-layout-offset 4
1498 haskell-indentation-left-offset 4
1499 flycheck-checker 'haskell-hlint
1500 flycheck-disabled-checkers '(haskell-stack-ghc haskell-ghc)))
1503 *** [[https://github.com/jyp/dante][dante]]
1505 #+begin_src emacs-lisp
1508 :commands dante-mode
1509 :hook (haskell-mode . dante-mode))
1512 *** [[https://github.com/mpickering/hlint-refactor-mode][hlint-refactor]]
1514 Emacs bindings for [[https://github.com/ndmitchell/hlint][hlint]]'s refactor option. This requires the refact
1515 executable from [[https://github.com/mpickering/apply-refact][apply-refact]].
1517 #+begin_src emacs-lisp
1518 (use-package hlint-refactor
1520 :bind (:map hlint-refactor-mode-map
1521 ("C-c l b" . hlint-refactor-refactor-buffer)
1522 ("C-c l r" . hlint-refactor-refactor-at-point))
1523 :hook (haskell-mode . hlint-refactor-mode))
1526 *** [[https://github.com/flycheck/flycheck-haskell][flycheck-haskell]]
1528 #+begin_src emacs-lisp
1529 (use-package flycheck-haskell
1530 :after haskell-mode)
1533 *** [[https://github.com/ndmitchell/hlint/blob/20e116a043f2073c57b17b24ae6364b5e433ba7e/data/hs-lint.el][hs-lint.el]]
1535 :header-args+: :tangle lisp/hs-lint.el :mkdirp yes
1538 Currently using =flycheck-haskell= with the =haskell-hlint= checker
1541 #+begin_src emacs-lisp :tangle no
1542 ;;; hs-lint.el --- minor mode for HLint code checking
1544 ;; Copyright 2009 (C) Alex Ott
1546 ;; Author: Alex Ott <alexott@gmail.com>
1547 ;; Keywords: haskell, lint, HLint
1549 ;; Status: distributed under terms of GPL2 or above
1551 ;; Typical message from HLint looks like:
1553 ;; /Users/ott/projects/lang-exp/haskell/test.hs:52:1: Eta reduce
1555 ;; count1 p l = length (filter p l)
1557 ;; count1 p = length . filter p
1562 (defgroup hs-lint nil
1563 "Run HLint as inferior of Emacs, parse error messages."
1567 (defcustom hs-lint-command "hlint"
1568 "The default hs-lint command for \\[hlint]."
1572 (defcustom hs-lint-save-files t
1573 "Save modified files when run HLint or no (ask user)"
1577 (defcustom hs-lint-replace-with-suggestions nil
1578 "Replace user's code with suggested replacements"
1582 (defcustom hs-lint-replace-without-ask nil
1583 "Replace user's code with suggested replacements automatically"
1587 (defun hs-lint-process-setup ()
1588 "Setup compilation variables and buffer for `hlint'."
1589 (run-hooks 'hs-lint-setup-hook))
1591 ;; regex for replace suggestions
1593 ;; ^\(.*?\):\([0-9]+\):\([0-9]+\): .*
1599 (defvar hs-lint-regex
1600 "^\\(.*?\\):\\([0-9]+\\):\\([0-9]+\\): .*[\n\C-m]Found:[\n\C-m]\\s +\\(.*\\)[\n\C-m]Why not:[\n\C-m]\\s +\\(.*\\)[\n\C-m]"
1601 "Regex for HLint messages")
1603 (defun make-short-string (str maxlen)
1604 (if (< (length str) maxlen)
1606 (concat (substring str 0 (- maxlen 3)) "...")))
1608 (defun hs-lint-replace-suggestions ()
1609 "Perform actual replacement of suggestions"
1610 (goto-char (point-min))
1611 (while (re-search-forward hs-lint-regex nil t)
1612 (let* ((fname (match-string 1))
1613 (fline (string-to-number (match-string 2)))
1614 (old-code (match-string 4))
1615 (new-code (match-string 5))
1616 (msg (concat "Replace '" (make-short-string old-code 30)
1617 "' with '" (make-short-string new-code 30) "'"))
1623 (switch-to-buffer (get-file-buffer fname))
1624 (goto-char (point-min))
1625 (forward-line (1- fline))
1627 (setf bline (point))
1628 (when (or hs-lint-replace-without-ask
1631 (setf eline (point))
1633 (setf old-code (regexp-quote old-code))
1634 (while (string-match "\\\\ " old-code spos)
1635 (setf new-old-code (concat new-old-code
1636 (substring old-code spos (match-beginning 0))
1638 (setf spos (match-end 0)))
1639 (setf new-old-code (concat new-old-code (substring old-code spos)))
1640 (remove-text-properties bline eline '(composition nil))
1641 (when (re-search-forward new-old-code eline t)
1642 (replace-match new-code nil t)))))))
1644 (defun hs-lint-finish-hook (buf msg)
1645 "Function, that is executed at the end of HLint execution"
1646 (if hs-lint-replace-with-suggestions
1647 (hs-lint-replace-suggestions)
1650 (define-compilation-mode hs-lint-mode "HLint"
1651 "Mode for check Haskell source code."
1652 (set (make-local-variable 'compilation-process-setup-function)
1653 'hs-lint-process-setup)
1654 (set (make-local-variable 'compilation-disable-input) t)
1655 (set (make-local-variable 'compilation-scroll-output) nil)
1656 (set (make-local-variable 'compilation-finish-functions)
1657 (list 'hs-lint-finish-hook))
1661 "Run HLint for current buffer with haskell source"
1663 (save-some-buffers hs-lint-save-files)
1664 (compilation-start (concat hs-lint-command " \"" buffer-file-name "\"")
1668 ;;; hs-lint.el ends here
1671 #+begin_src emacs-lisp :tangle no
1672 (use-package hs-lint
1674 :bind (:map haskell-mode-map
1675 ("C-c l l" . hs-lint)))
1682 #+begin_src emacs-lisp
1683 (use-package sgml-mode
1685 (setq sgml-basic-offset 2))
1690 #+begin_src emacs-lisp
1691 (use-package css-mode
1693 (setq css-indent-offset 2))
1698 #+begin_src emacs-lisp
1699 (use-package web-mode
1703 web-mode-code-indent-offset
1704 web-mode-css-indent-offset
1705 web-mode-markup-indent-offset))
1710 #+begin_src emacs-lisp
1711 (use-package emmet-mode
1712 :after (:any web-mode css-mode sgml-mode)
1713 :bind* (("C-)" . emmet-next-edit-point)
1714 ("C-(" . emmet-prev-edit-point))
1716 (unbind-key "C-j" emmet-mode-keymap)
1717 (setq emmet-move-cursor-between-quotes t)
1718 :hook (web-mode css-mode html-mode sgml-mode))
1723 *** COMMENT meghanada
1725 #+begin_src emacs-lisp
1726 (use-package meghanada
1728 (:map meghanada-mode-map
1729 (("C-M-o" . meghanada-optimize-import)
1730 ("C-M-t" . meghanada-import-all)))
1731 :hook (java-mode . meghanada-mode))
1734 *** COMMENT lsp-java
1752 #+begin_src emacs-lisp
1753 (use-package treemacs
1754 :config (setq treemacs-never-persist t))
1756 (use-package yasnippet
1758 ;; (yas-global-mode)
1761 (use-package lsp-mode
1762 :init (setq lsp-eldoc-render-all nil
1763 lsp-highlight-symbol-at-point nil)
1768 (use-package company-lsp
1771 (setq company-lsp-cache-candidates t
1772 company-lsp-async t))
1776 (setq lsp-ui-sideline-update-mode 'point))
1778 (use-package lsp-java
1780 (add-hook 'java-mode-hook
1782 (setq-local company-backends (list 'company-lsp))))
1784 (add-hook 'java-mode-hook 'lsp-java-enable)
1785 (add-hook 'java-mode-hook 'flycheck-mode)
1786 (add-hook 'java-mode-hook 'company-mode)
1787 (add-hook 'java-mode-hook 'lsp-ui-mode))
1789 (use-package dap-mode
1795 (use-package dap-java
1798 (use-package lsp-java-treemacs
1804 #+begin_src emacs-lisp
1806 :bind (:map eclim-mode-map ("S-SPC" . company-complete))
1807 :hook ((java-mode . eclim-mode)
1808 (eclim-mode . (lambda ()
1809 (make-local-variable 'company-idle-delay)
1810 (defvar company-idle-delay)
1811 ;; (setq company-idle-delay 0.7)
1812 (setq company-idle-delay nil))))
1814 (eclim-auto-save nil)
1815 ;; (eclimd-default-workspace "~/src/eclipse-workspace-exp")
1816 (eclim-executable "~/.p2/pool/plugins/org.eclim_2.8.0/bin/eclim")
1817 (eclim-eclipse-dirs '("~/usr/eclipse/dsl-2018-09/eclipse")))
1822 #+begin_src emacs-lisp
1823 (use-package geiser)
1825 (use-feature geiser-guile
1827 (setq geiser-guile-load-path "~/src/git/guix"))
1832 #+begin_src emacs-lisp
1836 * Emacs enhancements
1838 :CUSTOM_ID: emacs-enhancements
1843 #+begin_src emacs-lisp
1845 :config (setq Man-width 80))
1848 ** [[https://github.com/justbur/emacs-which-key][which-key]]
1851 Emacs package that displays available keybindings in popup
1854 #+begin_src emacs-lisp
1855 (use-package which-key
1858 (which-key-add-key-based-replacements
1859 ;; prefixes for global prefixes and minor modes
1863 "C-c 8 -" "typo/dashes"
1864 "C-c 8 <" "typo/left-brackets"
1865 "C-c 8 >" "typo/right-brackets"
1867 "C-x a" "abbrev/expand"
1868 "C-x r" "rectangle/register/bookmark"
1869 "C-x v" "version control"
1870 ;; prefixes for my personal bindings
1871 "C-c a" "applications"
1874 "C-c p" "package-management"
1875 ;; "C-c p e" "package-management/epkg"
1876 "C-c p s" "straight.el"
1879 "C-c c" "compile-and-comments"
1884 "C-c m" "multiple-cursors"
1885 "C-c P" "projectile"
1886 "C-c P s" "projectile/search"
1887 "C-c P x" "projectile/execute"
1888 "C-c P 4" "projectile/other-window"
1894 ;; prefixes for major modes
1895 (which-key-add-major-mode-key-based-replacements 'message-mode
1897 (which-key-add-major-mode-key-based-replacements 'org-mode
1898 "C-c C-v" "org-babel")
1899 (which-key-add-major-mode-key-based-replacements 'web-mode
1900 "C-c C-a" "web/attributes"
1901 "C-c C-b" "web/blocks"
1903 "C-c C-e" "web/element"
1904 "C-c C-t" "web/tags")
1908 (which-key-add-column-padding 5)
1909 (which-key-max-description-length 32))
1914 #+begin_src emacs-lisp
1915 (add-to-list 'custom-theme-load-path "~/.emacs.d/lisp")
1916 (load-theme 'tangomod t)
1921 #+begin_src emacs-lisp
1922 (use-package smart-mode-line
1923 :commands (sml/apply-theme)
1931 #+begin_src emacs-lisp
1932 (use-package doom-themes)
1935 ** theme helper functions
1937 #+begin_src emacs-lisp
1938 (defvar a/org-mode-font-lock-keywords
1939 '(("[ \t]*\\(#\\+\\(BEGIN\\|END\\|begin\\|end\\)_\\(\\S-+\\)\\)[ \t]*\\([^\n:]*\\)"
1940 (1 '(:foreground "#5a5b5a" :background "#292b2b") t) ; directive
1941 (3 '(:foreground "#81a2be" :background "#292b2b") t) ; kind
1942 (4 '(:foreground "#c5c8c6") t)))) ; title
1944 (defun a/lights-on ()
1945 "Enable my favourite light theme."
1947 (mapc #'disable-theme custom-enabled-themes)
1948 (load-theme 'tangomod t)
1949 (sml/apply-theme 'automatic)
1950 (font-lock-remove-keywords
1951 'org-mode a/org-mode-font-lock-keywords))
1953 (defun a/lights-off ()
1956 (mapc #'disable-theme custom-enabled-themes)
1957 (load-theme 'doom-tomorrow-night t)
1958 (sml/apply-theme 'automatic)
1959 (font-lock-add-keywords
1960 'org-mode a/org-mode-font-lock-keywords t))
1963 ("s-t d" . a/lights-off)
1964 ("s-t l" . a/lights-on))
1967 ** [[https://github.com/bbatsov/crux][crux]]
1969 #+begin_src emacs-lisp
1970 (use-package crux ; results in Waiting for git... [2 times]
1972 :bind (("C-c b k" . crux-kill-other-buffers)
1973 ("C-c d" . crux-duplicate-current-line-or-region)
1974 ("C-c D" . crux-duplicate-and-comment-current-line-or-region)
1975 ("C-c f c" . crux-copy-file-preserve-attributes)
1976 ("C-c f d" . crux-delete-file-and-buffer)
1977 ("C-c f r" . crux-rename-file-and-buffer)
1978 ("C-c j" . crux-top-join-line)
1979 ("C-S-j" . crux-top-join-line)))
1982 ** [[https://github.com/alezost/mwim.el][mwim]]
1984 #+begin_src emacs-lisp
1986 :bind (("C-a" . mwim-beginning-of-code-or-line)
1987 ("C-e" . mwim-end-of-code-or-line)
1988 ("<home>" . mwim-beginning-of-line-or-code)
1989 ("<end>" . mwim-end-of-line-or-code)))
1994 #+begin_src emacs-lisp
1995 (use-package projectile
1996 :bind-keymap ("C-c P" . projectile-command-map)
2000 (defun my-projectile-invalidate-cache (&rest _args)
2001 ;; ignore the args to `magit-checkout'
2002 (projectile-invalidate-cache nil))
2004 (eval-after-load 'magit-branch
2006 (advice-add 'magit-checkout
2007 :after #'my-projectile-invalidate-cache)
2008 (advice-add 'magit-branch-and-checkout
2009 :after #'my-projectile-invalidate-cache)))
2010 :custom (projectile-completion-system 'ivy))
2013 ** [[https://github.com/Wilfred/helpful][helpful]]
2015 #+begin_src emacs-lisp
2016 (use-package helpful
2019 (("C-S-h c" . helpful-command)
2020 ("C-S-h f" . helpful-callable) ; helpful-function
2021 ("C-S-h v" . helpful-variable)
2022 ("C-S-h k" . helpful-key)
2023 ("C-S-h p" . helpful-at-point)))
2026 ** [[https://github.com/EricCrosson/unkillable-scratch][unkillable-scratch]]
2028 Make =*scratch*= and =*Messages*= unkillable.
2030 #+begin_src emacs-lisp
2031 (use-package unkillable-scratch
2034 (unkillable-scratch 1)
2036 (unkillable-buffers '("^\\*scratch\\*$" "^\\*Messages\\*$")))
2039 ** [[https://github.com/davep/boxquote.el][boxquote.el]]
2043 | make pretty boxed quotes like this
2047 #+begin_src emacs-lisp
2048 (use-package boxquote
2051 (:prefix-map a/boxquote-prefix-map
2053 ("b" . boxquote-buffer)
2054 ("B" . boxquote-insert-buffer)
2055 ("d" . boxquote-defun)
2056 ("F" . boxquote-insert-file)
2057 ("hf" . boxquote-describe-function)
2058 ("hk" . boxquote-describe-key)
2059 ("hv" . boxquote-describe-variable)
2060 ("hw" . boxquote-where-is)
2061 ("k" . boxquote-kill)
2062 ("p" . boxquote-paragraph)
2063 ("q" . boxquote-boxquote)
2064 ("r" . boxquote-region)
2065 ("s" . boxquote-shell-command)
2066 ("t" . boxquote-text)
2067 ("T" . boxquote-title)
2068 ("u" . boxquote-unbox)
2069 ("U" . boxquote-unbox-region)
2070 ("y" . boxquote-yank)
2071 ("M-q" . boxquote-fill-paragraph)
2072 ("M-w" . boxquote-kill-ring-save)))
2075 Also see [[https://www.emacswiki.org/emacs/rebox2][rebox2]].
2079 #+begin_src emacs-lisp
2080 (use-package orgalist
2082 :hook (message-mode . orgalist-mode))
2087 #+begin_src emacs-lisp
2091 (typo-global-mode 1)
2092 :hook (text-mode . typo-mode))
2097 #+begin_src emacs-lisp
2098 (use-package hl-todo
2101 (global-hl-todo-mode))
2106 #+begin_src emacs-lisp
2107 (use-package shrink-path
2111 (defun +eshell/prompt ()
2112 (let ((base/dir (shrink-path-prompt default-directory)))
2113 (concat (propertize (car base/dir)
2114 'face 'font-lock-comment-face)
2115 (propertize (cdr base/dir)
2116 'face 'font-lock-constant-face)
2117 (propertize (+eshell--current-git-branch)
2118 'face 'font-lock-function-name-face)
2121 ;; (propertize "λ" 'face 'eshell-prompt)
2122 ;; needed for the input text to not have prompt face
2123 (propertize "λ " 'face 'default))))
2125 (defun +eshell--current-git-branch ()
2126 (let ((branch (car (loop for match in (split-string (shell-command-to-string "git branch") "\n")
2127 when (string-match "^\*" match)
2129 (if (not (eq branch nil))
2130 (concat " " (substring branch 2))
2132 (setq eshell-prompt-regexp "\\(.*\n\\)*λ "
2133 eshell-prompt-function #'+eshell/prompt))
2136 ** [[https://github.com/peterwvj/eshell-up][eshell-up]]
2138 #+begin_src emacs-lisp
2139 (use-package eshell-up
2141 :commands eshell-up)
2146 #+begin_src emacs-lisp
2147 (use-package multi-term
2149 :bind (("C-c a s m" . multi-term-dedicated-toggle)
2151 ("C-c C-j" . term-char-mode)
2153 ("C-c C-j" . term-line-mode))
2155 (setq multi-term-program "/bin/screen"
2156 ;; TODO: add separate bindings for connecting to existing
2157 ;; session vs. always creating a new one
2158 multi-term-dedicated-select-after-open-p t
2159 multi-term-dedicated-window-height 20
2160 multi-term-dedicated-max-window-height 30
2162 '(("C-c C-c" . term-interrupt-subjob)
2163 ("C-c C-e" . term-send-esc)
2165 ("C-y" . term-paste)
2166 ("M-f" . term-send-forward-word)
2167 ("M-b" . term-send-backward-word)
2168 ("M-p" . term-send-up)
2169 ("M-n" . term-send-down)
2170 ("<C-backspace>" . term-send-backward-kill-word)
2171 ("<M-DEL>" . term-send-backward-kill-word)
2172 ("M-d" . term-send-delete-word)
2173 ("M-," . term-send-raw)
2174 ("M-." . comint-dynamic-complete))
2175 term-unbind-key-alist
2176 '("C-z" "C-x" "C-c" "C-h" "C-y" "<ESC>")))
2181 #+begin_src emacs-lisp
2182 (use-package page-break-lines
2184 (global-page-break-lines-mode))
2189 #+begin_src emacs-lisp
2190 (use-package expand-region
2191 :bind ("C-=" . er/expand-region))
2196 #+begin_src emacs-lisp
2197 (use-package multiple-cursors
2199 (("C-S-<mouse-1>" . mc/add-cursor-on-click)
2200 (:prefix-map a/mc-prefix-map
2202 ("c" . mc/edit-lines)
2203 ("n" . mc/mark-next-like-this)
2204 ("p" . mc/mark-previous-like-this)
2205 ("a" . mc/mark-all-like-this))))
2210 #+begin_src emacs-lisp
2218 #+begin_src emacs-lisp
2219 (use-package yasnippet
2222 (defconst yas-verbosity-cur yas-verbosity)
2223 (setq yas-verbosity 2)
2224 (add-to-list 'yas-snippet-dirs "~/src/git/guix/etc/snippets")
2226 (setq yas-verbosity yas-verbosity-cur)
2228 (text-mode . yas-minor-mode))
2236 #+begin_src emacs-lisp
2237 (defvar a/maildir (expand-file-name "~/mail/"))
2238 (with-eval-after-load 'recentf
2239 (add-to-list 'recentf-exclude a/maildir))
2244 #+begin_src emacs-lisp
2246 a/gnus-init-file (no-littering-expand-etc-file-name "gnus")
2247 mail-user-agent 'gnus-user-agent
2248 read-mail-command 'gnus)
2251 :bind (("s-m" . gnus)
2252 ("s-M" . gnus-unplugged))
2255 gnus-select-method '(nnnil "")
2256 gnus-secondary-select-methods
2258 (nnimap-stream plain)
2259 (nnimap-address "127.0.0.1")
2260 (nnimap-server-port 143)
2261 (nnimap-authenticator plain)
2262 (nnimap-user "amin@bndl.org"))
2264 (nnimap-stream plain)
2265 (nnimap-address "127.0.0.1")
2266 (nnimap-server-port 143)
2267 (nnimap-authenticator plain)
2268 (nnimap-user "abandali@uwaterloo.ca"))
2270 (nnimap-stream plain)
2271 (nnimap-address "127.0.0.1")
2272 (nnimap-server-port 143)
2273 (nnimap-authenticator plain)
2274 (nnimap-user "abandali@csclub.uw")))
2275 gnus-message-archive-group "nnimap+amin:Sent"
2279 gnus-large-newsgroup 50
2280 gnus-home-directory (no-littering-expand-var-file-name "gnus/")
2281 gnus-directory (concat gnus-home-directory "news/")
2282 message-directory (concat gnus-home-directory "mail/")
2283 nndraft-directory (concat gnus-home-directory "drafts/")
2284 gnus-save-newsrc-file nil
2285 gnus-read-newsrc-file nil
2286 gnus-interactive-exit nil
2287 gnus-gcc-mark-as-read t)
2291 (require 'ebdb-gnus))
2293 (use-feature gnus-art
2296 gnus-visible-headers
2297 (concat gnus-visible-headers "\\|^List-Id:\\|^X-RT-Originator:\\|^User-Agent:")
2298 gnus-sorted-header-list
2299 '("^From:" "^Subject:" "^Summary:" "^Keywords:"
2300 "^Followup-To:" "^To:" "^Cc:" "X-RT-Originator"
2301 "^Newsgroups:" "List-Id:" "^Organization:"
2302 "^User-Agent:" "^Date:")
2303 ;; local-lapsed article dates
2304 ;; from https://www.emacswiki.org/emacs/GnusFormatting#toc11
2305 gnus-article-date-headers '(user-defined)
2306 gnus-article-time-format
2308 (let* ((date (format-time-string "%a, %d %b %Y %T %z" time))
2309 (local (article-make-date-line date 'local))
2310 (combined-lapsed (article-make-date-line date
2313 (string-match " (.+" combined-lapsed)
2314 (match-string 0 combined-lapsed))))
2315 (concat local lapsed))))
2317 :map gnus-article-mode-map
2318 ("r" . gnus-article-reply-with-original)
2319 ("R" . gnus-article-wide-reply-with-original)
2320 ("M-L" . org-store-link)))
2322 (use-feature gnus-sum
2323 :bind (:map gnus-summary-mode-map
2324 :prefix-map a/gnus-summary-prefix-map
2326 ("r" . gnus-summary-reply)
2327 ("w" . gnus-summary-wide-reply)
2328 ("v" . gnus-summary-show-raw-article))
2331 :map gnus-summary-mode-map
2332 ("r" . gnus-summary-reply-with-original)
2333 ("R" . gnus-summary-wide-reply-with-original)
2334 ("M-L" . org-store-link))
2335 :hook (gnus-summary-mode . a/no-mouse-autoselect-window))
2337 (use-feature gnus-msg
2339 (setq gnus-posting-styles
2341 (address "amin@bndl.org")
2342 (body "\nBest,\namin\n")
2343 (eval (setq a/message-cite-say-hi t)))
2345 (address "bandali@gnu.org")
2346 (eval (set (make-local-variable 'message-user-fqdn) "fencepost.gnu.org")))
2347 ((header "subject" "ThankCRM")
2348 (to "webmasters-comment@gnu.org")
2349 (body "\nAdded to 2019supporters.html.\n\nMoving to campaigns.\n\n-amin\n")
2350 (eval (setq a/message-cite-say-hi nil)))
2351 ("nnimap\\+uwaterloo:.*"
2352 (address "abandali@uwaterloo.ca")
2353 (gcc "\"nnimap+uwaterloo:Sent Items\""))
2354 ("nnimap\\+csclub:.*"
2355 (address "abandali@csclub.uwaterloo.ca")
2356 (gcc "nnimap+csclub:Sent")))))
2358 (use-feature gnus-topic
2359 :hook (gnus-group-mode . gnus-topic-mode)
2360 :config (setq gnus-topic-line-format "%i[ %A: %(%{%n%}%) ]%v\n"))
2362 (use-feature gnus-agent
2364 (setq gnus-agent-synchronize-flags 'ask)
2365 :hook (gnus-group-mode . gnus-agent-mode))
2367 (use-feature gnus-group
2369 (setq gnus-permanently-visible-groups "\\(:INBOX$\\|:gnu$\\)"))
2371 (use-feature mm-decode
2373 (setq mm-discouraged-alternatives '("text/html" "text/richtext")))
2378 #+begin_src emacs-lisp
2379 (use-feature sendmail
2381 (setq sendmail-program "/usr/bin/msmtp"
2382 ;; message-sendmail-extra-arguments '("-v" "-d")
2383 mail-specify-envelope-from t
2384 mail-envelope-from 'header))
2389 #+begin_src emacs-lisp
2390 (use-feature message
2392 ;; redefine for a simplified In-Reply-To header
2393 ;; (see https://todo.sr.ht/~sircmpwn/lists.sr.ht/67)
2394 (defun message-make-in-reply-to ()
2395 "Return the In-Reply-To header for this message."
2396 (when message-reply-headers
2397 (let ((from (mail-header-from message-reply-headers))
2398 (msg-id (mail-header-id message-reply-headers)))
2402 (defconst a/message-cite-style-format "On %Y-%m-%d %l:%M %p, %N wrote:")
2403 (defconst message-cite-style-bandali
2404 '((message-cite-function 'message-cite-original)
2405 (message-citation-line-function 'message-insert-formatted-citation-line)
2406 (message-cite-reply-position 'traditional)
2407 (message-yank-prefix "> ")
2408 (message-yank-cited-prefix ">")
2409 (message-yank-empty-prefix ">")
2410 (message-citation-line-format
2411 (if a/message-cite-say-hi
2412 (concat "Hi %F,\n\n" a/message-cite-style-format)
2413 a/message-cite-style-format)))
2414 "Citation style based on Mozilla Thunderbird's. Use with message-cite-style.")
2415 (setq message-cite-style 'message-cite-style-bandali
2416 message-kill-buffer-on-exit t
2417 message-send-mail-function 'message-send-mail-with-sendmail
2418 message-sendmail-envelope-from 'header
2419 message-dont-reply-to-names
2420 "\\(\\(amin@bndl\\.org\\)\\|\\(.*@\\(aminb\\|amin\\.bndl\\)\\.org\\)\\|\\(\\(bandali\\|aminb?\\|mab\\)@gnu\\.org\\)\\|\\(a\\(min\\.\\)?bandali@uwaterloo\\.ca\\)\\|\\(abandali@csclub\\.uwaterloo\\.ca\\)\\)")
2421 (require 'company-ebdb)
2422 :hook (;; (message-setup . mml-secure-message-sign-pgpmime)
2423 (message-mode . flyspell-mode)
2424 (message-mode . (lambda ()
2425 ;; (setq fill-column 65
2426 ;; message-fill-column 65)
2427 (make-local-variable 'company-idle-delay)
2428 (setq company-idle-delay 0.2))))
2430 ;; (message-header-subject ((t (:foreground "#111" :weight semi-bold))))
2431 ;; (message-header-to ((t (:foreground "#111" :weight normal))))
2432 ;; (message-header-cc ((t (:foreground "#333" :weight normal))))
2435 (with-eval-after-load 'mml-sec
2436 (setq mml-secure-openpgp-encrypt-to-self t
2437 mml-secure-openpgp-sign-with-sender t))
2442 Convenient footnotes in =message-mode=.
2444 #+begin_src emacs-lisp
2445 (use-feature footnote
2448 ;; (setq footnote-start-tag ""
2449 ;; footnote-end-tag ""
2450 ;; footnote-style 'unicode)
2452 (:map message-mode-map
2453 :prefix-map a/footnote-prefix-map
2455 ("a" . footnote-add-footnote)
2456 ("b" . footnote-back-to-message)
2457 ("c" . footnote-cycle-style)
2458 ("d" . footnote-delete-footnote)
2459 ("g" . footnote-goto-footnote)
2460 ("r" . footnote-renumber-footnotes)
2461 ("s" . footnote-set-style)))
2466 #+begin_src emacs-lisp
2468 :straight (:host github :repo "girzel/ebdb")
2470 :bind (:map gnus-group-mode-map ("e" . ebdb))
2472 (setq ebdb-sources (no-littering-expand-var-file-name "ebdb"))
2473 (with-eval-after-load 'swiper
2474 (add-to-list 'swiper-font-lock-exclude 'ebdb-mode t)))
2476 (use-feature ebdb-com
2479 ;; (use-package ebdb-complete
2482 ;; (ebdb-complete-enable))
2484 (use-package company-ebdb
2486 (defun company-ebdb--post-complete (_) nil))
2488 (use-feature ebdb-gnus
2491 (ebdb-gnus-window-configuration
2494 (summary 0.25 point)
2497 (ebdb-gnus 0.3))))))
2499 (use-feature ebdb-mua
2501 ;; :custom (ebdb-mua-pop-up nil)
2504 ;; (use-package ebdb-message
2508 ;; (use-package ebdb-vcard
2514 #+begin_src emacs-lisp
2515 (use-package message-x)
2518 #+begin_src emacs-lisp :tangle no
2519 (use-package message-x
2521 (message-x-completion-alist
2523 (("\\([rR]esent-\\|[rR]eply-\\)?[tT]o:\\|[bB]?[cC][cC]:" . gnus-harvest-find-address)
2526 (quote message-newgroups-header-regexp))
2527 message-newgroups-header-regexp message-newsgroups-header-regexp)
2528 . message-expand-group)))))
2531 ** COMMENT gnus-harvest
2533 #+begin_src emacs-lisp
2534 (use-package gnus-harvest
2535 :commands gnus-harvest-install
2538 (if (featurep 'message-x)
2539 (gnus-harvest-install 'message-x)
2540 (gnus-harvest-install)))
2550 #+begin_src emacs-lisp
2552 :straight (:host nil :repo "https://git.bndl.org/amin/znc.el")
2553 :bind (("C-c a e e" . znc-erc)
2554 ("C-c a e a" . znc-all))
2556 (let ((pwd (let ((auth (auth-source-search :host "znca")))
2558 ((null auth) (error "Couldn't find znca's authinfo"))
2559 (t (funcall (plist-get (car auth) :secret)))))))
2561 `(("znc.bndl.org" 1337 t
2562 ((freenode "amin/freenode" ,pwd)))
2563 ("znc.bndl.org" 1337 t
2564 ((moznet "amin/moznet" ,pwd)))))))
2567 * Post initialization
2569 :CUSTOM_ID: post-initialization
2572 Display how long it took to load the init file.
2574 #+begin_src emacs-lisp
2575 (message "Loading %s...done (%.3fs)" user-init-file
2576 (float-time (time-subtract (current-time)
2577 a/before-user-init-time)))
2585 #+begin_src emacs-lisp :comments none
2586 ;;; init.el ends here
2589 * COMMENT Local Variables :ARCHIVE:
2591 # eval: (add-hook 'after-save-hook #'a/async-babel-tangle 'append 'local)
2592 # eval: (when (featurep 'typo (typo-mode -1)))