emacs: briefly try out helm and auctex
[~bandali/configs] / .emacs.d / init.org
... / ...
CommitLineData
1#+title: Literate Emacs Configuration of Amin Bandali
2#+author: Amin Bandali
3#+babel: :cache yes
4#+property: header-args :tangle yes
5
6* About
7:PROPERTIES:
8:CUSTOM_ID: about
9:END:
10
11This org file is my literate configuration for GNU Emacs, and is
12tangled 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
14configurations of many different people. Some of the configurations
15that I can remember off the top of my head are:
16
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
20 packages
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]]
27
28I'd like to have a fully reproducible Emacs setup (part of the reason
29why I store my configuration in this repository) but unfortunately out
30of the box, that's not achievable with =package.el=, not currently
31anyway. So, I've opted to use =straight.el=. I also used Borg for a
32few months, but decided to try =straight.el= which allows direct use
33of the various package archives.
34
35** Installation
36:PROPERTIES:
37:CUSTOM_ID: installation
38:END:
39
40To use this config for your Emacs, first you need to clone this repo,
41then tangle =init.org= into =init.el=, and optionally byte-compile
42=init.el=.
43
44First, clone the repository and =cd= into it:
45
46#+begin_src sh :tangle no
47git clone https://git.sr.ht/~bandali/dotfiles ~/.emacs.d
48cd ~/.emacs.d
49#+end_src
50
51Then, decide if you would like to use a byte-compiled init file, and
52set the [[#byte-compiled-init][a/byte-compiled-init]] variable accordingly.
53
54Now, first tangle =init.org=, and only if you chose to have a
55byte-compiled init, build init as well:
56
57#+begin_src sh :tangle no
58make tangle-init
59make build-init
60#+end_src
61
62If you'd like to use a byte-compiled init, it's important that it be
63recompiled whenever =init.el= is generated from an updated =init.org=.
64Not only does my setup automatically and asynchronously tangle
65=init.org= to =init.el= every time you edit and save =init.org= in GNU
66Emacs, 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
68about manually tangling and compiling your init file whenever you
69change it. The output of the last byte-compilation in the current
70session is kept in a =*compilation*= buffer, which will automatically
71be displayed if compilation fails.
72
73* Contents :toc_1:noexport:
74
75- [[#about][About]]
76- [[#header][Header]]
77- [[#initial-setup][Initial setup]]
78- [[#core][Core]]
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]]
84- [[#email][Email]]
85- [[#blogging][Blogging]]
86- [[#post-initialization][Post initialization]]
87- [[#footer][Footer]]
88
89* Header
90:PROPERTIES:
91:CUSTOM_ID: header
92:END:
93
94** First line
95
96#+begin_src emacs-lisp :comments none
97;;; init.el --- Amin Bandali's Emacs config -*- lexical-binding: t; eval: (view-mode 1) -*-
98#+end_src
99
100Enable =view-mode=, which both makes the file read-only (as a reminder
101that =init.el= is an auto-generated file, not supposed to be edited),
102and provides some convenient key bindings for browsing through the
103file.
104
105** License
106
107#+begin_src emacs-lisp :comments none
108;; Copyright (C) 2018-2019 Amin Bandali <bandali@gnu.org>
109
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.
114
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.
119
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/>.
122#+end_src
123
124** Commentary
125
126#+begin_src emacs-lisp :comments none
127;;; Commentary:
128
129;; Emacs configuration of Amin Bandali, computer scientist, functional
130;; programmer, and free software advocate.
131
132;; THIS FILE IS AUTO-GENERATED FROM `init.org'.
133#+end_src
134
135* Initial setup
136:PROPERTIES:
137:CUSTOM_ID: initial-setup
138:END:
139
140** Byte-compiled init preference
141:PROPERTIES:
142:CUSTOM_ID: byte-compiled-init
143:END:
144
145If you would like a byte-compiled init file, set the following
146variable to ~t~, otherwise set it to ~nil~.
147
148#+begin_src emacs-lisp
149(defvar a/byte-compiled-init nil
150 "If non-nil, byte-(re)compile init.el on successful tangles.")
151#+end_src
152
153You can click on [[#installation][Installation]] to jump back up there if you like :)
154
155** Emacs initialization
156
157I'd like to do a couple of measurements of Emacs' startup time. First,
158let's see how long Emacs takes to start up, before even loading
159=init.el=, i.e. =user-init-file=:
160
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
166 before-init-time)))
167#+end_src
168
169Also, temporarily increase ~gc-cons-threshhold~ and
170~gc-cons-percentage~ during startup to reduce garbage collection
171frequency. Clearing the ~file-name-handler-alist~ seems to help reduce
172startup time as well.
173
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)
183#+end_src
184
185Of course, we'd like to set them back to their defaults once we're
186done initializing.
187
188#+begin_src emacs-lisp
189(add-hook
190 'after-init-hook
191 (lambda ()
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)))
195#+end_src
196
197Increase the number of lines kept in message logs (the =*Messages*=
198buffer).
199
200#+begin_src emacs-lisp
201(setq message-log-max 20000)
202#+end_src
203
204Optionally, we could suppress some byte compiler warnings like below,
205but for now I've decided to keep them enabled. See documentation for
206~byte-compile-warnings~ for more details.
207
208#+begin_src emacs-lisp
209;; (setq byte-compile-warnings
210;; '(not free-vars unresolved noruntime lexical make-local))
211#+end_src
212
213** whoami
214
215#+begin_src emacs-lisp
216(setq user-full-name "Amin Bandali"
217 user-mail-address "amin@bndl.org")
218#+end_src
219
220** Package management
221
222*** No =package.el=
223
224I can do all my package management things with =straight.el=, and
225don'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]]).
227
228#+begin_src emacs-lisp :tangle early-init.el
229(setq package-enable-at-startup nil)
230#+end_src
231
232But since Emacs 27 isn't out yet (Emacs 26 is just around the corner
233right now), and even when released it'll be long before most distros
234ship in their repos, I'll still put the old workaround with the
235commented call to ~package-initialize~ here anyway.
236
237#+begin_src emacs-lisp :tangle no
238(setq package-enable-at-startup nil)
239;; (package-initialize)
240#+end_src
241
242Update: the above is not necessary, since =straight.el= automatically
243does that (and more). See =straight-package-neutering-mode=.
244
245*** =straight.el=
246
247#+begin_quote
248Next-generation, purely functional package manager for the Emacs
249hacker.
250#+end_quote
251
252=straight.el= allows me to have a fully reproducible Emacs setup.
253
254#+begin_src emacs-lisp
255;; Main engine start...
256
257(setq straight-repository-branch "develop"
258 straight-check-for-modifications '(check-on-save find-when-checking))
259
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)
266 (with-current-buffer
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)))
273
274;; Solid rocket booster ignition...
275
276(defun a/build-init ()
277 (a/bootstrap-straight)
278 (byte-compile-file "init.el"))
279
280(a/bootstrap-straight)
281
282;; We have lift off!
283
284(setq straight-use-package-by-default t)
285#+end_src
286
287Since we enable =straight.el='s =straight-use-package-by-default=
288integration, we will define a =use-feature= for plain ole
289=use-package= without any of the =straight.el= stuff.
290
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))
295 `(use-package ,name
296 :straight nil
297 ,@args))
298#+end_src
299
300Also, here's a useful function for reloading the init file (useful
301after running =straight-check-all=:
302
303#+begin_src emacs-lisp
304(defun a/reload-init ()
305 "Reload init.el."
306 (interactive)
307 (straight-transaction
308 (straight-mark-transaction-as-init)
309 (load (if a/byte-compiled-init
310 (concat (file-name-sans-extension user-init-file) ".elc")
311 user-init-file))))
312#+end_src
313
314*** =use-package=
315
316#+begin_quote
317A use-package declaration for simplifying your .emacs
318#+end_quote
319
320[[https://github.com/jwiegley/use-package][use-package]] is an awesome utility for managing and configuring
321packages (in our case especially the latter) in a neatly organized way
322and without compromising on performance.
323
324#+begin_src emacs-lisp
325(straight-use-package 'use-package)
326(if nil ; set to t when need to debug init
327 (progn
328 (setq use-package-verbose t
329 use-package-expand-minimally nil
330 use-package-compute-statistics t
331 debug-on-error t)
332 (require 'use-package))
333 (setq use-package-verbose nil
334 use-package-expand-minimally t))
335
336(setq use-package-always-defer t)
337(require 'bind-key)
338#+end_src
339
340*** COMMENT Epkg
341
342#+begin_quote
343Browse the Emacsmirror package database
344#+end_quote
345
346Epkg provides access to a local copy of the [[https://emacsmirror.net][Emacsmirror]] package
347database, low-level functions for querying the database, and a
348=package.el=-like user interface for browsing the available packages.
349
350#+begin_src emacs-lisp
351(use-package epkg
352 :commands (epkg-list-packages epkg-describe-package)
353 :bind
354 (("C-c p e d" . epkg-describe-package)
355 ("C-c p e p" . epkg-list-packages))
356 :config
357 (setq epkg-repository "~/.emacs.d/straight/repos/epkgs/")
358 (eval-when-compile (defvar ivy-initial-inputs-alist))
359 (with-eval-after-load 'ivy
360 (add-to-list
361 'ivy-initial-inputs-alist '(epkg-describe-package . "^") t)))
362#+end_src
363
364** No littering in =~/.emacs.d=
365
366#+begin_quote
367Help keeping ~/.emacs.d clean
368#+end_quote
369
370By default, even for Emacs' built-in packages, the configuration files
371and persistent data are all over the place. Use =no-littering= to help
372contain the mess.
373
374#+begin_src emacs-lisp
375(use-package no-littering
376 :demand t
377 :config
378 (savehist-mode 1)
379 (add-to-list 'savehist-additional-variables 'kill-ring)
380 (save-place-mode 1)
381 (setq auto-save-file-name-transforms
382 `((".*" ,(no-littering-expand-var-file-name "auto-save/") t))))
383#+end_src
384
385** Custom file (=custom.el=)
386
387I'm not planning on using the custom file much, but even so, I
388definitely don't want it mixing with =init.el=. So, here; let's give
389it it's own file. While at it, treat themes as safe.
390
391#+begin_src emacs-lisp
392(use-feature custom
393 :no-require t
394 :config
395 (setq custom-file (no-littering-expand-etc-file-name "custom.el"))
396 (when (file-exists-p custom-file)
397 (load custom-file))
398 (setf custom-safe-themes t))
399#+end_src
400
401** Secrets file
402
403Load the secrets file if it exists, otherwise show a warning.
404
405#+begin_src emacs-lisp
406(with-demoted-errors
407 (load (no-littering-expand-etc-file-name "secrets")))
408#+end_src
409
410** Better =$PATH= handling
411
412Let's use [[https://github.com/purcell/exec-path-from-shell][exec-path-from-shell]] to make Emacs use the =$PATH= as set up
413in my shell.
414
415#+begin_src emacs-lisp
416(use-package exec-path-from-shell
417 :defer 0.4
418 :init
419 (setq exec-path-from-shell-arguments nil
420 exec-path-from-shell-check-startup-files nil)
421 :config
422 (exec-path-from-shell-initialize)
423 ;; while we're at it, let's fix access to our running ssh-agent
424 (exec-path-from-shell-copy-env "SSH_AGENT_PID")
425 (exec-path-from-shell-copy-env "SSH_AUTH_SOCK"))
426#+end_src
427
428** COMMENT Only one custom theme at a time
429
430#+begin_src emacs-lisp
431(defadvice load-theme (before clear-previous-themes activate)
432 "Clear existing theme settings instead of layering them"
433 (mapc #'disable-theme custom-enabled-themes))
434#+end_src
435
436** Server
437
438Start server if not already running. Alternatively, can be done by
439issuing =emacs --daemon= in the terminal, which can be automated with
440a systemd service or using =brew services start emacs= on macOS. I use
441Emacs as my window manager (via EXWM), so I always start Emacs on
442login; so starting the server from inside Emacs is good enough for me.
443
444See [[https://www.gnu.org/software/emacs/manual/html_node/emacs/Emacs-Server.html#Emacs-Server][Using Emacs as a Server]].
445
446#+begin_src emacs-lisp
447(use-feature server
448 :defer 0.4
449 :config (or (server-running-p) (server-mode)))
450#+end_src
451
452** COMMENT Unicode support
453
454Font stack with better unicode support, around =Ubuntu Mono= and
455=Hack=.
456
457#+begin_src emacs-lisp
458(dolist (ft (fontset-list))
459 (set-fontset-font
460 ft
461 'unicode
462 (font-spec :name "Source Code Pro" :size 14))
463 (set-fontset-font
464 ft
465 'unicode
466 (font-spec :name "DejaVu Sans Mono")
467 nil
468 'append)
469 ;; (set-fontset-font
470 ;; ft
471 ;; 'unicode
472 ;; (font-spec
473 ;; :name "Symbola monospacified for DejaVu Sans Mono")
474 ;; nil
475 ;; 'append)
476 ;; (set-fontset-font
477 ;; ft
478 ;; #x2115 ; ℕ
479 ;; (font-spec :name "DejaVu Sans Mono")
480 ;; nil
481 ;; 'append)
482 (set-fontset-font
483 ft
484 (cons ?Α ?ω)
485 (font-spec :name "DejaVu Sans Mono" :size 14)
486 nil
487 'prepend))
488#+end_src
489
490** Gentler font resizing
491
492#+begin_src emacs-lisp
493(setq text-scale-mode-step 1.05)
494#+end_src
495
496** Focus follows mouse
497
498I’d like focus to follow the mouse when I move the cursor from one
499window to the next.
500
501#+begin_src emacs-lisp
502(setq mouse-autoselect-window t)
503#+end_src
504
505Let’s define a function to conveniently disable this for certain
506buffers and/or modes.
507
508#+begin_src emacs-lisp
509(defun a/no-mouse-autoselect-window ()
510 (make-local-variable 'mouse-autoselect-window)
511 (setq mouse-autoselect-window nil))
512#+end_src
513
514** Better scrolling (arguably)
515
516#+begin_src emacs-lisp
517(setq ;; scroll-margin 1
518 ;; scroll-conservatively 10000
519 scroll-step 1
520 scroll-conservatively 10
521 scroll-preserve-screen-position 1)
522
523(use-feature mwheel
524 :defer 0.4
525 :config
526 (setq mouse-wheel-scroll-amount '(1 ((shift) . 1)) ; one line at a time
527 mouse-wheel-progressive-speed nil ; don't accelerate scrolling
528 mouse-wheel-follow-mouse t)) ; scroll window under mouse
529
530(use-feature pixel-scroll
531 :defer 0.4
532 :config (pixel-scroll-mode 1))
533#+end_src
534
535** Ask for GPG passphrase in minibuffer
536
537#+begin_src emacs-lisp
538(setq epg-pinentry-mode 'loopback)
539#+end_src
540
541** Libraries
542
543#+begin_src emacs-lisp
544(require 'cl-lib)
545(require 'subr-x)
546#+end_src
547
548** Useful utilities
549
550Convenience macro for =setq='ing multiple variables to the same value:
551
552#+begin_src emacs-lisp
553(defmacro a/setq-every (value &rest vars)
554 "Set all the variables from VARS to value VALUE."
555 (declare (indent defun) (debug t))
556 `(progn ,@(mapcar (lambda (x) (list 'setq x value)) vars)))
557#+end_src
558
559The following process-related stuff from [[https://github.com/alezost/emacs-config][alezost's emacs-config]].
560
561#+begin_src emacs-lisp
562(defun a/start-process (program &rest args)
563 "Same as `start-process', but doesn't bother about name and buffer."
564 (let ((process-name (concat program "_process"))
565 (buffer-name (generate-new-buffer-name
566 (concat program "_output"))))
567 (apply #'start-process
568 process-name buffer-name program args)))
569
570(defun a/dired-start-process (program &optional args)
571 "Open current file with a PROGRAM."
572 ;; Shell command looks like this: "program [ARGS]... FILE" (ARGS can
573 ;; be nil, so remove it).
574 (apply #'a/start-process
575 program
576 (remove nil (list args (dired-get-file-for-visit)))))
577#+end_src
578
579* Core
580:PROPERTIES:
581:CUSTOM_ID: core
582:END:
583
584** Defaults
585
586*** Time and battery in mode-line
587
588Enable displaying time and battery in the mode-line, since I'm not
589using the Xfce panel anymore. Also, I don't need to see the load
590average on a regular basis, so disable that.
591
592Note: using =i3status= on sway at the moment, so disabling this.
593
594#+begin_src emacs-lisp :tangle no
595(use-package time
596 :init
597 (setq display-time-default-load-average nil)
598 :config
599 (display-time-mode))
600
601(use-package battery
602 :config
603 (display-battery-mode))
604#+end_src
605
606*** Smaller fringe
607
608Might want to set the fringe to a smaller value, especially if using
609EXWM. I'm fine with the default for now.
610
611#+begin_src emacs-lisp
612;; (fringe-mode '(3 . 1))
613(fringe-mode nil)
614#+end_src
615
616*** Disable disabled commands
617
618Emacs disables some commands by default that could persumably be
619confusing for novice users. Let's disable that.
620
621#+begin_src emacs-lisp
622(setq disabled-command-function nil)
623#+end_src
624
625*** Kill-ring
626
627Save what I copy into clipboard from other applications into Emacs'
628kill-ring, which would allow me to still be able to easily access it
629in case I kill (cut or copy) something else inside Emacs before
630yanking (pasting) what I'd originally intended to.
631
632#+begin_src emacs-lisp
633(setq save-interprogram-paste-before-kill t)
634#+end_src
635
636*** Minibuffer
637
638#+begin_src emacs-lisp
639(setq enable-recursive-minibuffers t
640 resize-mini-windows t)
641#+end_src
642
643*** Lazy-person-friendly yes/no prompts
644
645Lazy people would prefer to type fewer keystrokes, especially for yes
646or no questions. I'm lazy.
647
648#+begin_src emacs-lisp
649(defalias 'yes-or-no-p #'y-or-n-p)
650#+end_src
651
652*** Startup screen and =*scratch*=
653
654Firstly, let Emacs know that I'd like to have =*scratch*= as my
655startup buffer.
656
657#+begin_src emacs-lisp
658(setq initial-buffer-choice t)
659#+end_src
660
661Now let's customize the =*scratch*= buffer a bit. First off, I don't
662need the default hint.
663
664#+begin_src emacs-lisp
665(setq initial-scratch-message nil)
666#+end_src
667
668Also, let's use Text mode as the major mode, in case I want to
669customize it (=*scratch*='s default major mode, Fundamental mode,
670can't really be customized).
671
672#+begin_src emacs-lisp
673(setq initial-major-mode 'text-mode)
674#+end_src
675
676Inhibit the buffer list when more than 2 files are loaded.
677
678#+begin_src emacs-lisp
679(setq inhibit-startup-buffer-menu t)
680#+end_src
681
682I don't really need to see the startup screen or echo area message
683either.
684
685#+begin_src emacs-lisp
686(advice-add #'display-startup-echo-area-message :override #'ignore)
687(setq inhibit-startup-screen t
688 inhibit-startup-echo-area-message user-login-name)
689#+end_src
690
691*** More useful frame titles
692
693Show either the file name or the buffer name (in case the buffer isn't
694visiting a file). Borrowed from Emacs Prelude.
695
696#+begin_src emacs-lisp
697(setq frame-title-format
698 '("" invocation-name " - "
699 (:eval (if (buffer-file-name)
700 (abbreviate-file-name (buffer-file-name))
701 "%b"))))
702#+end_src
703
704*** Backups
705
706Emacs' default backup settings aren't that great. Let's use more
707sensible options. See documentation for the ~make-backup-file~
708variable.
709
710#+begin_src emacs-lisp
711(setq backup-by-copying t
712 version-control t
713 delete-old-versions t)
714#+end_src
715
716*** Auto revert
717
718Enable automatic reloading of changed buffers and files.
719
720#+begin_src emacs-lisp
721(global-auto-revert-mode 1)
722(setq auto-revert-verbose nil
723 global-auto-revert-non-file-buffers nil)
724#+end_src
725
726*** Always use space for indentation
727
728#+begin_src emacs-lisp
729(setq-default
730 indent-tabs-mode nil
731 require-final-newline t
732 tab-width 4)
733#+end_src
734
735*** Winner mode
736
737Enable =winner-mode=.
738
739#+begin_src emacs-lisp
740(winner-mode 1)
741#+end_src
742
743*** Don’t display =*compilation*= on success
744
745Based on https://stackoverflow.com/a/17788551, with changes to use
746=cl-letf= instead of the now obsolete =flet=.
747
748#+begin_src emacs-lisp
749(with-eval-after-load 'compile
750 (defun a/compilation-finish-function (buffer outstr)
751 (unless (string-match "finished" outstr)
752 (switch-to-buffer-other-window buffer))
753 t)
754
755 (setq compilation-finish-functions #'a/compilation-finish-function)
756
757 (require 'cl-macs)
758
759 (defadvice compilation-start
760 (around inhibit-display
761 (command &optional mode name-function highlight-regexp))
762 (if (not (string-match "^\\(find\\|grep\\)" command))
763 (cl-letf (((symbol-function 'display-buffer) #'ignore))
764 (save-window-excursion ad-do-it))
765 ad-do-it))
766 (ad-activate 'compilation-start))
767#+end_src
768
769*** Search for non-ASCII characters
770
771I’d like non-ASCII characters such as ‘’“”«»‹›áⓐ𝒶 to be selected when
772I search for their ASCII counterpart. Shoutout to [[http://endlessparentheses.com/new-in-emacs-25-1-easily-search-non-ascii-characters.html][endlessparentheses]]
773for this.
774
775#+begin_src emacs-lisp
776(setq search-default-mode #'char-fold-to-regexp)
777
778;; uncomment to extend this behaviour to query-replace
779;; (setq replace-char-fold t)
780#+end_src
781
782*** Cursor shape
783
784#+begin_src emacs-lisp
785(setq-default cursor-type 'bar)
786#+end_src
787
788*** Allow scrolling in Isearch
789
790#+begin_src emacs-lisp
791(setq isearch-allow-scroll t)
792#+end_src
793
794** Bindings
795
796Some bindings for functions from built-in GNU Emacs packages:
797
798#+begin_src emacs-lisp
799(bind-keys
800 ("C-c a i" . ielm)
801
802 ("C-c e b" . eval-buffer)
803 ("C-c e r" . eval-region)
804
805 ("C-c e i" . emacs-init-time)
806 ("C-c e u" . emacs-uptime)
807
808 ("C-c F m" . make-frame-command)
809 ("C-c F d" . delete-frame)
810 ("C-c F D" . delete-other-frames)
811
812 ("C-c o" . other-window)
813
814 ("C-S-h C" . describe-char)
815 ("C-S-h F" . describe-face)
816
817 ("C-x k" . kill-this-buffer)
818 ("C-x K" . kill-buffer)
819
820 ("s-p" . beginning-of-buffer)
821 ("s-n" . end-of-buffer))
822
823(when (display-graphic-p)
824 (unbind-key "C-z" global-map))
825#+end_src
826
827While at it, let's bind a few for some =straight-*= functions too:
828
829#+begin_src emacs-lisp
830(bind-keys
831 :prefix-map a/straight-prefix-map
832 :prefix "C-c p s"
833 ("u" . straight-use-package)
834 ("f" . straight-freeze-versions)
835 ("t" . straight-thaw-versions)
836 ("P" . straight-prune-build)
837 ("g" . straight-get-recipe)
838 ("r" . a/reload-init)
839 ;; M-x ^straight-.*-all$
840 ("a c" . straight-check-all)
841 ("a f" . straight-fetch-all)
842 ("a m" . straight-merge-all)
843 ("a n" . straight-normalize-all)
844 ("a F" . straight-pull-all)
845 ("a P" . straight-push-all)
846 ("a r" . straight-rebuild-all)
847 ;; M-x ^straight-.*-package$
848 ("p c" . straight-check-package)
849 ("p f" . straight-fetch-package)
850 ("p m" . straight-merge-package)
851 ("p n" . straight-normalize-package)
852 ("p F" . straight-pull-package)
853 ("p P" . straight-push-package)
854 ("p r" . straight-rebuild-package))
855#+end_src
856
857** Packages
858
859The packages in this section are absolutely essential to my everyday
860workflow, and they play key roles in how I do my computing. They
861immensely enhance the Emacs experience for me; both using Emacs, and
862customizing it.
863
864*** [[https://github.com/emacscollective/auto-compile][auto-compile]]
865
866#+begin_src emacs-lisp
867(use-package auto-compile
868 :demand t
869 :config
870 (auto-compile-on-load-mode)
871 (auto-compile-on-save-mode)
872 (setq auto-compile-display-buffer nil
873 auto-compile-mode-line-counter t
874 auto-compile-source-recreate-deletes-dest t
875 auto-compile-toggle-deletes-nonlib-dest t
876 auto-compile-update-autoloads t)
877 (add-hook 'auto-compile-inhibit-compile-hook
878 'auto-compile-inhibit-compile-detached-git-head))
879#+end_src
880
881*** [[https://orgmode.org/][Org]]
882
883#+begin_quote
884Org mode is for keeping notes, maintaining TODO lists, planning
885projects, and authoring documents with a fast and effective plain-text
886system.
887#+end_quote
888
889In short, my favourite way of life.
890
891We will use the =org-plus-contrib= package to get the whole deal:
892
893#+begin_src emacs-lisp
894(straight-use-package 'org-plus-contrib)
895#+end_src
896
897And here's where my actual Org configurations begin:
898
899#+begin_src emacs-lisp
900(use-feature org
901 :defer 0.5
902 :config
903 (setq org-src-tab-acts-natively t
904 org-src-preserve-indentation nil
905 org-edit-src-content-indentation 0
906 org-link-email-description-format "Email %c: %s" ; %.30s
907 org-highlight-latex-and-related '(entities)
908 org-use-speed-commands t
909 org-startup-folded 'content
910 org-catch-invisible-edits 'show-and-error
911 org-log-done 'time)
912 (add-to-list 'org-structure-template-alist '("L" . "src emacs-lisp") t)
913 :bind
914 (("C-c a o a" . org-agenda)
915 :map org-mode-map
916 ("M-L" . org-insert-last-stored-link)
917 ("s-T" . org-todo))
918 :hook ((org-mode . org-indent-mode)
919 (org-mode . auto-fill-mode)
920 (org-mode . flyspell-mode))
921 :custom
922 (org-agenda-files '("~/usr/org/todos/personal.org"
923 "~/usr/org/todos/masters.org"))
924 (org-agenda-start-on-weekday 0)
925 (org-latex-packages-alist '(("" "listings") ("" "color")))
926 :custom-face
927 '(org-block-begin-line ((t (:foreground "#5a5b5a" :background "#1d1f21"))))
928 '(org-block ((t (:background "#1d1f21"))))
929 '(org-latex-and-related ((t (:foreground "#b294bb")))))
930
931(use-feature ox-latex
932 :after ox
933 :config
934 (setq org-latex-listings 'listings
935 ;; org-latex-prefer-user-labels t
936 )
937 (add-to-list 'org-latex-classes
938 '("IEEEtran" "\\documentclass[11pt]{IEEEtran}"
939 ("\\section{%s}" . "\\section*{%s}")
940 ("\\subsection{%s}" . "\\subsection*{%s}")
941 ("\\subsubsection{%s}" . "\\subsubsection*{%s}")
942 ("\\paragraph{%s}" . "\\paragraph*{%s}")
943 ("\\subparagraph{%s}" . "\\subparagraph*{%s}"))
944 t)
945 (require 'ox-beamer))
946#+end_src
947
948**** asynchronous tangle
949
950=a/async-babel-tangle= is a function closely inspired by [[https://github.com/dieggsy/dotfiles/tree/cc10edf7701958eff1cd94d4081da544d882a28c/emacs.d#dotfiles][dieggsy's
951d/async-babel-tangle]] which uses [[https://github.com/jwiegley/emacs-async][async]] to asynchronously tangle an org
952file.
953
954#+begin_src emacs-lisp
955(with-eval-after-load 'org
956 (defvar a/show-async-tangle-results nil
957 "Keep *emacs* async buffers around for later inspection.")
958
959 (defvar a/show-async-tangle-time nil
960 "Show the time spent tangling the file.")
961
962 (defvar a/async-tangle-post-compile
963 (when a/byte-compiled-init "make build-init")
964 "If non-nil, pass to `compile' after successful tangle.")
965
966 ;; TODO: look into why directly byte-compiling init.el causes a
967 ;; number of problems, including magit-status not loading (busy
968 ;; waiting).
969 (defvar a/async-tangle-byte-recompile nil
970 "If non-nil, byte-recompile the file on successful tangle.")
971
972 (defun a/async-babel-tangle ()
973 "Tangle org file asynchronously."
974 (interactive)
975 (let* ((file-tangle-start-time (current-time))
976 (file (buffer-file-name))
977 (file-nodir (file-name-nondirectory file))
978 ;; (async-quiet-switch "-q")
979 (file-noext (file-name-sans-extension file)))
980 (async-start
981 `(lambda ()
982 (require 'org)
983 (org-babel-tangle-file ,file))
984 (unless a/show-async-tangle-results
985 `(lambda (result)
986 (if result
987 (progn
988 ;; (setq byte-compile-warnings '(not noruntime unresolved))
989 (message "Tangled %s%s"
990 ,file-nodir
991 (if a/show-async-tangle-time
992 (format " (%.3fs)"
993 (float-time (time-subtract (current-time)
994 ',file-tangle-start-time)))
995 ""))
996 (when a/async-tangle-post-compile
997 (compile a/async-tangle-post-compile))
998 (when a/async-tangle-byte-recompile
999 (byte-recompile-file (concat ,file-noext ".el"))))
1000 (message "Tangling %s failed" ,file-nodir))))))))
1001
1002(add-to-list
1003 'safe-local-variable-values
1004 '(eval add-hook 'after-save-hook #'a/async-babel-tangle 'append 'local))
1005#+end_src
1006
1007*** [[https://magit.vc/][Magit]]
1008
1009#+begin_quote
1010It's Magit! A Git porcelain inside Emacs.
1011#+end_quote
1012
1013Not just how I do git, but /the/ way to do git.
1014
1015#+begin_src emacs-lisp
1016(use-package magit
1017 :defer 0.5
1018 :bind (("C-x g" . magit-status)
1019 ("s-g s" . magit-status)
1020 ("s-g l" . magit-log-buffer-file))
1021 :config
1022 (magit-add-section-hook 'magit-status-sections-hook
1023 'magit-insert-modules
1024 'magit-insert-stashes
1025 'append)
1026 (setq magit-repository-directories '(("~/" . 0)
1027 ("~/src/git/" . 1)))
1028 (nconc magit-section-initial-visibility-alist
1029 '(([unpulled status] . show)
1030 ([unpushed status] . show)))
1031 :custom-face (magit-diff-file-heading ((t (:weight normal)))))
1032#+end_src
1033
1034*** recentf
1035
1036Recently opened files.
1037
1038#+begin_src emacs-lisp
1039(use-feature recentf
1040 :defer 0.2
1041 :config
1042 (add-to-list 'recentf-exclude "^/\\(?:ssh\\|su\\|sudo\\)?:")
1043 (setq recentf-max-saved-items 40))
1044#+end_src
1045
1046*** smex
1047
1048#+begin_quote
1049A smart M-x enhancement for Emacs.
1050#+end_quote
1051
1052Mostly because =counsel= needs it to remember history.
1053
1054#+begin_src emacs-lisp
1055(use-package smex)
1056#+end_src
1057
1058*** [[https://github.com/abo-abo/swiper][Ivy]] (and friends)
1059
1060#+begin_quote
1061Ivy - a generic completion frontend for Emacs, Swiper - isearch with
1062an overview, and more. Oh, man!
1063#+end_quote
1064
1065There's no way I could top that, so I won't attempt to.
1066
1067**** Ivy
1068
1069#+begin_src emacs-lisp
1070(use-package ivy
1071 :defer 0.3
1072 :bind
1073 (:map ivy-minibuffer-map
1074 ([escape] . keyboard-escape-quit)
1075 ([S-up] . ivy-previous-history-element)
1076 ([S-down] . ivy-next-history-element)
1077 ("DEL" . ivy-backward-delete-char))
1078 :config
1079 (setq ivy-wrap t)
1080 (ivy-mode 1)
1081 ;; :custom-face
1082 ;; (ivy-minibuffer-match-face-2 ((t (:background "#e99ce8" :weight semi-bold))))
1083 ;; (ivy-minibuffer-match-face-3 ((t (:background "#bbbbff" :weight semi-bold))))
1084 ;; (ivy-minibuffer-match-face-4 ((t (:background "#ffbbff" :weight semi-bold))))
1085)
1086#+end_src
1087
1088**** Swiper
1089
1090#+begin_src emacs-lisp
1091(use-package swiper
1092 :after ivy
1093 :bind (("C-s" . swiper-isearch)
1094 ("C-r" . swiper)
1095 ("C-S-s" . isearch-forward)))
1096#+end_src
1097
1098**** Counsel
1099
1100#+begin_src emacs-lisp
1101(use-package counsel
1102 :after ivy
1103 :bind (([remap execute-extended-command] . counsel-M-x)
1104 ([remap find-file] . counsel-find-file)
1105 ("s-r" . counsel-recentf)
1106 ("C-c x" . counsel-M-x)
1107 ("C-c f ." . counsel-find-file)
1108 :map minibuffer-local-map
1109 ("C-r" . counsel-minibuffer-history))
1110 :config
1111 (counsel-mode 1)
1112 (defalias 'locate #'counsel-locate))
1113#+end_src
1114
1115*** COMMENT Helm
1116
1117#+begin_src emacs-lisp
1118(use-package helm
1119 :commands (helm-M-x helm-mini helm-resume)
1120 :bind (("M-x" . helm-M-x)
1121 ("M-y" . helm-show-kill-ring)
1122 ("C-x b" . helm-mini)
1123 ("C-x C-b" . helm-buffers-list)
1124 ("C-x C-f" . helm-find-files)
1125 ("C-h r" . helm-info-emacs)
1126 ("s-r" . helm-recentf)
1127 ("C-s-r" . helm-resume)
1128 :map helm-map
1129 ("<tab>" . helm-execute-persistent-action)
1130 ("C-i" . helm-execute-persistent-action) ; Make TAB work in terminals
1131 ("C-z" . helm-select-action)) ; List actions
1132 :config (helm-mode 1))
1133#+end_src
1134
1135*** eshell
1136
1137#+begin_src emacs-lisp
1138(use-feature eshell
1139 :defer 0.5
1140 :commands eshell
1141 :bind ("C-c a s e" . eshell)
1142 :config
1143 (eval-when-compile (defvar eshell-prompt-regexp))
1144 (defun a/eshell-quit-or-delete-char (arg)
1145 (interactive "p")
1146 (if (and (eolp) (looking-back eshell-prompt-regexp nil))
1147 (eshell-life-is-too-much)
1148 (delete-char arg)))
1149
1150 (defun a/eshell-clear ()
1151 (interactive)
1152 (let ((inhibit-read-only t))
1153 (erase-buffer))
1154 (eshell-send-input))
1155
1156 (defun a/eshell-setup ()
1157 (make-local-variable 'company-idle-delay)
1158 (defvar company-idle-delay)
1159 (setq company-idle-delay nil)
1160 (bind-keys :map eshell-mode-map
1161 ("C-d" . a/eshell-quit-or-delete-char)
1162 ("C-S-l" . a/eshell-clear)
1163 ("M-r" . counsel-esh-history)
1164 ([tab] . company-complete)))
1165
1166 :hook (eshell-mode . a/eshell-setup)
1167 :custom
1168 (eshell-hist-ignoredups t)
1169 (eshell-input-filter 'eshell-input-filter-initial-space))
1170#+end_src
1171
1172*** Ibuffer
1173
1174#+begin_src emacs-lisp
1175(use-feature ibuffer
1176 :bind
1177 (("C-x C-b" . ibuffer-other-window)
1178 :map ibuffer-mode-map
1179 ("P" . ibuffer-backward-filter-group)
1180 ("N" . ibuffer-forward-filter-group)
1181 ("M-p" . ibuffer-do-print)
1182 ("M-n" . ibuffer-do-shell-command-pipe-replace))
1183 :config
1184 ;; Use human readable Size column instead of original one
1185 (define-ibuffer-column size-h
1186 (:name "Size" :inline t)
1187 (cond
1188 ((> (buffer-size) 1000000) (format "%7.1fM" (/ (buffer-size) 1000000.0)))
1189 ((> (buffer-size) 100000) (format "%7.0fk" (/ (buffer-size) 1000.0)))
1190 ((> (buffer-size) 1000) (format "%7.1fk" (/ (buffer-size) 1000.0)))
1191 (t (format "%8d" (buffer-size)))))
1192 :custom
1193 (ibuffer-saved-filter-groups
1194 '(("default"
1195 ("dired" (mode . dired-mode))
1196 ("org" (mode . org-mode))
1197 ("gnus"
1198 (or
1199 (mode . gnus-group-mode)
1200 (mode . gnus-summary-mode)
1201 (mode . gnus-article-mode)
1202 ;; not really, but...
1203 (mode . message-mode)))
1204 ("web"
1205 (or
1206 (mode . web-mode)
1207 (mode . css-mode)
1208 (mode . scss-mode)
1209 (mode . js2-mode)))
1210 ("shell"
1211 (or
1212 (mode . eshell-mode)
1213 (mode . shell-mode)
1214 (mode . term-mode)))
1215 ("programming"
1216 (or
1217 (mode . python-mode)
1218 (mode . c-mode)
1219 (mode . c++-mode)
1220 (mode . java-mode)
1221 (mode . emacs-lisp-mode)
1222 (mode . scheme-mode)
1223 (mode . haskell-mode)
1224 (mode . lean-mode)
1225 (mode . alloy-mode)))
1226 ("tex"
1227 (or
1228 (mode . bibtex-mode)
1229 (mode . latex-mode)))
1230 ("emacs"
1231 (or
1232 (name . "^\\*scratch\\*$")
1233 (name . "^\\*Messages\\*$")))
1234 ("erc" (mode . erc-mode)))))
1235 (ibuffer-formats
1236 '((mark modified read-only locked " "
1237 (name 18 18 :left :elide)
1238 " "
1239 (size-h 9 -1 :right)
1240 " "
1241 (mode 16 16 :left :elide)
1242 " " filename-and-process)
1243 (mark " "
1244 (name 16 -1)
1245 " " filename)))
1246 :hook (ibuffer . (lambda () (ibuffer-switch-to-saved-filter-groups "default"))))
1247#+end_src
1248
1249*** Outline
1250
1251#+begin_src emacs-lisp
1252(use-feature outline
1253 :hook (prog-mode . outline-minor-mode)
1254 :bind
1255 (:map
1256 outline-minor-mode-map
1257 ("<s-tab>" . outline-toggle-children)
1258 ("M-p" . outline-previous-visible-heading)
1259 ("M-n" . outline-next-visible-heading)
1260 :prefix-map a/outline-prefix-map
1261 :prefix "s-o"
1262 ("TAB" . outline-toggle-children)
1263 ("a" . outline-hide-body)
1264 ("H" . outline-hide-body)
1265 ("S" . outline-show-all)
1266 ("h" . outline-hide-subtree)
1267 ("s" . outline-show-subtree)))
1268#+end_src
1269
1270*** Dired
1271
1272#+begin_src emacs-lisp
1273(use-feature ls-lisp
1274 :custom (ls-lisp-dirs-first t))
1275
1276(use-feature dired
1277 :config
1278 (setq dired-listing-switches "-alh"
1279 ls-lisp-use-insert-directory-program nil)
1280
1281 ;; easily diff 2 marked files
1282 ;; https://oremacs.com/2017/03/18/dired-ediff/
1283 (defun dired-ediff-files ()
1284 (interactive)
1285 (require 'dired-aux)
1286 (defvar ediff-after-quit-hook-internal)
1287 (let ((files (dired-get-marked-files))
1288 (wnd (current-window-configuration)))
1289 (if (<= (length files) 2)
1290 (let ((file1 (car files))
1291 (file2 (if (cdr files)
1292 (cadr files)
1293 (read-file-name
1294 "file: "
1295 (dired-dwim-target-directory)))))
1296 (if (file-newer-than-file-p file1 file2)
1297 (ediff-files file2 file1)
1298 (ediff-files file1 file2))
1299 (add-hook 'ediff-after-quit-hook-internal
1300 (lambda ()
1301 (setq ediff-after-quit-hook-internal nil)
1302 (set-window-configuration wnd))))
1303 (error "no more than 2 files should be marked"))))
1304 :bind (:map dired-mode-map
1305 ("b" . dired-up-directory)
1306 ("e" . dired-ediff-files)
1307 ("E" . dired-toggle-read-only)
1308 ("\\" . dired-hide-details-mode)
1309 ("z" . (lambda ()
1310 (interactive)
1311 (a/dired-start-process "zathura"))))
1312 :hook (dired-mode . dired-hide-details-mode))
1313#+end_src
1314
1315*** Help
1316
1317#+begin_src emacs-lisp
1318(use-feature help
1319 :config
1320 (temp-buffer-resize-mode)
1321 (setq help-window-select t))
1322#+end_src
1323
1324*** Tramp
1325
1326#+begin_src emacs-lisp
1327(use-feature tramp
1328 :config
1329 (add-to-list 'tramp-default-proxies-alist '(nil "\\`root\\'" "/ssh:%h:"))
1330 (add-to-list 'tramp-default-proxies-alist '("localhost" nil nil))
1331 (add-to-list 'tramp-default-proxies-alist
1332 (list (regexp-quote (system-name)) nil nil)))
1333#+end_src
1334
1335*** Dash
1336
1337#+begin_src emacs-lisp
1338(use-package dash
1339 :config (dash-enable-font-lock))
1340#+end_src
1341
1342*** DocView
1343
1344#+begin_src emacs-lisp
1345(use-package doc-view
1346 :bind (:map doc-view-mode-map
1347 ("M-RET" . image-previous-line)))
1348#+end_src
1349
1350* Editing
1351:PROPERTIES:
1352:CUSTOM_ID: editing
1353:END:
1354
1355** =diff-hl=
1356
1357Highlight uncommitted changes in the left fringe.
1358
1359#+begin_src emacs-lisp
1360(use-package diff-hl
1361 :config
1362 (setq diff-hl-draw-borders nil)
1363 (global-diff-hl-mode)
1364 :hook (magit-post-refresh . diff-hl-magit-post-refresh))
1365#+end_src
1366
1367** ElDoc
1368
1369Display Lisp objects at point in the echo area.
1370
1371#+begin_src emacs-lisp
1372(use-feature eldoc
1373 :when (version< "25" emacs-version)
1374 :config (global-eldoc-mode))
1375#+end_src
1376
1377** paren
1378
1379Highlight matching parens.
1380
1381#+begin_src emacs-lisp
1382(use-feature paren
1383 :demand
1384 :config (show-paren-mode))
1385#+end_src
1386
1387** simple (for column numbers)
1388
1389#+begin_src emacs-lisp
1390(use-feature simple
1391 :config (column-number-mode))
1392#+end_src
1393
1394** =savehist=
1395
1396Save minibuffer history.
1397
1398#+begin_src emacs-lisp
1399(use-feature savehist
1400 :config (savehist-mode))
1401#+end_src
1402
1403** =saveplace=
1404
1405Automatically save place in each file.
1406
1407#+begin_src emacs-lisp
1408(use-feature saveplace
1409 :when (version< "25" emacs-version)
1410 :config (save-place-mode))
1411#+end_src
1412
1413** =prog-mode=
1414
1415#+begin_src emacs-lisp
1416(use-feature prog-mode
1417 :config (global-prettify-symbols-mode)
1418 (defun indicate-buffer-boundaries-left ()
1419 (setq indicate-buffer-boundaries 'left))
1420 (add-hook 'prog-mode-hook #'indicate-buffer-boundaries-left))
1421#+end_src
1422
1423** =text-mode=
1424
1425#+begin_src emacs-lisp
1426(use-feature text-mode
1427 :hook ((text-mode . indicate-buffer-boundaries-left)
1428 (text-mode . abbrev-mode)))
1429#+end_src
1430
1431** Company
1432
1433#+begin_src emacs-lisp
1434(use-package company
1435 :defer 0.6
1436 :bind
1437 (:map company-active-map
1438 ([tab] . company-complete-common-or-cycle)
1439 ([escape] . company-abort))
1440 :custom
1441 (company-minimum-prefix-length 1)
1442 (company-selection-wrap-around t)
1443 (company-dabbrev-char-regexp "\\sw\\|\\s_\\|[-_]")
1444 (company-dabbrev-downcase nil)
1445 (company-dabbrev-ignore-case nil)
1446 :config
1447 (global-company-mode t))
1448#+end_src
1449
1450** Flycheck
1451
1452#+begin_src emacs-lisp
1453(use-package flycheck
1454 :defer 0.6
1455 :hook (prog-mode . flycheck-mode)
1456 :bind
1457 (:map flycheck-mode-map
1458 ("M-P" . flycheck-previous-error)
1459 ("M-N" . flycheck-next-error))
1460 :config
1461 ;; Use the load-path from running Emacs when checking elisp files
1462 (setq flycheck-emacs-lisp-load-path 'inherit)
1463
1464 ;; Only flycheck when I actually save the buffer
1465 (setq flycheck-check-syntax-automatically '(mode-enabled save)))
1466
1467;; http://endlessparentheses.com/ispell-and-apostrophes.html
1468(use-package ispell
1469 :defer 0.6
1470 :config
1471 ;; ’ can be part of a word
1472 (setq ispell-local-dictionary-alist
1473 `((nil "[[:alpha:]]" "[^[:alpha:]]"
1474 "['\x2019]" nil ("-B") nil utf-8)))
1475 ;; don't send ’ to the subprocess
1476 (defun endless/replace-apostrophe (args)
1477 (cons (replace-regexp-in-string
1478 "’" "'" (car args))
1479 (cdr args)))
1480 (advice-add #'ispell-send-string :filter-args
1481 #'endless/replace-apostrophe)
1482
1483 ;; convert ' back to ’ from the subprocess
1484 (defun endless/replace-quote (args)
1485 (if (not (derived-mode-p 'org-mode))
1486 args
1487 (cons (replace-regexp-in-string
1488 "'" "’" (car args))
1489 (cdr args))))
1490 (advice-add #'ispell-parse-output :filter-args
1491 #'endless/replace-quote))
1492#+end_src
1493
1494* Programming modes
1495:PROPERTIES:
1496:CUSTOM_ID: programming-modes
1497:END:
1498
1499** Lisp
1500
1501#+begin_src emacs-lisp
1502(use-feature lisp-mode
1503 :config
1504 (add-hook 'emacs-lisp-mode-hook 'outline-minor-mode)
1505 (add-hook 'emacs-lisp-mode-hook 'reveal-mode)
1506 (defun indent-spaces-mode ()
1507 (setq indent-tabs-mode nil))
1508 (add-hook 'lisp-interaction-mode-hook #'indent-spaces-mode))
1509#+end_src
1510
1511** [[http://alloytools.org][Alloy]] (with [[https://github.com/dwwmmn/alloy-mode][alloy-mode]])
1512
1513#+begin_src emacs-lisp
1514(use-package alloy-mode
1515 :straight (:host github :repo "dwwmmn/alloy-mode")
1516 :mode "\\.als\\'"
1517 :config (setq alloy-basic-offset 2))
1518#+end_src
1519
1520** [[https://coq.inria.fr][Coq]] (with [[https://github.com/ProofGeneral/PG][Proof General]])
1521
1522#+begin_src emacs-lisp
1523(use-package proof-site ; Proof General
1524 :straight proof-general)
1525#+end_src
1526
1527** [[https://leanprover.github.io][Lean]] (with [[https://github.com/leanprover/lean-mode][lean-mode]])
1528
1529#+begin_src emacs-lisp
1530(eval-when-compile (defvar lean-mode-map))
1531(use-package lean-mode
1532 :defer 0.4
1533 :bind (:map lean-mode-map
1534 ("S-SPC" . company-complete))
1535 :config
1536 (require 'lean-input)
1537 (setq default-input-method "Lean"
1538 lean-input-tweak-all '(lean-input-compose
1539 (lean-input-prepend "/")
1540 (lean-input-nonempty))
1541 lean-input-user-translations '(("/" "/")))
1542 (lean-input-setup))
1543 #+end_src
1544
1545** Haskell
1546
1547*** [[https://github.com/haskell/haskell-mode][haskell-mode]]
1548
1549#+begin_src emacs-lisp
1550(use-package haskell-mode
1551 :config
1552 (setq haskell-indentation-layout-offset 4
1553 haskell-indentation-left-offset 4
1554 flycheck-checker 'haskell-hlint
1555 flycheck-disabled-checkers '(haskell-stack-ghc haskell-ghc)))
1556#+end_src
1557
1558*** [[https://github.com/jyp/dante][dante]]
1559
1560#+begin_src emacs-lisp
1561(use-package dante
1562 :after haskell-mode
1563 :commands dante-mode
1564 :hook (haskell-mode . dante-mode))
1565#+end_src
1566
1567*** [[https://github.com/mpickering/hlint-refactor-mode][hlint-refactor]]
1568
1569Emacs bindings for [[https://github.com/ndmitchell/hlint][hlint]]'s refactor option. This requires the refact
1570executable from [[https://github.com/mpickering/apply-refact][apply-refact]].
1571
1572#+begin_src emacs-lisp
1573(use-package hlint-refactor
1574 :after haskell-mode
1575 :bind (:map hlint-refactor-mode-map
1576 ("C-c l b" . hlint-refactor-refactor-buffer)
1577 ("C-c l r" . hlint-refactor-refactor-at-point))
1578 :hook (haskell-mode . hlint-refactor-mode))
1579#+end_src
1580
1581*** [[https://github.com/flycheck/flycheck-haskell][flycheck-haskell]]
1582
1583#+begin_src emacs-lisp
1584(use-package flycheck-haskell
1585 :after haskell-mode)
1586#+end_src
1587
1588*** [[https://github.com/ndmitchell/hlint/blob/20e116a043f2073c57b17b24ae6364b5e433ba7e/data/hs-lint.el][hs-lint.el]]
1589:PROPERTIES:
1590:header-args+: :tangle lisp/hs-lint.el :mkdirp yes
1591:END:
1592
1593Currently using =flycheck-haskell= with the =haskell-hlint= checker
1594instead.
1595
1596#+begin_src emacs-lisp :tangle no
1597;;; hs-lint.el --- minor mode for HLint code checking
1598
1599;; Copyright 2009 (C) Alex Ott
1600;;
1601;; Author: Alex Ott <alexott@gmail.com>
1602;; Keywords: haskell, lint, HLint
1603;; Requirements:
1604;; Status: distributed under terms of GPL2 or above
1605
1606;; Typical message from HLint looks like:
1607;;
1608;; /Users/ott/projects/lang-exp/haskell/test.hs:52:1: Eta reduce
1609;; Found:
1610;; count1 p l = length (filter p l)
1611;; Why not:
1612;; count1 p = length . filter p
1613
1614
1615(require 'compile)
1616
1617(defgroup hs-lint nil
1618 "Run HLint as inferior of Emacs, parse error messages."
1619 :group 'tools
1620 :group 'haskell)
1621
1622(defcustom hs-lint-command "hlint"
1623 "The default hs-lint command for \\[hlint]."
1624 :type 'string
1625 :group 'hs-lint)
1626
1627(defcustom hs-lint-save-files t
1628 "Save modified files when run HLint or no (ask user)"
1629 :type 'boolean
1630 :group 'hs-lint)
1631
1632(defcustom hs-lint-replace-with-suggestions nil
1633 "Replace user's code with suggested replacements"
1634 :type 'boolean
1635 :group 'hs-lint)
1636
1637(defcustom hs-lint-replace-without-ask nil
1638 "Replace user's code with suggested replacements automatically"
1639 :type 'boolean
1640 :group 'hs-lint)
1641
1642(defun hs-lint-process-setup ()
1643 "Setup compilation variables and buffer for `hlint'."
1644 (run-hooks 'hs-lint-setup-hook))
1645
1646;; regex for replace suggestions
1647;;
1648;; ^\(.*?\):\([0-9]+\):\([0-9]+\): .*
1649;; Found:
1650;; \s +\(.*\)
1651;; Why not:
1652;; \s +\(.*\)
1653
1654(defvar hs-lint-regex
1655 "^\\(.*?\\):\\([0-9]+\\):\\([0-9]+\\): .*[\n\C-m]Found:[\n\C-m]\\s +\\(.*\\)[\n\C-m]Why not:[\n\C-m]\\s +\\(.*\\)[\n\C-m]"
1656 "Regex for HLint messages")
1657
1658(defun make-short-string (str maxlen)
1659 (if (< (length str) maxlen)
1660 str
1661 (concat (substring str 0 (- maxlen 3)) "...")))
1662
1663(defun hs-lint-replace-suggestions ()
1664 "Perform actual replacement of suggestions"
1665 (goto-char (point-min))
1666 (while (re-search-forward hs-lint-regex nil t)
1667 (let* ((fname (match-string 1))
1668 (fline (string-to-number (match-string 2)))
1669 (old-code (match-string 4))
1670 (new-code (match-string 5))
1671 (msg (concat "Replace '" (make-short-string old-code 30)
1672 "' with '" (make-short-string new-code 30) "'"))
1673 (bline 0)
1674 (eline 0)
1675 (spos 0)
1676 (new-old-code ""))
1677 (save-excursion
1678 (switch-to-buffer (get-file-buffer fname))
1679 (goto-char (point-min))
1680 (forward-line (1- fline))
1681 (beginning-of-line)
1682 (setf bline (point))
1683 (when (or hs-lint-replace-without-ask
1684 (yes-or-no-p msg))
1685 (end-of-line)
1686 (setf eline (point))
1687 (beginning-of-line)
1688 (setf old-code (regexp-quote old-code))
1689 (while (string-match "\\\\ " old-code spos)
1690 (setf new-old-code (concat new-old-code
1691 (substring old-code spos (match-beginning 0))
1692 "\\ *"))
1693 (setf spos (match-end 0)))
1694 (setf new-old-code (concat new-old-code (substring old-code spos)))
1695 (remove-text-properties bline eline '(composition nil))
1696 (when (re-search-forward new-old-code eline t)
1697 (replace-match new-code nil t)))))))
1698
1699(defun hs-lint-finish-hook (buf msg)
1700 "Function, that is executed at the end of HLint execution"
1701 (if hs-lint-replace-with-suggestions
1702 (hs-lint-replace-suggestions)
1703 (next-error 1 t)))
1704
1705(define-compilation-mode hs-lint-mode "HLint"
1706 "Mode for check Haskell source code."
1707 (set (make-local-variable 'compilation-process-setup-function)
1708 'hs-lint-process-setup)
1709 (set (make-local-variable 'compilation-disable-input) t)
1710 (set (make-local-variable 'compilation-scroll-output) nil)
1711 (set (make-local-variable 'compilation-finish-functions)
1712 (list 'hs-lint-finish-hook))
1713 )
1714
1715(defun hs-lint ()
1716 "Run HLint for current buffer with haskell source"
1717 (interactive)
1718 (save-some-buffers hs-lint-save-files)
1719 (compilation-start (concat hs-lint-command " \"" buffer-file-name "\"")
1720 'hs-lint-mode))
1721
1722(provide 'hs-lint)
1723;;; hs-lint.el ends here
1724#+end_src
1725
1726#+begin_src emacs-lisp :tangle no
1727(use-package hs-lint
1728 :load-path "lisp/"
1729 :bind (:map haskell-mode-map
1730 ("C-c l l" . hs-lint)))
1731#+end_src
1732
1733** Web
1734
1735*** SGML and HTML
1736
1737#+begin_src emacs-lisp
1738(use-package sgml-mode
1739 :config
1740 (setq sgml-basic-offset 2))
1741#+end_src
1742
1743*** CSS and SCSS
1744
1745#+begin_src emacs-lisp
1746(use-package css-mode
1747 :config
1748 (setq css-indent-offset 2))
1749#+end_src
1750
1751*** Web mode
1752
1753#+begin_src emacs-lisp
1754(use-package web-mode
1755 :mode "\\.html\\'"
1756 :config
1757 (a/setq-every 2
1758 web-mode-code-indent-offset
1759 web-mode-css-indent-offset
1760 web-mode-markup-indent-offset))
1761#+end_src
1762
1763*** Emmet mode
1764
1765#+begin_src emacs-lisp
1766(use-package emmet-mode
1767 :after (:any web-mode css-mode sgml-mode)
1768 :bind* (("C-)" . emmet-next-edit-point)
1769 ("C-(" . emmet-prev-edit-point))
1770 :config
1771 (unbind-key "C-j" emmet-mode-keymap)
1772 (setq emmet-move-cursor-between-quotes t)
1773 :hook (web-mode css-mode html-mode sgml-mode))
1774#+end_src
1775
1776** Java
1777
1778*** COMMENT meghanada
1779
1780#+begin_src emacs-lisp
1781(use-package meghanada
1782 :bind
1783 (:map meghanada-mode-map
1784 (("C-M-o" . meghanada-optimize-import)
1785 ("C-M-t" . meghanada-import-all)))
1786 :hook (java-mode . meghanada-mode))
1787#+end_src
1788
1789*** COMMENT lsp-java
1790
1791#+begin_comment
1792dependencies:
1793
1794ace-window
1795avy
1796bui
1797company-lsp
1798dap-mode
1799lsp-java
1800lsp-mode
1801lsp-ui
1802pfuture
1803tree-mode
1804treemacs
1805#+end_comment
1806
1807#+begin_src emacs-lisp
1808(use-package treemacs
1809 :config (setq treemacs-never-persist t))
1810
1811(use-package yasnippet
1812 :config
1813 ;; (yas-global-mode)
1814 )
1815
1816(use-package lsp-mode
1817 :init (setq lsp-eldoc-render-all nil
1818 lsp-highlight-symbol-at-point nil)
1819 )
1820
1821(use-package hydra)
1822
1823(use-package company-lsp
1824 :after company
1825 :config
1826 (setq company-lsp-cache-candidates t
1827 company-lsp-async t))
1828
1829(use-package lsp-ui
1830 :config
1831 (setq lsp-ui-sideline-update-mode 'point))
1832
1833(use-package lsp-java
1834 :config
1835 (add-hook 'java-mode-hook
1836 (lambda ()
1837 (setq-local company-backends (list 'company-lsp))))
1838
1839 (add-hook 'java-mode-hook 'lsp-java-enable)
1840 (add-hook 'java-mode-hook 'flycheck-mode)
1841 (add-hook 'java-mode-hook 'company-mode)
1842 (add-hook 'java-mode-hook 'lsp-ui-mode))
1843
1844(use-package dap-mode
1845 :after lsp-mode
1846 :config
1847 (dap-mode t)
1848 (dap-ui-mode t))
1849
1850(use-package dap-java
1851 :after (lsp-java))
1852
1853(use-package lsp-java-treemacs
1854 :after (treemacs))
1855#+end_src
1856
1857*** COMMENT eclim
1858
1859#+begin_src emacs-lisp
1860(use-package eclim
1861 :bind (:map eclim-mode-map ("S-SPC" . company-complete))
1862 :hook ((java-mode . eclim-mode)
1863 (eclim-mode . (lambda ()
1864 (make-local-variable 'company-idle-delay)
1865 (defvar company-idle-delay)
1866 ;; (setq company-idle-delay 0.7)
1867 (setq company-idle-delay nil))))
1868 :custom
1869 (eclim-auto-save nil)
1870 ;; (eclimd-default-workspace "~/src/eclipse-workspace-exp")
1871 (eclim-executable "~/.p2/pool/plugins/org.eclim_2.8.0/bin/eclim")
1872 (eclim-eclipse-dirs '("~/usr/eclipse/dsl-2018-09/eclipse")))
1873#+end_src
1874
1875** geiser
1876
1877#+begin_src emacs-lisp
1878(use-package geiser)
1879
1880(use-feature geiser-guile
1881 :config
1882 (setq geiser-guile-load-path "~/src/git/guix"))
1883#+end_src
1884
1885** guix
1886
1887#+begin_src emacs-lisp
1888(use-package guix)
1889#+end_src
1890
1891** COMMENT TeX
1892
1893#+begin_src emacs-lisp
1894(use-package auctex
1895 :custom
1896 (font-latex-fontify-sectioning 'color))
1897#+end_src
1898
1899* Emacs enhancements
1900:PROPERTIES:
1901:CUSTOM_ID: emacs-enhancements
1902:END:
1903
1904** man
1905
1906#+begin_src emacs-lisp
1907(use-feature man
1908 :config (setq Man-width 80))
1909#+end_src
1910
1911** [[https://github.com/justbur/emacs-which-key][which-key]]
1912
1913#+begin_quote
1914Emacs package that displays available keybindings in popup
1915#+end_quote
1916
1917#+begin_src emacs-lisp
1918(use-package which-key
1919 :defer 0.4
1920 :config
1921 (which-key-add-key-based-replacements
1922 ;; prefixes for global prefixes and minor modes
1923 "C-c @" "outline"
1924 "C-c !" "flycheck"
1925 "C-c 8" "typo"
1926 "C-c 8 -" "typo/dashes"
1927 "C-c 8 <" "typo/left-brackets"
1928 "C-c 8 >" "typo/right-brackets"
1929 "C-x 8" "unicode"
1930 "C-x a" "abbrev/expand"
1931 "C-x r" "rectangle/register/bookmark"
1932 "C-x v" "version control"
1933 ;; prefixes for my personal bindings
1934 "C-c a" "applications"
1935 "C-c a e" "erc"
1936 "C-c a o" "org"
1937 "C-c a s" "shells"
1938 "C-c p" "package-management"
1939 ;; "C-c p e" "package-management/epkg"
1940 "C-c p s" "straight.el"
1941 "C-c psa" "all"
1942 "C-c psp" "package"
1943 "C-c c" "compile-and-comments"
1944 "C-c e" "eval"
1945 "C-c f" "files"
1946 "C-c F" "frames"
1947 "C-S-h" "help(ful)"
1948 "C-c m" "multiple-cursors"
1949 "C-c P" "projectile"
1950 "C-c P s" "projectile/search"
1951 "C-c P x" "projectile/execute"
1952 "C-c P 4" "projectile/other-window"
1953 "C-c q" "boxquote"
1954 "s-g" "magit"
1955 "s-o" "outline"
1956 "s-t" "themes")
1957
1958 ;; prefixes for major modes
1959 (which-key-add-major-mode-key-based-replacements 'message-mode
1960 "C-c f" "footnote")
1961 (which-key-add-major-mode-key-based-replacements 'org-mode
1962 "C-c C-v" "org-babel")
1963 (which-key-add-major-mode-key-based-replacements 'web-mode
1964 "C-c C-a" "web/attributes"
1965 "C-c C-b" "web/blocks"
1966 "C-c C-d" "web/dom"
1967 "C-c C-e" "web/element"
1968 "C-c C-t" "web/tags")
1969
1970 (which-key-mode)
1971 :custom
1972 (which-key-add-column-padding 5)
1973 (which-key-max-description-length 32))
1974#+end_src
1975
1976** theme
1977
1978#+begin_src emacs-lisp
1979(add-to-list 'custom-theme-load-path "~/.emacs.d/lisp")
1980(load-theme 'tangomod t)
1981#+end_src
1982
1983** smart-mode-line
1984
1985#+begin_src emacs-lisp
1986(use-package smart-mode-line
1987 :commands (sml/apply-theme)
1988 :demand
1989 :config
1990 (sml/setup))
1991#+end_src
1992
1993** doom-themes
1994
1995#+begin_src emacs-lisp
1996(use-package doom-themes)
1997#+end_src
1998
1999** theme helper functions
2000
2001#+begin_src emacs-lisp
2002(defvar a/org-mode-font-lock-keywords
2003 '(("[ \t]*\\(#\\+\\(BEGIN\\|END\\|begin\\|end\\)_\\(\\S-+\\)\\)[ \t]*\\([^\n:]*\\)"
2004 (1 '(:foreground "#5a5b5a" :background "#292b2b") t) ; directive
2005 (3 '(:foreground "#81a2be" :background "#292b2b") t) ; kind
2006 (4 '(:foreground "#c5c8c6") t)))) ; title
2007
2008(defun a/lights-on ()
2009 "Enable my favourite light theme."
2010 (interactive)
2011 (mapc #'disable-theme custom-enabled-themes)
2012 (load-theme 'tangomod t)
2013 (sml/apply-theme 'automatic)
2014 (font-lock-remove-keywords
2015 'org-mode a/org-mode-font-lock-keywords))
2016
2017(defun a/lights-off ()
2018 "Go dark."
2019 (interactive)
2020 (mapc #'disable-theme custom-enabled-themes)
2021 (load-theme 'doom-tomorrow-night t)
2022 (sml/apply-theme 'automatic)
2023 (font-lock-add-keywords
2024 'org-mode a/org-mode-font-lock-keywords t))
2025
2026(bind-keys
2027 ("s-t d" . a/lights-off)
2028 ("s-t l" . a/lights-on))
2029#+end_src
2030
2031** [[https://github.com/bbatsov/crux][crux]]
2032
2033#+begin_src emacs-lisp
2034(use-package crux ; results in Waiting for git... [2 times]
2035 :defer 0.4
2036 :bind (("C-c b k" . crux-kill-other-buffers)
2037 ("C-c d" . crux-duplicate-current-line-or-region)
2038 ("C-c D" . crux-duplicate-and-comment-current-line-or-region)
2039 ("C-c f c" . crux-copy-file-preserve-attributes)
2040 ("C-c f d" . crux-delete-file-and-buffer)
2041 ("C-c f r" . crux-rename-file-and-buffer)
2042 ("C-c j" . crux-top-join-line)
2043 ("C-S-j" . crux-top-join-line)))
2044#+end_src
2045
2046** [[https://github.com/alezost/mwim.el][mwim]]
2047
2048#+begin_src emacs-lisp
2049(use-package mwim
2050 :bind (("C-a" . mwim-beginning-of-code-or-line)
2051 ("C-e" . mwim-end-of-code-or-line)
2052 ("<home>" . mwim-beginning-of-line-or-code)
2053 ("<end>" . mwim-end-of-line-or-code)))
2054#+end_src
2055
2056** projectile
2057
2058#+begin_src emacs-lisp
2059(use-package projectile
2060 :bind-keymap ("C-c P" . projectile-command-map)
2061 :config
2062 (projectile-mode)
2063
2064 (defun my-projectile-invalidate-cache (&rest _args)
2065 ;; ignore the args to `magit-checkout'
2066 (projectile-invalidate-cache nil))
2067
2068 (eval-after-load 'magit-branch
2069 '(progn
2070 (advice-add 'magit-checkout
2071 :after #'my-projectile-invalidate-cache)
2072 (advice-add 'magit-branch-and-checkout
2073 :after #'my-projectile-invalidate-cache)))
2074 :custom (projectile-completion-system 'ivy))
2075#+end_src
2076
2077** [[https://github.com/Wilfred/helpful][helpful]]
2078
2079#+begin_src emacs-lisp
2080(use-package helpful
2081 :defer 0.6
2082 :bind
2083 (("C-S-h c" . helpful-command)
2084 ("C-S-h f" . helpful-callable) ; helpful-function
2085 ("C-S-h v" . helpful-variable)
2086 ("C-S-h k" . helpful-key)
2087 ("C-S-h p" . helpful-at-point)))
2088#+end_src
2089
2090** [[https://github.com/EricCrosson/unkillable-scratch][unkillable-scratch]]
2091
2092Make =*scratch*= and =*Messages*= unkillable.
2093
2094#+begin_src emacs-lisp
2095(use-package unkillable-scratch
2096 :defer 0.6
2097 :config
2098 (unkillable-scratch 1)
2099 :custom
2100 (unkillable-buffers '("^\\*scratch\\*$" "^\\*Messages\\*$")))
2101#+end_src
2102
2103** [[https://github.com/davep/boxquote.el][boxquote.el]]
2104
2105#+begin_example
2106,----
2107| make pretty boxed quotes like this
2108`----
2109#+end_example
2110
2111#+begin_src emacs-lisp
2112(use-package boxquote
2113 :defer 0.6
2114 :bind
2115 (:prefix-map a/boxquote-prefix-map
2116 :prefix "C-c q"
2117 ("b" . boxquote-buffer)
2118 ("B" . boxquote-insert-buffer)
2119 ("d" . boxquote-defun)
2120 ("F" . boxquote-insert-file)
2121 ("hf" . boxquote-describe-function)
2122 ("hk" . boxquote-describe-key)
2123 ("hv" . boxquote-describe-variable)
2124 ("hw" . boxquote-where-is)
2125 ("k" . boxquote-kill)
2126 ("p" . boxquote-paragraph)
2127 ("q" . boxquote-boxquote)
2128 ("r" . boxquote-region)
2129 ("s" . boxquote-shell-command)
2130 ("t" . boxquote-text)
2131 ("T" . boxquote-title)
2132 ("u" . boxquote-unbox)
2133 ("U" . boxquote-unbox-region)
2134 ("y" . boxquote-yank)
2135 ("M-q" . boxquote-fill-paragraph)
2136 ("M-w" . boxquote-kill-ring-save)))
2137#+end_src
2138
2139Also see [[https://www.emacswiki.org/emacs/rebox2][rebox2]].
2140
2141** orgalist
2142
2143#+begin_src emacs-lisp
2144(use-package orgalist
2145 :disabled t
2146 :after message
2147 :hook (message-mode . orgalist-mode))
2148#+end_src
2149
2150** typo.el
2151
2152#+begin_src emacs-lisp
2153(use-package typo
2154 :defer 0.5
2155 :config
2156 (typo-global-mode 1)
2157 :hook (text-mode . typo-mode))
2158#+end_src
2159
2160** hl-todo
2161
2162#+begin_src emacs-lisp
2163(use-package hl-todo
2164 :defer 0.5
2165 :config
2166 (global-hl-todo-mode))
2167#+end_src
2168
2169** shrink-path
2170
2171#+begin_src emacs-lisp
2172(use-package shrink-path
2173 :defer 0.5
2174 :after eshell
2175 :config
2176 (defvar user-@-host (concat (user-login-name) "@" (system-name) " "))
2177 (defun +eshell/prompt ()
2178 (let ((base/dir (shrink-path-prompt default-directory)))
2179 (concat (propertize user-@-host 'face 'default)
2180 (propertize (car base/dir)
2181 'face 'font-lock-comment-face)
2182 (propertize (cdr base/dir)
2183 'face 'font-lock-constant-face)
2184 (propertize "> " 'face 'default))))
2185 (setq eshell-prompt-regexp (concat user-@-host ".*> ")
2186 eshell-prompt-function #'+eshell/prompt))
2187#+end_src
2188
2189** [[https://github.com/peterwvj/eshell-up][eshell-up]]
2190
2191#+begin_src emacs-lisp
2192(use-package eshell-up
2193 :after eshell
2194 :commands eshell-up)
2195#+end_src
2196
2197** multi-term
2198
2199#+begin_src emacs-lisp
2200(use-package multi-term
2201 :defer 0.6
2202 :bind (("C-c a s m" . multi-term-dedicated-toggle)
2203 :map term-mode-map
2204 ("C-c C-j" . term-char-mode)
2205 :map term-raw-map
2206 ("C-c C-j" . term-line-mode))
2207 :config
2208 (setq multi-term-program "/bin/screen"
2209 ;; TODO: add separate bindings for connecting to existing
2210 ;; session vs. always creating a new one
2211 multi-term-dedicated-select-after-open-p t
2212 multi-term-dedicated-window-height 20
2213 multi-term-dedicated-max-window-height 30
2214 term-bind-key-alist
2215 '(("C-c C-c" . term-interrupt-subjob)
2216 ("C-c C-e" . term-send-esc)
2217 ("C-k" . kill-line)
2218 ("C-y" . term-paste)
2219 ("M-f" . term-send-forward-word)
2220 ("M-b" . term-send-backward-word)
2221 ("M-p" . term-send-up)
2222 ("M-n" . term-send-down)
2223 ("<C-backspace>" . term-send-backward-kill-word)
2224 ("<M-DEL>" . term-send-backward-kill-word)
2225 ("M-d" . term-send-delete-word)
2226 ("M-," . term-send-raw)
2227 ("M-." . comint-dynamic-complete))
2228 term-unbind-key-alist
2229 '("C-z" "C-x" "C-c" "C-h" "C-y" "<ESC>")))
2230#+end_src
2231
2232** page-break-lines
2233
2234#+begin_src emacs-lisp
2235(use-package page-break-lines
2236 :config
2237 (global-page-break-lines-mode))
2238#+end_src
2239
2240** expand-region
2241
2242#+begin_src emacs-lisp
2243(use-package expand-region
2244 :bind ("C-=" . er/expand-region))
2245#+end_src
2246
2247** multiple-cursors
2248
2249#+begin_src emacs-lisp
2250(use-package multiple-cursors
2251 :bind
2252 (("C-S-<mouse-1>" . mc/add-cursor-on-click)
2253 (:prefix-map a/mc-prefix-map
2254 :prefix "C-c m"
2255 ("c" . mc/edit-lines)
2256 ("n" . mc/mark-next-like-this)
2257 ("p" . mc/mark-previous-like-this)
2258 ("a" . mc/mark-all-like-this))))
2259#+end_src
2260
2261** forge
2262
2263#+begin_src emacs-lisp
2264(use-package forge
2265 :after magit
2266 :demand)
2267#+end_src
2268
2269** yasnippet
2270
2271#+begin_src emacs-lisp
2272(use-package yasnippet
2273 :defer 0.6
2274 :config
2275 (defconst yas-verbosity-cur yas-verbosity)
2276 (setq yas-verbosity 2)
2277 (add-to-list 'yas-snippet-dirs "~/src/git/guix/etc/snippets")
2278 (yas-reload-all)
2279 (setq yas-verbosity yas-verbosity-cur)
2280 :hook
2281 (text-mode . yas-minor-mode))
2282#+end_src
2283
2284** Debbugs
2285
2286#+begin_src emacs-lisp
2287(use-package debbugs
2288 :straight (debbugs
2289 :host github
2290 :repo "emacs-straight/debbugs"
2291 :files (:defaults "Debbugs.wsdl")))
2292#+end_src
2293
2294** org-ref
2295
2296#+begin_src emacs-lisp
2297(use-package org-ref
2298 :init
2299 (a/setq-every '("~/usr/org/references.bib")
2300 reftex-default-bibliography
2301 org-ref-default-bibliography)
2302 (setq
2303 org-ref-bibliography-notes "~/usr/org/notes.org"
2304 org-ref-pdf-directory "~/usr/org/bibtex-pdfs/"))
2305#+end_src
2306
2307** slack
2308
2309Not an "Emacs enhancement", but just so that I don't have to use the
2310proprietary web interface. Needed for interacting with the URA(s)
2311this summer.
2312
2313#+begin_src emacs-lisp
2314(use-package slack
2315 :commands (slack-start)
2316 :init
2317 (eval-when-compile ; silence the byte-compiler
2318 (defvar url-http-data nil)
2319 (defvar url-http-extra-headers nil)
2320 (defvar url-http-method nil)
2321 (defvar url-callback-function nil)
2322 (defvar url-callback-arguments nil)
2323 (defvar oauth--token-data nil))
2324 (setq slack-buffer-emojify t
2325 slack-prefer-current-team t)
2326 :config
2327 (slack-register-team
2328 :name "nday-students"
2329 :default t
2330 :token nday-students-token
2331 :subscribed-channels '(general)
2332 :full-and-display-names t)
2333 (add-to-list 'swiper-font-lock-exclude 'slack-message-buffer-mode t)
2334 (setq lui-time-stamp-format "[%Y-%m-%d %H:%M:%S]"
2335 lui-time-stamp-only-when-changed-p t
2336 lui-time-stamp-position 'right)
2337 :bind
2338 (("C-c s s" . slack-start)
2339 ("C-c s u" . slack-select-unread-rooms)
2340 ("C-c s b" . slack-select-rooms)
2341 ("C-c s t" . slack-change-current-team)
2342 ("C-c s c" . slack-ws-close)
2343 :map slack-mode-map
2344 ("M-p" . slack-buffer-goto-prev-message)
2345 ("M-n" . slack-buffer-goto-next-message)
2346 ("C-c e" . slack-message-edit)
2347 ("C-c k" . slack-message-delete)
2348 ("C-c C-k" . slack-channel-leave)
2349 ("C-c r a" . slack-message-add-reaction)
2350 ("C-c r r" . slack-message-remove-reaction)
2351 ("C-c r s" . slack-message-show-reaction-users)
2352 ("C-c p l" . slack-room-pins-list)
2353 ("C-c p a" . slack-message-pins-add)
2354 ("C-c p r" . slack-message-pins-remove)
2355 ("@" . slack-message-embed-mention)
2356 ("#" . slack-message-embed-channel)))
2357
2358(use-package alert
2359 :commands (alert)
2360 :init
2361 (setq alert-default-style 'notifier))
2362#+end_src
2363
2364* Email
2365:PROPERTIES:
2366:CUSTOM_ID: email
2367:END:
2368
2369#+begin_src emacs-lisp
2370(defvar a/maildir (expand-file-name "~/mail/"))
2371(with-eval-after-load 'recentf
2372 (add-to-list 'recentf-exclude a/maildir))
2373#+end_src
2374
2375** Gnus
2376
2377#+begin_src emacs-lisp
2378(setq
2379 a/gnus-init-file (no-littering-expand-etc-file-name "gnus")
2380 mail-user-agent 'gnus-user-agent
2381 read-mail-command 'gnus)
2382
2383(use-feature gnus
2384 :bind (("s-m" . gnus)
2385 ("s-M" . gnus-unplugged))
2386 :init
2387 (setq
2388 gnus-select-method '(nnnil "")
2389 gnus-secondary-select-methods
2390 '((nnimap "amin"
2391 (nnimap-stream plain)
2392 (nnimap-address "127.0.0.1")
2393 (nnimap-server-port 143)
2394 (nnimap-authenticator plain)
2395 (nnimap-user "amin@bndl.org"))
2396 (nnimap "uwaterloo"
2397 (nnimap-stream plain)
2398 (nnimap-address "127.0.0.1")
2399 (nnimap-server-port 143)
2400 (nnimap-authenticator plain)
2401 (nnimap-user "abandali@uwaterloo.ca"))
2402 (nnimap "csclub"
2403 (nnimap-stream plain)
2404 (nnimap-address "127.0.0.1")
2405 (nnimap-server-port 143)
2406 (nnimap-authenticator plain)
2407 (nnimap-user "abandali@csclub.uw")))
2408 gnus-message-archive-group "nnimap+amin:Sent"
2409 gnus-parameters
2410 '(("gnu\\.deepspec"
2411 (to-address . "deepspec@lists.cs.princeton.edu")
2412 (to-list . "deepspec@lists.cs.princeton.edu"))
2413 ("gnu\\.emacs-devel"
2414 (to-address . "emacs-devel@gnu.org")
2415 (to-list . "emacs-devel@gnu.org"))
2416 ("gnu\\.emacs-orgmode"
2417 (to-address . "emacs-orgmode@gnu.org")
2418 (to-list . "emacs-orgmode@gnu.org"))
2419 ("gnu\\.emacsconf-discuss"
2420 (to-address . "emacsconf-discuss@gnu.org")
2421 (to-list . "emacsconf-discuss@gnu.org"))
2422 ("gnu\\.fencepost-users"
2423 (to-address . "fencepost-users@gnu.org")
2424 (to-list . "fencepost-users@gnu.org"))
2425 ("gnu\\.gnunet-developers"
2426 (to-address . "gnunet-developers@gnu.org")
2427 (to-list . "gnunet-developers@gnu.org"))
2428 ("gnu\\.guile-devel"
2429 (to-address . "guile-devel@gnu.org")
2430 (to-list . "guile-devel@gnu.org"))
2431 ("gnu\\.guix-devel"
2432 (to-address . "guix-devel@gnu.org")
2433 (to-list . "guix-devel@gnu.org"))
2434 ("gnu\\.haskell-art"
2435 (to-address . "haskell-art@we.lurk.org")
2436 (to-list . "haskell-art@we.lurk.org"))
2437 ("gnu\\.haskell-cafe"
2438 (to-address . "haskell-cafe@haskell.org")
2439 (to-list . "haskell-cafe@haskell.org"))
2440 ("gnu\\.help-gnu-emacs"
2441 (to-address . "help-gnu-emacs@gnu.org")
2442 (to-list . "help-gnu-emacs@gnu.org"))
2443 ("gnu\\.info-gnu-emacs"
2444 (to-address . "info-gnu-emacs@gnu.org")
2445 (to-list . "info-gnu-emacs@gnu.org"))
2446 ("gnu\\.info-guix"
2447 (to-address . "info-guix@gnu.org")
2448 (to-list . "info-guix@gnu.org"))
2449 ("gnu\\.notmuch"
2450 (to-address . "notmuch@notmuchmail.org")
2451 (to-list . "notmuch@notmuchmail.org"))
2452 ("gnu\\.parabola-dev"
2453 (to-address . "dev@lists.parabola.nu")
2454 (to-list . "dev@lists.parabola.nu"))
2455 ("gnu\\.webmasters"
2456 (to-address . "webmasters@gnu.org")
2457 (to-list . "webmasters@gnu.org"))
2458 ("gnu\\.www-commits"
2459 (to-address . "www-commits@gnu.org")
2460 (to-list . "www-commits@gnu.org"))
2461 ("gnu\\.www-discuss"
2462 (to-address . "www-discuss@gnu.org")
2463 (to-list . "www-discuss@gnu.org"))
2464 ("gnu\\.~bandali\\.public-inbox"
2465 (to-address . "~bandali/public-inbox@lists.sr.ht")
2466 (to-list . "~bandali/public-inbox@lists.sr.ht"))
2467 ("gnu\\.~sircmpwn\\.srht-admins"
2468 (to-address . "~sircmpwn/sr.ht-admins@lists.sr.ht")
2469 (to-list . "~sircmpwn/sr.ht-admins@lists.sr.ht"))
2470 ("gnu\\.~sircmpwn\\.srht-announce"
2471 (to-address . "~sircmpwn/sr.ht-announce@lists.sr.ht")
2472 (to-list . "~sircmpwn/sr.ht-announce@lists.sr.ht"))
2473 ("gnu\\.~sircmpwn\\.srht-dev"
2474 (to-address . "~sircmpwn/sr.ht-dev@lists.sr.ht")
2475 (to-list . "~sircmpwn/sr.ht-dev@lists.sr.ht"))
2476 ("gnu\\.~sircmpwn\\.srht-discuss"
2477 (to-address . "~sircmpwn/sr.ht-discuss@lists.sr.ht")
2478 (to-list . "~sircmpwn/sr.ht-discuss@lists.sr.ht"))
2479 ("gnu.*"
2480 (gcc-self . t))
2481 ("gnu\\."
2482 (subscribed . t)))
2483 gnus-large-newsgroup 50
2484 gnus-home-directory (no-littering-expand-var-file-name "gnus/")
2485 gnus-directory (concat gnus-home-directory "news/")
2486 message-directory (concat gnus-home-directory "mail/")
2487 nndraft-directory (concat gnus-home-directory "drafts/")
2488 gnus-save-newsrc-file nil
2489 gnus-read-newsrc-file nil
2490 gnus-interactive-exit nil
2491 gnus-gcc-mark-as-read t)
2492 :config
2493 (require 'ebdb)
2494 (require 'ebdb-mua)
2495 (require 'ebdb-gnus))
2496
2497(use-feature gnus-art
2498 :config
2499 (setq
2500 gnus-visible-headers
2501 (concat gnus-visible-headers "\\|^List-Id:\\|^X-RT-Originator:\\|^User-Agent:")
2502 gnus-sorted-header-list
2503 '("^From:" "^Subject:" "^Summary:" "^Keywords:"
2504 "^Followup-To:" "^To:" "^Cc:" "X-RT-Originator"
2505 "^Newsgroups:" "List-Id:" "^Organization:"
2506 "^User-Agent:" "^Date:")
2507 ;; local-lapsed article dates
2508 ;; from https://www.emacswiki.org/emacs/GnusFormatting#toc11
2509 gnus-article-date-headers '(user-defined)
2510 gnus-article-time-format
2511 (lambda (time)
2512 (let* ((date (format-time-string "%a, %d %b %Y %T %z" time))
2513 (local (article-make-date-line date 'local))
2514 (combined-lapsed (article-make-date-line date
2515 'combined-lapsed))
2516 (lapsed (progn
2517 (string-match " (.+" combined-lapsed)
2518 (match-string 0 combined-lapsed))))
2519 (concat local lapsed))))
2520 (bind-keys
2521 :map gnus-article-mode-map
2522 ("M-L" . org-store-link)))
2523
2524(use-feature gnus-sum
2525 :bind (:map gnus-summary-mode-map
2526 :prefix-map a/gnus-summary-prefix-map
2527 :prefix "v"
2528 ("r" . gnus-summary-reply)
2529 ("w" . gnus-summary-wide-reply)
2530 ("v" . gnus-summary-show-raw-article))
2531 :config
2532 (bind-keys
2533 :map gnus-summary-mode-map
2534 ("M-L" . org-store-link))
2535 :hook (gnus-summary-mode . a/no-mouse-autoselect-window))
2536
2537(use-feature gnus-msg
2538 :config
2539 (setq gnus-posting-styles
2540 '((".*"
2541 (address "amin@bndl.org")
2542 (body "\nBest,\n")
2543 (eval (setq a/message-cite-say-hi t)))
2544 ("gnu.*"
2545 (address "bandali@gnu.org")
2546 (eval (set (make-local-variable 'message-user-fqdn) "fencepost.gnu.org")))
2547 ((header "subject" "ThankCRM")
2548 (to "webmasters-comment@gnu.org")
2549 (body "Added to 2019supporters.html.\n\nMoving to campaigns.\n\n-amin\n")
2550 (eval (setq a/message-cite-say-hi nil)))
2551 ("nnimap\\+uwaterloo:.*"
2552 (address "abandali@uwaterloo.ca")
2553 (gcc "\"nnimap+uwaterloo:Sent Items\""))
2554 ("nnimap\\+csclub:.*"
2555 (address "abandali@csclub.uwaterloo.ca")
2556 (gcc "nnimap+csclub:Sent")))))
2557
2558(use-feature gnus-topic
2559 :hook (gnus-group-mode . gnus-topic-mode)
2560 :config (setq gnus-topic-line-format "%i[ %A: %(%{%n%}%) ]%v\n"))
2561
2562(use-feature gnus-agent
2563 :config
2564 (setq gnus-agent-synchronize-flags 'ask)
2565 :hook (gnus-group-mode . gnus-agent-mode))
2566
2567(use-feature gnus-group
2568 :config
2569 (setq gnus-permanently-visible-groups "\\(:INBOX$\\|:gnu$\\)"))
2570
2571(use-feature mm-decode
2572 :config
2573 (setq mm-discouraged-alternatives '("text/html" "text/richtext")))
2574#+end_src
2575
2576** sendmail
2577
2578#+begin_src emacs-lisp
2579(use-feature sendmail
2580 :config
2581 (setq sendmail-program "/usr/bin/msmtp"
2582 ;; message-sendmail-extra-arguments '("-v" "-d")
2583 mail-specify-envelope-from t
2584 mail-envelope-from 'header))
2585#+end_src
2586
2587** message
2588
2589#+begin_src emacs-lisp
2590(use-feature message
2591 :config
2592 ;; redefine for a simplified In-Reply-To header
2593 ;; (see https://todo.sr.ht/~sircmpwn/lists.sr.ht/67)
2594 (defun message-make-in-reply-to ()
2595 "Return the In-Reply-To header for this message."
2596 (when message-reply-headers
2597 (let ((from (mail-header-from message-reply-headers))
2598 (msg-id (mail-header-id message-reply-headers)))
2599 (when from
2600 msg-id))))
2601
2602 (defconst a/message-cite-style-format "On %Y-%m-%d %l:%M %p, %N wrote:")
2603 (defconst message-cite-style-bandali
2604 '((message-cite-function 'message-cite-original)
2605 (message-citation-line-function 'message-insert-formatted-citation-line)
2606 (message-cite-reply-position 'traditional)
2607 (message-yank-prefix "> ")
2608 (message-yank-cited-prefix ">")
2609 (message-yank-empty-prefix ">")
2610 (message-citation-line-format
2611 (if a/message-cite-say-hi
2612 (concat "Hi %F,\n\n" a/message-cite-style-format)
2613 a/message-cite-style-format)))
2614 "Citation style based on Mozilla Thunderbird's. Use with message-cite-style.")
2615 (setq ;; message-cite-style 'message-cite-style-bandali
2616 message-kill-buffer-on-exit t
2617 message-send-mail-function 'message-send-mail-with-sendmail
2618 message-sendmail-envelope-from 'header
2619 message-subscribed-address-functions
2620 '(gnus-find-subscribed-addresses)
2621 message-dont-reply-to-names
2622 "\\(\\(amin@bndl\\.org\\)\\|\\(.*@\\(aminb\\|amin\\.bndl\\)\\.org\\)\\|\\(\\(bandali\\|aminb?\\|mab\\)@gnu\\.org\\)\\|\\(a\\(min\\.\\)?bandali@uwaterloo\\.ca\\)\\|\\(abandali@csclub\\.uwaterloo\\.ca\\)\\)")
2623 (require 'company-ebdb)
2624 :hook (;; (message-setup . mml-secure-message-sign-pgpmime)
2625 (message-mode . flyspell-mode)
2626 (message-mode . (lambda ()
2627 ;; (setq fill-column 65
2628 ;; message-fill-column 65)
2629 (make-local-variable 'company-idle-delay)
2630 (setq company-idle-delay 0.2))))
2631 ;; :custom-face
2632 ;; (message-header-subject ((t (:foreground "#111" :weight semi-bold))))
2633 ;; (message-header-to ((t (:foreground "#111" :weight normal))))
2634 ;; (message-header-cc ((t (:foreground "#333" :weight normal))))
2635 )
2636
2637(with-eval-after-load 'mml-sec
2638 (setq mml-secure-openpgp-encrypt-to-self t
2639 mml-secure-openpgp-sign-with-sender t))
2640#+end_src
2641
2642** footnote
2643
2644Convenient footnotes in =message-mode=.
2645
2646#+begin_src emacs-lisp
2647(use-feature footnote
2648 :after message
2649 ;; :config
2650 ;; (setq footnote-start-tag ""
2651 ;; footnote-end-tag ""
2652 ;; footnote-style 'unicode)
2653 :bind
2654 (:map message-mode-map
2655 :prefix-map a/footnote-prefix-map
2656 :prefix "C-c f"
2657 ("a" . footnote-add-footnote)
2658 ("b" . footnote-back-to-message)
2659 ("c" . footnote-cycle-style)
2660 ("d" . footnote-delete-footnote)
2661 ("g" . footnote-goto-footnote)
2662 ("r" . footnote-renumber-footnotes)
2663 ("s" . footnote-set-style)))
2664#+end_src
2665
2666** ebdb
2667
2668#+begin_src emacs-lisp
2669(use-package ebdb
2670 :straight (:host github :repo "girzel/ebdb")
2671 :after gnus
2672 :bind (:map gnus-group-mode-map ("e" . ebdb))
2673 :config
2674 (setq ebdb-sources (no-littering-expand-var-file-name "ebdb"))
2675 (with-eval-after-load 'swiper
2676 (add-to-list 'swiper-font-lock-exclude 'ebdb-mode t)))
2677
2678(use-feature ebdb-com
2679 :after ebdb)
2680
2681;; (use-package ebdb-complete
2682;; :after ebdb
2683;; :config
2684;; (ebdb-complete-enable))
2685
2686(use-package company-ebdb
2687 :config
2688 (defun company-ebdb--post-complete (_) nil))
2689
2690(use-feature ebdb-gnus
2691 :after ebdb
2692 :custom
2693 (ebdb-gnus-window-configuration
2694 '(article
2695 (vertical 1.0
2696 (summary 0.25 point)
2697 (horizontal 1.0
2698 (article 1.0)
2699 (ebdb-gnus 0.3))))))
2700
2701(use-feature ebdb-mua
2702 :after ebdb
2703 ;; :custom (ebdb-mua-pop-up nil)
2704 )
2705
2706;; (use-package ebdb-message
2707;; :after ebdb)
2708
2709
2710;; (use-package ebdb-vcard
2711;; :after ebdb)
2712#+end_src
2713
2714** message-x
2715
2716#+begin_src emacs-lisp
2717(use-package message-x)
2718#+end_src
2719
2720#+begin_src emacs-lisp :tangle no
2721(use-package message-x
2722 :custom
2723 (message-x-completion-alist
2724 (quote
2725 (("\\([rR]esent-\\|[rR]eply-\\)?[tT]o:\\|[bB]?[cC][cC]:" . gnus-harvest-find-address)
2726 ((if
2727 (boundp
2728 (quote message-newgroups-header-regexp))
2729 message-newgroups-header-regexp message-newsgroups-header-regexp)
2730 . message-expand-group)))))
2731#+end_src
2732
2733** COMMENT gnus-harvest
2734
2735#+begin_src emacs-lisp
2736(use-package gnus-harvest
2737 :commands gnus-harvest-install
2738 :demand t
2739 :config
2740 (if (featurep 'message-x)
2741 (gnus-harvest-install 'message-x)
2742 (gnus-harvest-install)))
2743#+end_src
2744
2745* IRC
2746:PROPERTIES:
2747:CUSTOM_ID: irc
2748:END:
2749
2750Now with ERC!
2751
2752#+begin_src emacs-lisp
2753(use-package znc
2754 :straight (:host nil :repo "https://git.bndl.org/amin/znc.el")
2755 :bind (("C-c a e e" . znc-erc)
2756 ("C-c a e a" . znc-all))
2757 :config
2758 (let ((pwd (let ((auth (auth-source-search :host "znca")))
2759 (cond
2760 ((null auth) (error "Couldn't find znca's authinfo"))
2761 (t (funcall (plist-get (car auth) :secret)))))))
2762 (setq znc-servers
2763 `(("znc.bndl.org" 1337 t
2764 ((freenode "amin/freenode" ,pwd)))
2765 ("znc.bndl.org" 1337 t
2766 ((moznet "amin/moznet" ,pwd)))))))
2767#+end_src
2768
2769* Post initialization
2770:PROPERTIES:
2771:CUSTOM_ID: post-initialization
2772:END:
2773
2774Display how long it took to load the init file.
2775
2776#+begin_src emacs-lisp
2777(message "Loading %s...done (%.3fs)" user-init-file
2778 (float-time (time-subtract (current-time)
2779 a/before-user-init-time)))
2780#+end_src
2781
2782* Footer
2783:PROPERTIES:
2784:CUSTOM_ID: footer
2785:END:
2786
2787#+begin_src emacs-lisp :comments none
2788;;; init.el ends here
2789#+end_src
2790
2791* COMMENT Local Variables :ARCHIVE:
2792# Local Variables:
2793# eval: (add-hook 'after-save-hook #'a/async-babel-tangle 'append 'local)
2794# eval: (when (featurep 'typo (typo-mode -1)))
2795# End: