[emacs] remove evil and go back to vanilla emacs bindings
[~bandali/configs] / init.org
... / ...
CommitLineData
1#+title: =aminb='s Literate Emacs Configuration
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/emacscollective/borg][Borg]]. Over the years, I've taken inspiration from configurations of
14many different people. Some of the configurations that I can remember
15off 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 Borg. For what it's worth, I briefly
32experimented with [[https://github.com/raxod502/straight.el][straight.el]], but found that it added about 2 seconds
33to my init time; which is unacceptable for me: I use Emacs as my
34window manager (via EXWM) and coming from bspwm, I'm too used to
35having fast startup times.
36
37** Installation
38
39To use this config for your Emacs, first you need to clone this repo,
40then bootstrap Borg, tell Borg to retrieve package submodules, and
41byte-compiled the packages. Something along these lines should work:
42
43#+begin_src sh :tangle no
44git clone https://github.com/aminb/dotfiles ~/.emacs.d
45cd ~/.emacs.d
46make bootstrap-borg
47make bootstrap
48make build
49#+end_src
50
51* Contents :toc_1:noexport:
52
53- [[#about][About]]
54- [[#header][Header]]
55- [[#initial-setup][Initial setup]]
56- [[#core][Core]]
57- [[#post-initialization][Post initialization]]
58- [[#footer][Footer]]
59
60* Header
61:PROPERTIES:
62:CUSTOM_ID: header
63:END:
64
65** First line
66
67#+begin_src emacs-lisp :comments none
68;;; init.el --- Amin Bandali's Emacs config -*- lexical-binding: t ; eval: (view-mode 1)-*-
69#+end_src
70
71Enable =view-mode=, which both makes the file read-only (as a reminder
72that =init.el= is an auto-generated file, not supposed to be edited),
73and provides some convenient key bindings for browsing through the
74file.
75
76** License
77
78#+begin_src emacs-lisp :comments none
79;; Copyright (C) 2018 Amin Bandali <bandali@gnu.org>
80
81;; This program is free software: you can redistribute it and/or modify
82;; it under the terms of the GNU General Public License as published by
83;; the Free Software Foundation, either version 3 of the License, or
84;; (at your option) any later version.
85
86;; This program is distributed in the hope that it will be useful,
87;; but WITHOUT ANY WARRANTY; without even the implied warranty of
88;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
89;; GNU General Public License for more details.
90
91;; You should have received a copy of the GNU General Public License
92;; along with this program. If not, see <https://www.gnu.org/licenses/>.
93#+end_src
94
95** Commentary
96
97#+begin_src emacs-lisp :comments none
98;;; Commentary:
99
100;; Emacs configuration of Amin Bandali, computer scientist and functional
101;; programmer.
102
103;; THIS FILE IS AUTO-GENERATED FROM `init.org'.
104#+end_src
105
106** Naming conventions
107
108The conventions below were inspired by [[https://github.com/hlissner/doom-emacs][Doom]]'s, found [[https://github.com/hlissner/doom-emacs/blob/5dacbb7cb1c6ac246a9ccd15e6c4290def67757c/core/core.el#L3-L17][here]].
109
110#+begin_src emacs-lisp :comments none
111;; Naming conventions:
112;;
113;; amin-... public variables or non-interactive functions
114;; amin--... private anything (non-interactive), not safe for direct use
115;; amin/... an interactive function; safe for M-x or keybinding
116;; amin:... an evil operator, motion, or command
117;; amin|... a hook function
118;; amin*... an advising function
119;; amin@... a hydra command
120;; ...! a macro
121#+end_src
122
123* Initial setup
124:PROPERTIES:
125:CUSTOM_ID: initial-setup
126:END:
127
128#+begin_src emacs-lisp :comments none
129;;; Code:
130#+end_src
131
132** Emacs initialization
133
134I'd like to do a couple of measurements of Emacs' startup time. First,
135let's see how long Emacs takes to start up, before even loading
136=init.el=, i.e. =user-init-file=:
137
138#+begin_src emacs-lisp
139(defvar amin--before-user-init-time (current-time)
140 "Value of `current-time' when Emacs begins loading `user-init-file'.")
141(message "Loading Emacs...done (%.3fs)"
142 (float-time (time-subtract amin--before-user-init-time
143 before-init-time)))
144#+end_src
145
146Also, temporarily increase ~gc-cons-threshhold~ and
147~gc-cons-percentage~ during startup to reduce garbage collection
148frequency. Clearing the ~file-name-handler-alist~ seems to help reduce
149startup time as well.
150
151#+begin_src emacs-lisp
152(defvar amin--gc-cons-threshold gc-cons-threshold)
153(defvar amin--gc-cons-percentage gc-cons-percentage)
154(defvar amin--file-name-handler-alist file-name-handler-alist)
155(setq gc-cons-threshold (* 400 1024 1024) ; 400 MiB
156 gc-cons-percentage 0.6
157 file-name-handler-alist nil
158 ;; sidesteps a bug when profiling with esup
159 esup-child-profile-require-level 0)
160#+end_src
161
162Of course, we'd like to set them back to their defaults once we're
163done initializing.
164
165#+begin_src emacs-lisp
166(add-hook
167 'after-init-hook
168 (lambda ()
169 (setq gc-cons-threshold amin--gc-cons-threshold
170 gc-cons-percentage amin--gc-cons-percentage
171 file-name-handler-alist amin--file-name-handler-alist)))
172#+end_src
173
174Increase the number of lines kept in message logs (the =*Messages*=
175buffer).
176
177#+begin_src emacs-lisp
178(setq message-log-max 20000)
179#+end_src
180
181Optionally, we could suppress some byte compiler warnings like below,
182but for now I've decided to keep them enabled. See documentation for
183~byte-compile-warnings~ for more details.
184
185#+begin_src emacs-lisp
186;; (setq byte-compile-warnings
187;; '(not free-vars unresolved noruntime lexical make-local))
188#+end_src
189
190** whoami
191
192#+begin_src emacs-lisp
193(setq user-full-name "Amin Bandali"
194 user-mail-address "amin@aminb.org")
195#+end_src
196
197** Package management
198
199*** No =package.el=
200
201I can do all my package management things with Borg, and don't need
202Emacs' built-in =package.el=. Emacs 27 lets us disable =package.el= in
203the =early-init-file= (see [[https://git.savannah.gnu.org/cgit/emacs.git/commit/?id=24acb31c04b4048b85311d794e600ecd7ce60d3b][here]]).
204
205#+begin_src emacs-lisp :tangle early-init.el
206(setq package-enable-at-startup nil)
207#+end_src
208
209But since Emacs 27 isn't out yet (Emacs 26 is just around the corner
210right now), and even when released it'll be long before most distros
211ship in their repos, I'll still put the old workaround with the
212commented call to ~package-initialize~ here anyway.
213
214#+begin_src emacs-lisp
215(setq package-enable-at-startup nil)
216;; (package-initialize)
217#+end_src
218
219*** Borg
220
221#+begin_quote
222Assimilate Emacs packages as Git submodules
223#+end_quote
224
225[[https://github.com/emacscollective/borg][Borg]] is at the heart of package management of my Emacs setup. In
226short, it creates a git submodule in =lib/= for each package, which
227can then be managed with the help of Magit or other tools.
228
229#+begin_src emacs-lisp
230(setq user-init-file (or load-file-name buffer-file-name)
231 user-emacs-directory (file-name-directory user-init-file))
232(add-to-list 'load-path
233 (expand-file-name "lib/borg" user-emacs-directory))
234(require 'borg)
235(borg-initialize)
236
237;; (require 'borg-nix-shell)
238;; (setq borg-build-shell-command 'borg-nix-shell-build-command)
239
240(with-eval-after-load 'bind-key
241 (bind-keys
242 :package borg
243 ("C-c b A" . borg-activate)
244 ("C-c b a" . borg-assimilate)
245 ("C-c b b" . borg-build)
246 ("C-c b c" . borg-clone)
247 ("C-c b r" . borg-remove)))
248#+end_src
249
250*** =use-package=
251
252#+begin_quote
253A use-package declaration for simplifying your .emacs
254#+end_quote
255
256[[https://github.com/jwiegley/use-package][use-package]] is an awesome utility for managing and configuring
257packages (in our case especially the latter) in a neatly organized way
258and without compromising on performance.
259
260#+begin_src emacs-lisp
261(require 'use-package)
262(if nil ; set to t when need to debug init
263 (setq use-package-verbose t
264 use-package-expand-minimally nil
265 use-package-compute-statistics t
266 debug-on-error t)
267 (setq use-package-verbose nil
268 use-package-expand-minimally t))
269#+end_src
270
271*** Epkg
272
273#+begin_quote
274Browse the Emacsmirror package database
275#+end_quote
276
277Epkg provides access to a local copy of the [[https://emacsmirror.net][Emacsmirror]] package
278database, low-level functions for querying the database, and a
279=package.el=-like user interface for browsing the available packages.
280
281#+begin_src emacs-lisp
282(use-package epkg
283 :defer t
284 :bind
285 (("C-c b d" . epkg-describe-package)
286 ("C-c b p" . epkg-list-packages)
287 ("C-c b u" . epkg-update)))
288#+end_src
289
290** No littering in =~/.emacs.d=
291
292#+begin_quote
293Help keeping ~/.emacs.d clean
294#+end_quote
295
296By default, even for Emacs' built-in packages, the configuration files
297and persistent data are all over the place. Use =no-littering= to help
298contain the mess.
299
300#+begin_src emacs-lisp
301(use-package no-littering
302 :demand t
303 :config
304 (savehist-mode 1)
305 (add-to-list 'savehist-additional-variables 'kill-ring)
306 (save-place-mode 1)
307 (setq auto-save-file-name-transforms
308 `((".*" ,(no-littering-expand-var-file-name "auto-save/") t))))
309#+end_src
310
311** Custom file (=custom.el=)
312
313I'm not planning on using the custom file much, but even so, I
314definitely don't want it mixing with =init.el=. So, here; let's give
315it it's own file. While at it, treat themes as safe.
316
317#+begin_src emacs-lisp
318(use-package custom
319 :no-require t
320 :config
321 (setq custom-file (no-littering-expand-etc-file-name "custom.el"))
322 (when (file-exists-p custom-file)
323 (load custom-file))
324 (setf custom-safe-themes t))
325#+end_src
326
327** Secrets file
328
329Load the secrets file if it exists, otherwise show a warning.
330
331#+begin_src emacs-lisp
332(with-demoted-errors
333 (load (no-littering-expand-etc-file-name "secrets")))
334#+end_src
335
336** Better =$PATH= handling
337
338Let's use [[https://github.com/purcell/exec-path-from-shell][exec-path-from-shell]] to make Emacs use the =$PATH= as set up
339in my shell.
340
341#+begin_src emacs-lisp
342(use-package exec-path-from-shell
343 :defer 1
344 :init
345 (setq exec-path-from-shell-check-startup-files nil)
346 :config
347 (exec-path-from-shell-initialize)
348 ;; while we're at it, let's fix access to our running ssh-agent
349 (exec-path-from-shell-copy-env "SSH_AGENT_PID")
350 (exec-path-from-shell-copy-env "SSH_AUTH_SOCK"))
351#+end_src
352
353** COMMENT Only one custom theme at a time
354
355#+begin_src emacs-lisp
356(defadvice load-theme (before clear-previous-themes activate)
357 "Clear existing theme settings instead of layering them"
358 (mapc #'disable-theme custom-enabled-themes))
359#+end_src
360
361** Server
362
363Start server if not already running. Alternatively, can be done by
364issuing =emacs --daemon= in the terminal, which can be automated with
365a systemd service or using =brew services start emacs= on macOS. I use
366Emacs as my window manager (via EXWM), so I always start Emacs on
367login; so starting the server from inside Emacs is good enough for me.
368
369See [[https://www.gnu.org/software/emacs/manual/html_node/emacs/Emacs-Server.html#Emacs-Server][Using Emacs as a Server]].
370
371#+begin_src emacs-lisp
372(use-package server
373 :defer 1
374 :config (or (server-running-p) (server-mode)))
375#+end_src
376
377** COMMENT Unicode support
378
379Font stack with better unicode support, around =Ubuntu Mono= and
380=Hack=.
381
382#+begin_src emacs-lisp :tangle no
383(dolist (ft (fontset-list))
384 (set-fontset-font
385 ft
386 'unicode
387 (font-spec :name "Source Code Pro" :size 14))
388 (set-fontset-font
389 ft
390 'unicode
391 (font-spec :name "DejaVu Sans Mono")
392 nil
393 'append)
394 ;; (set-fontset-font
395 ;; ft
396 ;; 'unicode
397 ;; (font-spec
398 ;; :name "Symbola monospacified for DejaVu Sans Mono")
399 ;; nil
400 ;; 'append)
401 ;; (set-fontset-font
402 ;; ft
403 ;; #x2115 ; ℕ
404 ;; (font-spec :name "DejaVu Sans Mono")
405 ;; nil
406 ;; 'append)
407 (set-fontset-font
408 ft
409 (cons ?Α ?ω)
410 (font-spec :name "DejaVu Sans Mono" :size 14)
411 nil
412 'prepend))
413#+end_src
414
415** Gentler font resizing
416
417#+begin_src emacs-lisp
418(setq text-scale-mode-step 1.05)
419#+end_src
420
421** Focus follows mouse
422
423I’d like focus to follow the mouse when I move the cursor from one
424window to the next.
425
426#+begin_src emacs-lisp
427(setq mouse-autoselect-window t)
428#+end_src
429
430Let’s define a function to conveniently disable this for certain
431buffers and/or modes.
432
433#+begin_src emacs-lisp
434(defun amin--no-mouse-autoselect-window ()
435 (make-local-variable 'mouse-autoselect-window)
436 (setq mouse-autoselect-window nil))
437#+end_src
438
439** Libraries
440
441#+begin_src emacs-lisp
442(require 'cl-lib)
443(require 'subr-x)
444#+end_src
445
446** Useful utilities
447
448#+begin_src emacs-lisp
449(defun amin-enlist (exp)
450 "Return EXP wrapped in a list, or as-is if already a list."
451(if (listp exp) exp (list exp)))
452
453; from https://github.com/hlissner/doom-emacs/commit/589108fdb270f24a98ba6209f6955fe41530b3ef
454(defmacro after! (features &rest body)
455 "A smart wrapper around `with-eval-after-load'. Supresses warnings during
456compilation."
457 (declare (indent defun) (debug t))
458 (list (if (or (not (bound-and-true-p byte-compile-current-file))
459 (dolist (next (amin-enlist features))
460 (if (symbolp next)
461 (require next nil :no-error)
462 (load next :no-message :no-error))))
463 #'progn
464 #'with-no-warnings)
465 (cond ((symbolp features)
466 `(eval-after-load ',features '(progn ,@body)))
467 ((and (consp features)
468 (memq (car features) '(:or :any)))
469 `(progn
470 ,@(cl-loop for next in (cdr features)
471 collect `(after! ,next ,@body))))
472 ((and (consp features)
473 (memq (car features) '(:and :all)))
474 (dolist (next (cdr features))
475 (setq body `(after! ,next ,@body)))
476 body)
477 ((listp features)
478 `(after! (:all ,@features) ,@body)))))
479#+end_src
480
481Convenience macro for =setq='ing multiple variables to the same value:
482
483#+begin_src emacs-lisp
484(defmacro setq-every! (value &rest vars)
485 "Set all the variables from VARS to value VALUE."
486 (declare (indent defun) (debug t))
487 `(progn ,@(mapcar (lambda (x) (list 'setq x value)) vars)))
488#+end_src
489
490* Core
491:PROPERTIES:
492:CUSTOM_ID: core
493:END:
494
495** Defaults
496
497*** Time and battery in mode-line
498
499Enable displaying time and battery in the mode-line, since I'm not
500using the Xfce panel anymore. Also, I don't need to see the load
501average on a regular basis, so disable that.
502
503Note: using =i3status= on sway at the moment, so disabling this.
504
505#+begin_src emacs-lisp :tangle no
506(use-package time
507 :init
508 (setq display-time-default-load-average nil)
509 :config
510 (display-time-mode))
511
512(use-package battery
513 :config
514 (display-battery-mode))
515#+end_src
516
517*** Smaller fringe
518
519Might want to set the fringe to a smaller value, especially if using
520EXWM. I'm fine with the default for now.
521
522#+begin_src emacs-lisp
523;; (fringe-mode '(3 . 1))
524(fringe-mode nil)
525#+end_src
526
527*** Disable disabled commands
528
529Emacs disables some commands by default that could persumably be
530confusing for novice users. Let's disable that.
531
532#+begin_src emacs-lisp
533(setq disabled-command-function nil)
534#+end_src
535
536*** Kill-ring
537
538Save what I copy into clipboard from other applications into Emacs'
539kill-ring, which would allow me to still be able to easily access it
540in case I kill (cut or copy) something else inside Emacs before
541yanking (pasting) what I'd originally intended to.
542
543#+begin_src emacs-lisp
544(setq save-interprogram-paste-before-kill t)
545#+end_src
546
547*** Minibuffer
548
549#+begin_src emacs-lisp
550(setq enable-recursive-minibuffers t
551 resize-mini-windows t)
552#+end_src
553
554*** Lazy-person-friendly yes/no prompts
555
556Lazy people would prefer to type fewer keystrokes, especially for yes
557or no questions. I'm lazy.
558
559#+begin_src emacs-lisp
560(defalias 'yes-or-no-p #'y-or-n-p)
561#+end_src
562
563*** Startup screen and =*scratch*=
564
565Firstly, let Emacs know that I'd like to have =*scratch*= as my
566startup buffer.
567
568#+begin_src emacs-lisp
569(setq initial-buffer-choice t)
570#+end_src
571
572Now let's customize the =*scratch*= buffer a bit. First off, I don't
573need the default hint.
574
575#+begin_src emacs-lisp
576(setq initial-scratch-message nil)
577#+end_src
578
579Also, let's use Text mode as the major mode, in case I want to
580customize it (=*scratch*='s default major mode, Fundamental mode,
581can't really be customized).
582
583#+begin_src emacs-lisp
584(setq initial-major-mode 'text-mode)
585#+end_src
586
587Inhibit the buffer list when more than 2 files are loaded.
588
589#+begin_src emacs-lisp
590(setq inhibit-startup-buffer-menu t)
591#+end_src
592
593I don't really need to see the startup screen or echo area message
594either.
595
596#+begin_src emacs-lisp
597(advice-add #'display-startup-echo-area-message :override #'ignore)
598(setq inhibit-startup-screen t
599 inhibit-startup-echo-area-message user-login-name)
600#+end_src
601
602*** More useful frame titles
603
604Show either the file name or the buffer name (in case the buffer isn't
605visiting a file). Borrowed from Emacs Prelude.
606
607#+begin_src emacs-lisp
608(setq frame-title-format
609 '("" invocation-name " - "
610 (:eval (if (buffer-file-name)
611 (abbreviate-file-name (buffer-file-name))
612 "%b"))))
613#+end_src
614
615*** Backups
616
617Emacs' default backup settings aren't that great. Let's use more
618sensible options. See documentation for the ~make-backup-file~
619variable.
620
621#+begin_src emacs-lisp
622(setq backup-by-copying t
623 version-control t
624 delete-old-versions t)
625#+end_src
626
627*** Auto revert
628
629Enable automatic reloading of changed buffers and files.
630
631#+begin_src emacs-lisp
632(global-auto-revert-mode 1)
633(setq auto-revert-verbose nil
634 global-auto-revert-non-file-buffers nil)
635#+end_src
636
637*** Always use space for indentation
638
639#+begin_src emacs-lisp
640(setq-default
641 indent-tabs-mode nil
642 require-final-newline t
643 tab-width 4)
644#+end_src
645
646*** Winner mode
647
648Enable =winner-mode=.
649
650#+begin_src emacs-lisp
651(winner-mode 1)
652#+end_src
653
654*** Close =*compilation*= on success
655
656#+begin_src emacs-lisp
657(setq compilation-exit-message-function
658 (lambda (status code msg)
659 "Close the compilation window if successful."
660 ;; if M-x compile exits with 0
661 (when (and (eq status 'exit) (zerop code))
662 (bury-buffer)
663 (delete-window (get-buffer-window (get-buffer "*compilation*"))))
664 ;; return the result of compilation-exit-message-function
665 (cons msg code)))
666#+end_src
667
668*** Search for non-ASCII characters
669
670I’d like non-ASCII characters such as ‘’“”«»‹›áⓐ𝒶 to be selected when
671I search for their ASCII counterpart. Shoutout to [[http://endlessparentheses.com/new-in-emacs-25-1-easily-search-non-ascii-characters.html][endlessparentheses]]
672for this.
673
674#+begin_src emacs-lisp
675(setq search-default-mode #'char-fold-to-regexp)
676
677;; uncomment to extend this behaviour to query-replace
678;; (setq replace-char-fold t)
679#+end_src
680
681** Bindings
682
683#+begin_src emacs-lisp
684(bind-keys
685 ("C-c a i" . ielm)
686
687 ("C-c e b" . eval-buffer)
688 ("C-c e r" . eval-region)
689
690 ("C-c F m" . make-frame-command)
691 ("C-c F d" . delete-frame)
692 ("C-c F D" . delete-other-frames)
693
694 ("C-c o" . other-window)
695
696 ("C-c Q" . save-buffers-kill-terminal)
697
698 ("C-S-h C" . describe-char)
699 ("C-S-h F" . describe-face)
700
701 ("C-x K" . kill-this-buffer)
702
703 ("s-p" . beginning-of-buffer)
704 ("s-n" . end-of-buffer))
705#+end_src
706
707** Packages
708
709The packages in this section are absolutely essential to my everyday
710workflow, and they play key roles in how I do my computing. They
711immensely enhance the Emacs experience for me; both using Emacs, and
712customizing it.
713
714*** [[https://github.com/emacscollective/auto-compile][auto-compile]]
715
716#+begin_src emacs-lisp
717(use-package auto-compile
718 :demand t
719 :config
720 (auto-compile-on-load-mode)
721 (auto-compile-on-save-mode)
722 (setq auto-compile-display-buffer nil
723 auto-compile-mode-line-counter t
724 auto-compile-source-recreate-deletes-dest t
725 auto-compile-toggle-deletes-nonlib-dest t
726 auto-compile-update-autoloads t)
727 (add-hook 'auto-compile-inhibit-compile-hook
728 'auto-compile-inhibit-compile-detached-git-head))
729#+end_src
730
731*** [[https://orgmode.org/][Org mode]]
732
733#+begin_quote
734Org mode is for keeping notes, maintaining TODO lists, planning
735projects, and authoring documents with a fast and effective plain-text
736system.
737#+end_quote
738
739In short, my favourite way of life.
740
741#+begin_src emacs-lisp
742(use-package org
743 :defer 1
744 :config
745 (setq org-src-tab-acts-natively t
746 org-src-preserve-indentation nil
747 org-edit-src-content-indentation 0
748 org-email-link-description-format "Email %c: %s" ; %.30s
749 org-highlight-latex-and-related '(entities)
750 org-log-done 'time)
751 (add-to-list 'org-structure-template-alist '("L" . "src emacs-lisp") t)
752 (font-lock-add-keywords
753 'org-mode
754 '(("[ \t]*\\(#\\+\\(BEGIN\\|END\\|begin\\|end\\)_\\(\\S-+\\)\\)[ \t]*\\([^\n:]*\\)"
755 (1 '(:foreground "#5a5b5a" :background "#292b2b") t) ; directive
756 (3 '(:foreground "#81a2be" :background "#292b2b") t) ; kind
757 (4 '(:foreground "#c5c8c6") t))) ; title
758 t)
759 :bind (:map org-mode-map ("M-L" . org-insert-last-stored-link))
760 :hook ((org-mode . org-indent-mode)
761 (org-mode . auto-fill-mode)
762 (org-mode . flyspell-mode))
763 :custom
764 (org-latex-packages-alist '(("" "listings") ("" "color")))
765 :custom-face
766 '(org-block-begin-line ((t (:foreground "#5a5b5a" :background "#1d1f21"))))
767 '(org-block ((t (:background "#1d1f21"))))
768 '(org-latex-and-related ((t (:foreground "#b294bb")))))
769
770(use-package ox-latex
771 :after ox
772 :config
773 (setq org-latex-listings 'listings
774 ;; org-latex-prefer-user-labels t
775 )
776 (add-to-list 'org-latex-packages-alist '("" "listings"))
777 (add-to-list 'org-latex-packages-alist '("" "color"))
778 (add-to-list 'org-latex-classes
779 '("IEEEtran" "\\documentclass[11pt]{IEEEtran}"
780 ("\\section{%s}" . "\\section*{%s}")
781 ("\\subsection{%s}" . "\\subsection*{%s}")
782 ("\\subsubsection{%s}" . "\\subsubsection*{%s}")
783 ("\\paragraph{%s}" . "\\paragraph*{%s}")
784 ("\\subparagraph{%s}" . "\\subparagraph*{%s}"))
785 t))
786
787(use-package ox-beamer
788 :after ox)
789
790(use-package orgalist
791 :after message
792 :hook (message-mode . orgalist-mode))
793#+end_src
794
795**** asynchronous tangle
796
797=amin/async-babel-tangle= is a function closely inspired by [[https://github.com/dieggsy/dotfiles/tree/cc10edf7701958eff1cd94d4081da544d882a28c/emacs.d#dotfiles][dieggsy's
798d/async-babel-tangle]] which uses [[https://github.com/jwiegley/emacs-async][async]] to asynchronously tangle an org
799file.
800
801#+begin_src emacs-lisp
802(after! org
803 (defvar amin-show-async-tangle-results nil
804 "Keep *emacs* async buffers around for later inspection.")
805
806 (defvar amin-show-async-tangle-time nil
807 "Show the time spent tangling the file.")
808
809 (defvar amin-async-tangle-post-compile "make ti"
810 "If non-nil, pass to `compile' after successful tangle.")
811
812 (defun amin/async-babel-tangle ()
813 "Tangle org file asynchronously."
814 (interactive)
815 (let* ((file-tangle-start-time (current-time))
816 (file (buffer-file-name))
817 (file-nodir (file-name-nondirectory file))
818 (async-quiet-switch "-q"))
819 (async-start
820 `(lambda ()
821 (require 'org)
822 (org-babel-tangle-file ,file))
823 (unless amin-show-async-tangle-results
824 `(lambda (result)
825 (if result
826 (progn
827 (message "Tangled %s%s"
828 ,file-nodir
829 (if amin-show-async-tangle-time
830 (format " (%.3fs)"
831 (float-time (time-subtract (current-time)
832 ',file-tangle-start-time)))
833 ""))
834 (when amin-async-tangle-post-compile
835 (compile amin-async-tangle-post-compile)))
836 (message "Tangling %s failed" ,file-nodir))))))))
837
838(add-to-list
839 'safe-local-variable-values
840 '(eval add-hook 'after-save-hook #'amin/async-babel-tangle 'append 'local))
841#+end_src
842
843*** [[https://magit.vc/][Magit]]
844
845#+begin_quote
846It's Magit! A Git porcelain inside Emacs.
847#+end_quote
848
849Not just how I do git, but /the/ way to do git.
850
851#+begin_src emacs-lisp
852(use-package magit
853 :defer 1
854 :bind (("C-x g" . magit-status)
855 ("s-g s" . magit-status)
856 ("s-g l" . magit-log-buffer-file))
857 :config
858 (magit-add-section-hook 'magit-status-sections-hook
859 'magit-insert-modules
860 'magit-insert-stashes
861 'append)
862 (setq
863 magit-repository-directories '(("~/.emacs.d/" . 0)
864 ("~/src/git/" . 1)))
865 (nconc magit-section-initial-visibility-alist
866 '(([unpulled status] . show)
867 ([unpushed status] . show)))
868 :custom-face (magit-diff-file-heading ((t (:weight normal)))))
869#+end_src
870
871*** [[https://github.com/abo-abo/swiper][Ivy]] (and friends)
872
873#+begin_quote
874Ivy - a generic completion frontend for Emacs, Swiper - isearch with
875an overview, and more. Oh, man!
876#+end_quote
877
878There's no way I could top that, so I won't attempt to.
879
880**** Ivy
881
882#+begin_src emacs-lisp
883(use-package ivy
884 :defer 1
885 :bind
886 (:map ivy-minibuffer-map
887 ([escape] . keyboard-escape-quit)
888 ([S-up] . ivy-previous-history-element)
889 ([S-down] . ivy-next-history-element)
890 ("DEL" . ivy-backward-delete-char))
891 :config
892 (setq ivy-wrap t)
893 (ivy-mode 1)
894 ;; :custom-face
895 ;; (ivy-minibuffer-match-face-2 ((t (:background "#e99ce8" :weight semi-bold))))
896 ;; (ivy-minibuffer-match-face-3 ((t (:background "#bbbbff" :weight semi-bold))))
897 ;; (ivy-minibuffer-match-face-4 ((t (:background "#ffbbff" :weight semi-bold))))
898)
899#+end_src
900
901**** Swiper
902
903#+begin_src emacs-lisp
904(use-package swiper
905 :bind (("C-s" . swiper)
906 ("C-r" . swiper)))
907#+end_src
908
909**** Counsel
910
911#+begin_src emacs-lisp
912(use-package counsel
913 :defer 1
914 :bind (([remap execute-extended-command] . counsel-M-x)
915 ([remap find-file] . counsel-find-file)
916 ("s-r" . counsel-recentf)
917 ("C-c x" . counsel-M-x)
918 ("C-c f ." . counsel-find-file)
919 :map minibuffer-local-map
920 ("C-r" . counsel-minibuffer-history))
921 :config
922 (counsel-mode 1)
923 (defalias 'locate #'counsel-locate))
924#+end_src
925
926*** eshell
927
928#+begin_src emacs-lisp
929(use-package eshell
930 :defer 1
931 :commands eshell
932 :config
933 (eval-when-compile (defvar eshell-prompt-regexp))
934 (defun amin/eshell-quit-or-delete-char (arg)
935 (interactive "p")
936 (if (and (eolp) (looking-back eshell-prompt-regexp nil))
937 (eshell-life-is-too-much)
938 (delete-char arg)))
939
940 (defun amin/eshell-clear ()
941 (interactive)
942 (let ((inhibit-read-only t))
943 (erase-buffer))
944 (eshell-send-input))
945
946 (defun amin|eshell-setup ()
947 (make-local-variable 'company-idle-delay)
948 (setq company-idle-delay nil)
949 (bind-keys :map eshell-mode-map
950 ("C-d" . amin/eshell-quit-or-delete-char)
951 ("C-S-l" . amin/eshell-clear)
952 ("M-r" . counsel-esh-history)
953 ([tab] . company-complete)))
954
955 :hook (eshell-mode . amin|eshell-setup)
956 :custom
957 (eshell-hist-ignoredups t)
958 (eshell-input-filter 'eshell-input-filter-initial-space))
959#+end_src
960
961*** Ibuffer
962
963#+begin_src emacs-lisp
964(use-package ibuffer
965 :defer t
966 :bind
967 (("C-x C-b" . ibuffer-other-window)
968 :map ibuffer-mode-map
969 ("P" . ibuffer-backward-filter-group)
970 ("N" . ibuffer-forward-filter-group)
971 ("M-p" . ibuffer-do-print)
972 ("M-n" . ibuffer-do-shell-command-pipe-replace))
973 :config
974 ;; Use human readable Size column instead of original one
975 (define-ibuffer-column size-h
976 (:name "Size" :inline t)
977 (cond
978 ((> (buffer-size) 1000000) (format "%7.1fM" (/ (buffer-size) 1000000.0)))
979 ((> (buffer-size) 100000) (format "%7.0fk" (/ (buffer-size) 1000.0)))
980 ((> (buffer-size) 1000) (format "%7.1fk" (/ (buffer-size) 1000.0)))
981 (t (format "%8d" (buffer-size)))))
982 :custom
983 (ibuffer-saved-filter-groups
984 '(("default"
985 ("dired" (mode . dired-mode))
986 ("org" (mode . org-mode))
987 ("web"
988 (or
989 (mode . web-mode)
990 (mode . css-mode)
991 (mode . scss-mode)
992 (mode . js2-mode)))
993 ("shell"
994 (or
995 (mode . eshell-mode)
996 (mode . shell-mode)))
997 ("notmuch" (name . "\*notmuch\*"))
998 ("programming"
999 (or
1000 (mode . python-mode)
1001 (mode . c++-mode)
1002 (mode . emacs-lisp-mode)))
1003 ("emacs"
1004 (or
1005 (name . "^\\*scratch\\*$")
1006 (name . "^\\*Messages\\*$")))
1007 ("slack"
1008 (or
1009 (name . "^\\*Slack*"))))))
1010 (ibuffer-formats
1011 '((mark modified read-only locked " "
1012 (name 18 18 :left :elide)
1013 " "
1014 (size-h 9 -1 :right)
1015 " "
1016 (mode 16 16 :left :elide)
1017 " " filename-and-process)
1018 (mark " "
1019 (name 16 -1)
1020 " " filename)))
1021 :hook (ibuffer . (lambda () (ibuffer-switch-to-saved-filter-groups "default"))))
1022#+end_src
1023
1024*** Outline
1025
1026#+begin_src emacs-lisp
1027(use-package outline
1028 :defer t
1029 :hook (prog-mode . outline-minor-mode)
1030 :bind
1031 (:map
1032 outline-minor-mode-map
1033 ("<s-tab>" . outline-toggle-children)
1034 ("M-p" . outline-previous-visible-heading)
1035 ("M-n" . outline-next-visible-heading)
1036 :prefix-map amin--outline-prefix-map
1037 :prefix "s-o"
1038 ("TAB" . outline-toggle-children)
1039 ("a" . outline-hide-body)
1040 ("H" . outline-hide-body)
1041 ("S" . outline-show-all)
1042 ("h" . outline-hide-subtree)
1043 ("s" . outline-show-subtree)))
1044#+end_src
1045
1046* Borg's =layer/essentials=
1047
1048TODO: break this giant source block down into individual org sections.
1049
1050#+begin_src emacs-lisp
1051(use-package dash
1052 :config (dash-enable-font-lock))
1053
1054(use-package diff-hl
1055 :config
1056 (setq diff-hl-draw-borders nil)
1057 (global-diff-hl-mode)
1058 (add-hook 'magit-post-refresh-hook 'diff-hl-magit-post-refresh t))
1059
1060(use-package dired
1061 :defer t
1062 :config (setq dired-listing-switches "-alh"))
1063
1064(use-package eldoc
1065 :when (version< "25" emacs-version)
1066 :config (global-eldoc-mode))
1067
1068(use-package help
1069 :defer t
1070 :config
1071 (temp-buffer-resize-mode)
1072 (setq help-window-select t))
1073
1074(progn ; `isearch'
1075 (setq isearch-allow-scroll t))
1076
1077(use-package lisp-mode
1078 :config
1079 (add-hook 'emacs-lisp-mode-hook 'outline-minor-mode)
1080 (add-hook 'emacs-lisp-mode-hook 'reveal-mode)
1081 (defun indent-spaces-mode ()
1082 (setq indent-tabs-mode nil))
1083 (add-hook 'lisp-interaction-mode-hook #'indent-spaces-mode))
1084
1085(use-package man
1086 :defer t
1087 :config (setq Man-width 80))
1088
1089(use-package paren
1090 :config (show-paren-mode))
1091
1092(use-package prog-mode
1093 :config (global-prettify-symbols-mode)
1094 (defun indicate-buffer-boundaries-left ()
1095 (setq indicate-buffer-boundaries 'left))
1096 (add-hook 'prog-mode-hook #'indicate-buffer-boundaries-left))
1097
1098(use-package recentf
1099 :defer 0.5
1100 :config
1101 (add-to-list 'recentf-exclude "^/\\(?:ssh\\|su\\|sudo\\)?:")
1102 (setq recentf-max-saved-items 40))
1103
1104(use-package savehist
1105 :config (savehist-mode))
1106
1107(use-package saveplace
1108 :when (version< "25" emacs-version)
1109 :config (save-place-mode))
1110
1111(use-package simple
1112 :config (column-number-mode))
1113
1114(progn ; `text-mode'
1115 (add-hook 'text-mode-hook #'indicate-buffer-boundaries-left)
1116 (add-hook 'text-mode-hook #'abbrev-mode))
1117
1118(use-package tramp
1119 :defer t
1120 :config
1121 (add-to-list 'tramp-default-proxies-alist '(nil "\\`root\\'" "/ssh:%h:"))
1122 (add-to-list 'tramp-default-proxies-alist '("localhost" nil nil))
1123 (add-to-list 'tramp-default-proxies-alist
1124 (list (regexp-quote (system-name)) nil nil)))
1125
1126(use-package undo-tree
1127 :config
1128 (global-undo-tree-mode -1))
1129 ;; :bind (("C-?" . undo-tree-undo)
1130 ;; ("M-_" . undo-tree-redo))
1131 ;; :config
1132 ;; (global-undo-tree-mode)
1133 ;; (setq undo-tree-mode-lighter ""
1134 ;; undo-tree-auto-save-history t))
1135#+end_src
1136
1137* Editing
1138
1139** Company
1140
1141#+begin_src emacs-lisp
1142(use-package company
1143 :defer 1
1144 :bind
1145 (:map company-active-map
1146 ([tab] . company-complete-common-or-cycle)
1147 ([escape] . company-abort))
1148 :custom
1149 (company-minimum-prefix-length 1)
1150 (company-selection-wrap-around t)
1151 (company-dabbrev-char-regexp "\\sw\\|\\s_\\|[-_]")
1152 (company-dabbrev-downcase nil)
1153 (company-dabbrev-ignore-case nil)
1154 :config
1155 (global-company-mode t))
1156#+end_src
1157
1158* Syntax and spell checking
1159#+begin_src emacs-lisp
1160(use-package flycheck
1161 :defer 3
1162 :hook (prog-mode . flycheck-mode)
1163 :bind
1164 (:map flycheck-mode-map
1165 ("M-P" . flycheck-previous-error)
1166 ("M-N" . flycheck-next-error))
1167 :config
1168 ;; Use the load-path from running Emacs when checking elisp files
1169 (setq flycheck-emacs-lisp-load-path 'inherit)
1170
1171 ;; Only flycheck when I actually save the buffer
1172 (setq flycheck-check-syntax-automatically '(mode-enabled save)))
1173
1174;; http://endlessparentheses.com/ispell-and-apostrophes.html
1175(use-package ispell
1176 :defer 3
1177 :config
1178 ;; ’ can be part of a word
1179 (setq ispell-local-dictionary-alist
1180 `((nil "[[:alpha:]]" "[^[:alpha:]]"
1181 "['\x2019]" nil ("-B") nil utf-8)))
1182 ;; don't send ’ to the subprocess
1183 (defun endless/replace-apostrophe (args)
1184 (cons (replace-regexp-in-string
1185 "’" "'" (car args))
1186 (cdr args)))
1187 (advice-add #'ispell-send-string :filter-args
1188 #'endless/replace-apostrophe)
1189
1190 ;; convert ' back to ’ from the subprocess
1191 (defun endless/replace-quote (args)
1192 (if (not (derived-mode-p 'org-mode))
1193 args
1194 (cons (replace-regexp-in-string
1195 "'" "’" (car args))
1196 (cdr args))))
1197 (advice-add #'ispell-parse-output :filter-args
1198 #'endless/replace-quote))
1199#+end_src
1200* Programming modes
1201
1202** [[http://alloytools.org][Alloy]] (with [[https://github.com/dwwmmn/alloy-mode][alloy-mode]])
1203
1204#+begin_src emacs-lisp
1205(use-package alloy-mode
1206 :defer t
1207 :config (setq alloy-basic-offset 2))
1208#+end_src
1209
1210** [[https://coq.inria.fr][Coq]] (with [[https://github.com/ProofGeneral/PG][Proof General]])
1211
1212#+begin_src emacs-lisp
1213(use-package proof-site ; Proof General
1214 :defer t
1215 :load-path "lib/proof-site/generic/")
1216#+end_src
1217
1218** [[https://leanprover.github.io][Lean]] (with [[https://github.com/leanprover/lean-mode][lean-mode]])
1219
1220#+begin_src emacs-lisp
1221(eval-when-compile (defvar lean-mode-map))
1222(use-package lean-mode
1223 :defer 1
1224 :bind (:map lean-mode-map
1225 ("S-SPC" . company-complete))
1226 :config
1227 (require 'lean-input)
1228 (setq default-input-method "Lean"
1229 lean-input-tweak-all '(lean-input-compose
1230 (lean-input-prepend "/")
1231 (lean-input-nonempty))
1232 lean-input-user-translations '(("/" "/")))
1233 (lean-input-setup))
1234 #+end_src
1235
1236** Haskell
1237
1238*** [[https://github.com/haskell/haskell-mode][haskell-mode]]
1239
1240#+begin_src emacs-lisp
1241(use-package haskell-mode
1242 :defer t
1243 :config
1244 (setq haskell-indentation-layout-offset 4
1245 haskell-indentation-left-offset 4
1246 flycheck-checker 'haskell-hlint
1247 flycheck-disabled-checkers '(haskell-stack-ghc haskell-ghc)))
1248#+end_src
1249
1250*** [[https://github.com/jyp/dante][dante]]
1251
1252#+begin_src emacs-lisp
1253(use-package dante
1254 :after haskell-mode
1255 :commands dante-mode
1256 :hook (haskell-mode . dante-mode))
1257#+end_src
1258
1259*** [[https://github.com/mpickering/hlint-refactor-mode][hlint-refactor]]
1260
1261Emacs bindings for [[https://github.com/ndmitchell/hlint][hlint]]'s refactor option. This requires the refact
1262executable from [[https://github.com/mpickering/apply-refact][apply-refact]].
1263
1264#+begin_src emacs-lisp
1265(use-package hlint-refactor
1266 :after haskell-mode
1267 :bind (:map hlint-refactor-mode-map
1268 ("C-c l b" . hlint-refactor-refactor-buffer)
1269 ("C-c l r" . hlint-refactor-refactor-at-point))
1270 :hook (haskell-mode . hlint-refactor-mode))
1271#+end_src
1272
1273*** [[https://github.com/flycheck/flycheck-haskell][flycheck-haskell]]
1274
1275#+begin_src emacs-lisp
1276(use-package flycheck-haskell
1277 :after haskell-mode)
1278#+end_src
1279
1280*** [[https://github.com/ndmitchell/hlint/blob/20e116a043f2073c57b17b24ae6364b5e433ba7e/data/hs-lint.el][hs-lint.el]]
1281:PROPERTIES:
1282:header-args+: :tangle lisp/hs-lint.el :mkdirp yes
1283:END:
1284
1285Currently using =flycheck-haskell= with the =haskell-hlint= checker
1286instead.
1287
1288#+begin_src emacs-lisp :tangle no
1289;;; hs-lint.el --- minor mode for HLint code checking
1290
1291;; Copyright 2009 (C) Alex Ott
1292;;
1293;; Author: Alex Ott <alexott@gmail.com>
1294;; Keywords: haskell, lint, HLint
1295;; Requirements:
1296;; Status: distributed under terms of GPL2 or above
1297
1298;; Typical message from HLint looks like:
1299;;
1300;; /Users/ott/projects/lang-exp/haskell/test.hs:52:1: Eta reduce
1301;; Found:
1302;; count1 p l = length (filter p l)
1303;; Why not:
1304;; count1 p = length . filter p
1305
1306
1307(require 'compile)
1308
1309(defgroup hs-lint nil
1310 "Run HLint as inferior of Emacs, parse error messages."
1311 :group 'tools
1312 :group 'haskell)
1313
1314(defcustom hs-lint-command "hlint"
1315 "The default hs-lint command for \\[hlint]."
1316 :type 'string
1317 :group 'hs-lint)
1318
1319(defcustom hs-lint-save-files t
1320 "Save modified files when run HLint or no (ask user)"
1321 :type 'boolean
1322 :group 'hs-lint)
1323
1324(defcustom hs-lint-replace-with-suggestions nil
1325 "Replace user's code with suggested replacements"
1326 :type 'boolean
1327 :group 'hs-lint)
1328
1329(defcustom hs-lint-replace-without-ask nil
1330 "Replace user's code with suggested replacements automatically"
1331 :type 'boolean
1332 :group 'hs-lint)
1333
1334(defun hs-lint-process-setup ()
1335 "Setup compilation variables and buffer for `hlint'."
1336 (run-hooks 'hs-lint-setup-hook))
1337
1338;; regex for replace suggestions
1339;;
1340;; ^\(.*?\):\([0-9]+\):\([0-9]+\): .*
1341;; Found:
1342;; \s +\(.*\)
1343;; Why not:
1344;; \s +\(.*\)
1345
1346(defvar hs-lint-regex
1347 "^\\(.*?\\):\\([0-9]+\\):\\([0-9]+\\): .*[\n\C-m]Found:[\n\C-m]\\s +\\(.*\\)[\n\C-m]Why not:[\n\C-m]\\s +\\(.*\\)[\n\C-m]"
1348 "Regex for HLint messages")
1349
1350(defun make-short-string (str maxlen)
1351 (if (< (length str) maxlen)
1352 str
1353 (concat (substring str 0 (- maxlen 3)) "...")))
1354
1355(defun hs-lint-replace-suggestions ()
1356 "Perform actual replacement of suggestions"
1357 (goto-char (point-min))
1358 (while (re-search-forward hs-lint-regex nil t)
1359 (let* ((fname (match-string 1))
1360 (fline (string-to-number (match-string 2)))
1361 (old-code (match-string 4))
1362 (new-code (match-string 5))
1363 (msg (concat "Replace '" (make-short-string old-code 30)
1364 "' with '" (make-short-string new-code 30) "'"))
1365 (bline 0)
1366 (eline 0)
1367 (spos 0)
1368 (new-old-code ""))
1369 (save-excursion
1370 (switch-to-buffer (get-file-buffer fname))
1371 (goto-char (point-min))
1372 (forward-line (1- fline))
1373 (beginning-of-line)
1374 (setf bline (point))
1375 (when (or hs-lint-replace-without-ask
1376 (yes-or-no-p msg))
1377 (end-of-line)
1378 (setf eline (point))
1379 (beginning-of-line)
1380 (setf old-code (regexp-quote old-code))
1381 (while (string-match "\\\\ " old-code spos)
1382 (setf new-old-code (concat new-old-code
1383 (substring old-code spos (match-beginning 0))
1384 "\\ *"))
1385 (setf spos (match-end 0)))
1386 (setf new-old-code (concat new-old-code (substring old-code spos)))
1387 (remove-text-properties bline eline '(composition nil))
1388 (when (re-search-forward new-old-code eline t)
1389 (replace-match new-code nil t)))))))
1390
1391(defun hs-lint-finish-hook (buf msg)
1392 "Function, that is executed at the end of HLint execution"
1393 (if hs-lint-replace-with-suggestions
1394 (hs-lint-replace-suggestions)
1395 (next-error 1 t)))
1396
1397(define-compilation-mode hs-lint-mode "HLint"
1398 "Mode for check Haskell source code."
1399 (set (make-local-variable 'compilation-process-setup-function)
1400 'hs-lint-process-setup)
1401 (set (make-local-variable 'compilation-disable-input) t)
1402 (set (make-local-variable 'compilation-scroll-output) nil)
1403 (set (make-local-variable 'compilation-finish-functions)
1404 (list 'hs-lint-finish-hook))
1405 )
1406
1407(defun hs-lint ()
1408 "Run HLint for current buffer with haskell source"
1409 (interactive)
1410 (save-some-buffers hs-lint-save-files)
1411 (compilation-start (concat hs-lint-command " \"" buffer-file-name "\"")
1412 'hs-lint-mode))
1413
1414(provide 'hs-lint)
1415;;; hs-lint.el ends here
1416#+end_src
1417
1418#+begin_src emacs-lisp :tangle no
1419(use-package hs-lint
1420 :load-path "lisp/"
1421 :bind (:map haskell-mode-map
1422 ("C-c l l" . hs-lint)))
1423#+end_src
1424
1425** Web dev
1426
1427*** SGML and HTML
1428
1429#+begin_src emacs-lisp
1430(use-package sgml-mode
1431 :defer t
1432 :config
1433 (setq sgml-basic-offset 2))
1434#+end_src
1435
1436*** CSS and SCSS
1437
1438#+begin_src emacs-lisp
1439(use-package css-mode
1440 :defer t
1441 :config
1442 (setq css-indent-offset 2))
1443#+end_src
1444
1445*** Web mode
1446
1447#+begin_src emacs-lisp
1448(use-package web-mode
1449 :defer t
1450 :mode "\\.html\\'"
1451 :config
1452 (setq-every! 2
1453 web-mode-code-indent-offset
1454 web-mode-css-indent-offset
1455 web-mode-markup-indent-offset))
1456#+end_src
1457
1458*** Emmet mode
1459
1460#+begin_src emacs-lisp
1461(use-package emmet-mode
1462 :after (:any web-mode css-mode sgml-mode)
1463 :bind* (("C-)" . emmet-next-edit-point)
1464 ("C-(" . emmet-prev-edit-point))
1465 :config
1466 (unbind-key "C-j" emmet-mode-keymap)
1467 (setq emmet-move-cursor-between-quotes t)
1468 :hook (web-mode css-mode html-mode sgml-mode))
1469#+end_src
1470
1471** Nix
1472
1473#+begin_src emacs-lisp
1474(use-package nix-mode
1475 :defer t
1476 :mode "\\.nix\\'")
1477#+end_src
1478
1479** Java
1480
1481*** meghanada
1482
1483#+begin_src emacs-lisp :tangle no
1484(use-package meghanada
1485 :bind
1486 (:map meghanada-mode-map
1487 (("C-M-o" . meghanada-optimize-import)
1488 ("C-M-t" . meghanada-import-all)))
1489 :hook (java-mode . meghanada-mode))
1490#+end_src
1491
1492*** lsp-java
1493
1494#+begin_comment
1495dependencies:
1496
1497ace-window
1498avy
1499bui
1500company-lsp
1501dap-mode
1502lsp-java
1503lsp-mode
1504lsp-ui
1505pfuture
1506tree-mode
1507treemacs
1508#+end_comment
1509
1510#+begin_src emacs-lisp :tangle no
1511(use-package treemacs
1512 :config (setq treemacs-never-persist t))
1513
1514(use-package yasnippet
1515 :config
1516 ;; (yas-global-mode)
1517 )
1518
1519(use-package lsp-mode
1520 :init (setq lsp-eldoc-render-all nil
1521 lsp-highlight-symbol-at-point nil)
1522 )
1523
1524(use-package hydra)
1525
1526(use-package company-lsp
1527 :after company
1528 :config
1529 (setq company-lsp-cache-candidates t
1530 company-lsp-async t))
1531
1532(use-package lsp-ui
1533 :config
1534 (setq lsp-ui-sideline-update-mode 'point))
1535
1536(use-package lsp-java
1537 :config
1538 (add-hook 'java-mode-hook
1539 (lambda ()
1540 (setq-local company-backends (list 'company-lsp))))
1541
1542 (add-hook 'java-mode-hook 'lsp-java-enable)
1543 (add-hook 'java-mode-hook 'flycheck-mode)
1544 (add-hook 'java-mode-hook 'company-mode)
1545 (add-hook 'java-mode-hook 'lsp-ui-mode))
1546
1547(use-package dap-mode
1548 :after lsp-mode
1549 :config
1550 (dap-mode t)
1551 (dap-ui-mode t))
1552
1553(use-package dap-java
1554 :after (lsp-java))
1555
1556(use-package lsp-java-treemacs
1557 :after (treemacs))
1558#+end_src
1559
1560* Emacs Enhancements
1561
1562** [[https://github.com/justbur/emacs-which-key][which-key]]
1563
1564#+begin_quote
1565Emacs package that displays available keybindings in popup
1566#+end_quote
1567
1568#+begin_src emacs-lisp
1569(use-package which-key
1570 :defer 1
1571 :config (which-key-mode))
1572#+end_src
1573
1574** theme
1575
1576#+begin_src emacs-lisp
1577(add-to-list 'custom-theme-load-path "~/.emacs.d/lisp")
1578(load-theme 'tangomod t)
1579#+end_src
1580
1581** doom-modeline
1582
1583#+begin_src emacs-lisp
1584(use-package doom-modeline
1585 :demand t
1586 :config (setq doom-modeline-height 32)
1587 :hook (after-init . doom-modeline-init))
1588#+end_src
1589
1590** doom-themes
1591
1592#+begin_src emacs-lisp
1593(use-package doom-themes)
1594#+end_src
1595
1596** theme helper functions
1597
1598#+begin_src emacs-lisp
1599(defun amin/lights-on ()
1600 "Enable my favourite light theme."
1601 (interactive)
1602 (progn
1603 (mapc #'disable-theme custom-enabled-themes)
1604 (load-theme 'tangomod t)))
1605
1606(defun amin/lights-off ()
1607 "Go dark."
1608 (interactive)
1609 (progn
1610 (mapc #'disable-theme custom-enabled-themes)
1611 (load-theme 'doom-tomorrow-night t)))
1612
1613(bind-keys
1614 ("s-t d" . amin/lights-off)
1615 ("s-t l" . amin/lights-on))
1616#+end_src
1617
1618** [[https://github.com/bbatsov/crux][crux]]
1619
1620#+begin_src emacs-lisp
1621(use-package crux
1622 :defer 1
1623 :bind (("C-c b k" . crux-kill-other-buffers)
1624 ("C-c d" . crux-duplicate-current-line-or-region)
1625 ("C-c D" . crux-duplicate-and-comment-current-line-or-region)
1626 ("C-c f c" . crux-copy-file-preserve-attributes)
1627 ("C-c f d" . crux-delete-file-and-buffer)
1628 ("C-c f r" . crux-rename-file-and-buffer)
1629 ("C-c j" . crux-top-join-line)
1630 ("C-S-j" . crux-top-join-line)))
1631#+end_src
1632
1633** [[https://github.com/alezost/mwim.el][mwim]]
1634
1635#+begin_src emacs-lisp
1636(use-package mwim
1637 :bind (("C-a" . mwim-beginning-of-code-or-line)
1638 ("C-e" . mwim-end-of-code-or-line)
1639 ("<home>" . mwim-beginning-of-line-or-code)
1640 ("<end>" . mwim-end-of-line-or-code)))
1641#+end_src
1642
1643** projectile
1644
1645#+begin_src emacs-lisp
1646(use-package projectile
1647 :defer t
1648 :bind-keymap ("C-c p" . projectile-command-map)
1649 :config
1650 (projectile-mode)
1651
1652 (defun my-projectile-invalidate-cache (&rest _args)
1653 ;; ignore the args to `magit-checkout'
1654 (projectile-invalidate-cache nil))
1655
1656 (eval-after-load 'magit-branch
1657 '(progn
1658 (advice-add 'magit-checkout
1659 :after #'my-projectile-invalidate-cache)
1660 (advice-add 'magit-branch-and-checkout
1661 :after #'my-projectile-invalidate-cache))))
1662#+end_src
1663
1664** [[https://github.com/Wilfred/helpful][helpful]]
1665
1666#+begin_src emacs-lisp
1667(use-package helpful
1668 :defer 1
1669 :bind
1670 (("C-S-h c" . helpful-command)
1671 ("C-S-h f" . helpful-callable) ; helpful-function
1672 ("C-S-h v" . helpful-variable)
1673 ("C-S-h k" . helpful-key)
1674 ("C-S-h p" . helpful-at-point)))
1675#+end_src
1676
1677** [[https://github.com/knu/shell-toggle.el][shell-toggle]]
1678
1679#+begin_src emacs-lisp
1680(use-package shell-toggle
1681 :after eshell
1682 :bind ("C-c a s e" . amin/shell-toggle)
1683 :config
1684 (defun amin/shell-toggle (make-cd)
1685 "Toggle between the shell buffer and whatever buffer you are editing.
1686With a prefix argument MAKE-CD also insert a \"cd DIR\" command
1687into the shell, where DIR is the directory of the current buffer.
1688
1689When called in the shell buffer returns you to the buffer you were editing
1690before calling this the first time.
1691
1692Options: `shell-toggle-goto-eob'"
1693 (interactive "P")
1694 ;; Try to decide on one of three possibilities:
1695 ;; If not in shell-buffer, switch to it.
1696 ;; If in shell-buffer, return to state before going to the shell-buffer
1697 (if (eq (current-buffer) shell-toggle-shell-buffer)
1698 (shell-toggle-buffer-return-from-shell)
1699 (progn
1700 (shell-toggle-buffer-goto-shell make-cd)
1701 (if shell-toggle-full-screen-window-only (delete-other-windows)))))
1702
1703 ;; override to split horizontally instead
1704 (defun shell-toggle-buffer-switch-to-other-window ()
1705 "Switch to other window.
1706If the current window is the only window in the current frame,
1707create a new window and switch to it.
1708
1709\(This is less intrusive to the current window configuration than
1710`switch-buffer-other-window')"
1711 (let ((this-window (selected-window)))
1712 (other-window 1)
1713 ;; If we did not switch window then we only have one window and need to
1714 ;; create a new one.
1715 (if (eq this-window (selected-window))
1716 (progn
1717 (split-window-horizontally)
1718 (other-window 1)))))
1719
1720 :custom
1721 (shell-toggle-launch-shell 'shell-toggle-eshell))
1722#+end_src
1723
1724** [[https://github.com/EricCrosson/unkillable-scratch][unkillable-scratch]]
1725
1726Make =*scratch*= and =*Messages*= unkillable.
1727
1728#+begin_src emacs-lisp
1729(use-package unkillable-scratch
1730 :defer 3
1731 :config
1732 (unkillable-scratch 1)
1733 :custom
1734 (unkillable-scratch-behavior 'do-nothing)
1735 (unkillable-buffers '("^\\*scratch\\*$" "^\\*Messages\\*$")))
1736#+end_src
1737
1738** [[https://github.com/davep/boxquote.el][boxquote.el]]
1739
1740#+begin_example
1741,----
1742| make pretty boxed quotes like this
1743`----
1744#+end_example
1745
1746#+begin_src emacs-lisp
1747(use-package boxquote
1748 :defer 3
1749 :bind
1750 (:prefix-map amin--boxquote-prefix-map
1751 :prefix "C-c q"
1752 ("b" . boxquote-buffer)
1753 ("B" . boxquote-insert-buffer)
1754 ("d" . boxquote-defun)
1755 ("F" . boxquote-insert-file)
1756 ("hf" . boxquote-describe-function)
1757 ("hk" . boxquote-describe-key)
1758 ("hv" . boxquote-describe-variable)
1759 ("hw" . boxquote-where-is)
1760 ("k" . boxquote-kill)
1761 ("p" . boxquote-paragraph)
1762 ("q" . boxquote-boxquote)
1763 ("r" . boxquote-region)
1764 ("s" . boxquote-shell-command)
1765 ("t" . boxquote-text)
1766 ("T" . boxquote-title)
1767 ("u" . boxquote-unbox)
1768 ("U" . boxquote-unbox-region)
1769 ("y" . boxquote-yank)
1770 ("M-q" . boxquote-fill-paragraph)
1771 ("M-w" . boxquote-kill-ring-save)))
1772#+end_src
1773
1774Also see [[https://www.emacswiki.org/emacs/rebox2][rebox2]].
1775
1776** COMMENT [[https://github.com/DarthFennec/highlight-indent-guides][highlight-indent-guides]] :ARCHIVE:
1777
1778#+begin_src emacs-lisp
1779(use-package highlight-indent-guides
1780 :defer 3
1781 :hook ((prog-mode . highlight-indent-guides-mode)
1782 ;; (org-mode . highlight-indent-guides-mode)
1783 )
1784 :config
1785 (setq highlight-indent-guides-character ?\|)
1786 (setq highlight-indent-guides-auto-enabled nil)
1787 (setq highlight-indent-guides-method 'character)
1788 (setq highlight-indent-guides-responsive 'top)
1789 (set-face-foreground 'highlight-indent-guides-character-face "gainsboro")
1790 (set-face-foreground 'highlight-indent-guides-top-character-face "grey40")) ; grey13 is nice too
1791#+end_src
1792
1793** pdf-tools
1794
1795#+begin_src emacs-lisp
1796(use-package pdf-tools
1797 :defer t
1798 :magic ("%PDF" . pdf-view-mode)
1799 :config
1800 (setq pdf-view-resize-factor 1.05)
1801 (pdf-tools-install)
1802 :bind
1803 (:map pdf-view-mode-map
1804 ("C-s" . isearch-forward)
1805 ("C-r" . isearch-backward)
1806 ("j" . pdf-view-next-line-or-next-page)
1807 ("k" . pdf-view-previous-line-or-previous-page)
1808 ("h" . image-backward-hscroll)
1809 ("l" . image-forward-hscroll)))
1810#+end_src
1811
1812** anzu
1813
1814#+begin_src emacs-lisp
1815(use-package anzu)
1816#+end_src
1817
1818** typo.el
1819
1820#+begin_src emacs-lisp
1821(use-package typo
1822 :defer 2
1823 :config
1824 (typo-global-mode 1)
1825 :hook (text-mode . typo-mode))
1826#+end_src
1827
1828** hl-todo
1829
1830#+begin_src emacs-lisp
1831(use-package hl-todo
1832 :defer 4
1833 :config
1834 (global-hl-todo-mode))
1835#+end_src
1836
1837** shrink-path
1838
1839#+begin_src emacs-lisp
1840(use-package shrink-path
1841 :after eshell
1842 :config
1843 (setq eshell-prompt-regexp "\\(.*\n\\)*λ "
1844 eshell-prompt-function #'+eshell/prompt)
1845
1846 (defun +eshell/prompt ()
1847 (let ((base/dir (shrink-path-prompt default-directory)))
1848 (concat (propertize (car base/dir)
1849 'face 'font-lock-comment-face)
1850 (propertize (cdr base/dir)
1851 'face 'font-lock-constant-face)
1852 (propertize (+eshell--current-git-branch)
1853 'face 'font-lock-function-name-face)
1854 "\n"
1855 (propertize "λ" 'face 'eshell-prompt-face)
1856 ;; needed for the input text to not have prompt face
1857 (propertize " " 'face 'default))))
1858
1859 (defun +eshell--current-git-branch ()
1860 (let ((branch (car (loop for match in (split-string (shell-command-to-string "git branch") "\n")
1861 when (string-match "^\*" match)
1862 collect match))))
1863 (if (not (eq branch nil))
1864 (concat " " (substring branch 2))
1865 ""))))
1866#+end_src
1867
1868** COMMENT slack :ARCHIVE:
1869
1870Hopefully temporary.
1871
1872#+begin_src emacs-lisp
1873(use-package slack
1874 :commands (slack-start)
1875 :init
1876 (eval-when-compile ; silence the byte-compiler
1877 (defvar url-http-data nil)
1878 (defvar url-http-extra-headers nil)
1879 (defvar url-http-method nil)
1880 (defvar url-callback-function nil)
1881 (defvar url-callback-arguments nil)
1882 (defvar oauth--token-data nil))
1883 (setq slack-buffer-emojify t
1884 slack-prefer-current-team t)
1885 :config
1886 (slack-register-team
1887 :name "uw-apv"
1888 :default t
1889 :client-id uw-apv-client-id
1890 :client-secret uw-apv-client-secret
1891 :token uw-apv-token
1892 :subscribed-channels '(general)
1893 :full-and-display-names t)
1894 (slack-register-team
1895 :name "watform"
1896 :default nil
1897 :client-id watform-client-id
1898 :client-secret watform-client-secret
1899 :token watform-token
1900 :subscribed-channels '(general)
1901 :full-and-display-names t)
1902 (add-to-list 'swiper-font-lock-exclude 'slack-message-buffer-mode t)
1903 (setq lui-time-stamp-format "[%Y-%m-%d %H:%M:%S]"
1904 lui-time-stamp-only-when-changed-p t
1905 lui-time-stamp-position 'right)
1906 :bind
1907 (("C-c s s" . slack-start)
1908 ("C-c s u" . slack-select-unread-rooms)
1909 ("C-c s b" . slack-select-rooms)
1910 ("C-c s t" . slack-change-current-team)
1911 ("C-c s c" . slack-ws-close)
1912 :map slack-mode-map
1913 ("M-p" . slack-buffer-goto-prev-message)
1914 ("M-n" . slack-buffer-goto-next-message)
1915 ("C-c e" . slack-message-edit)
1916 ("C-c k" . slack-message-delete)
1917 ("C-c C-k" . slack-channel-leave)
1918 ("C-c r a" . slack-message-add-reaction)
1919 ("C-c r r" . slack-message-remove-reaction)
1920 ("C-c r s" . slack-message-show-reaction-users)
1921 ("C-c p l" . slack-room-pins-list)
1922 ("C-c p a" . slack-message-pins-add)
1923 ("C-c p r" . slack-message-pins-remove)
1924 ("@" . slack-message-embed-mention)
1925 ("#" . slack-message-embed-channel)))
1926
1927(use-package alert
1928 :commands (alert)
1929 :init
1930 (setq alert-default-style 'notifier))
1931#+end_src
1932
1933** COMMENT magithub :ARCHIVE:
1934
1935For when I /have to/ use GH.
1936
1937#+begin_src emacs-lisp
1938(use-package magithub
1939 :after magit
1940 :config
1941 (magithub-feature-autoinject t)
1942 (setq magithub-clone-default-directory "~/src/git"))
1943#+end_src
1944
1945** [[https://github.com/peterwvj/eshell-up][eshell-up]]
1946
1947#+begin_src emacs-lisp
1948(use-package eshell-up
1949 :after eshell)
1950#+end_src
1951
1952** multi-term
1953
1954#+begin_src emacs-lisp
1955(use-package multi-term
1956 :defer 1
1957 :bind (("C-c C-j" . term-line-mode)
1958 ("C-c m m" . multi-term)
1959 ("C-c m p" . multi-term-dedicated-toggle))
1960 :config
1961 (setq multi-term-program "/bin/screen"
1962 ;; TODO: add separate bindings for connecting to existing
1963 ;; session vs. always creating a new one
1964 multi-term-dedicated-select-after-open-p t
1965 multi-term-dedicated-window-height 20
1966 multi-term-dedicated-max-window-height 30
1967 term-bind-key-alist
1968 '(("C-c C-c" . term-interrupt-subjob)
1969 ("C-c C-e" . term-send-esc)
1970 ("C-k" . kill-line)
1971 ("C-y" . term-paste)
1972 ("M-f" . term-send-forward-word)
1973 ("M-b" . term-send-backward-word)
1974 ("M-p" . term-send-up)
1975 ("M-n" . term-send-down)
1976 ("<C-backspace>" . term-send-backward-kill-word)
1977 ("<M-DEL>" . term-send-backward-kill-word)
1978 ("M-d" . term-send-delete-word)
1979 ("M-," . term-send-raw)
1980 ("M-." . comint-dynamic-complete))
1981 term-unbind-key-alist
1982 '("C-z" "C-x" "C-c" "C-h" "C-y" "<ESC>")))
1983#+end_src
1984
1985* Email
1986
1987#+begin_src emacs-lisp
1988(defvar amin-maildir (expand-file-name "~/mail/"))
1989(after! recentf
1990 (add-to-list 'recentf-exclude amin-maildir))
1991#+end_src
1992
1993** Gnus
1994
1995#+begin_src emacs-lisp
1996(setq
1997 amin-gnus-init-file (no-littering-expand-etc-file-name "gnus")
1998 mail-user-agent 'gnus-user-agent
1999 read-mail-command 'gnus)
2000
2001(use-package gnus
2002 :bind (("s-m" . gnus)
2003 ("s-M" . gnus-unplugged))
2004 :init
2005 (setq
2006 gnus-select-method '(nnnil "")
2007 gnus-secondary-select-methods
2008 '((nnimap "amin"
2009 (nnimap-stream plain)
2010 (nnimap-address "127.0.0.1")
2011 (nnimap-server-port 143)
2012 (nnimap-authenticator plain)
2013 (nnimap-user "amin@aminb.org"))
2014 (nnimap "uwaterloo"
2015 (nnimap-stream plain)
2016 (nnimap-address "127.0.0.1")
2017 (nnimap-server-port 143)
2018 (nnimap-authenticator plain)
2019 (nnimap-user "abandali@uwaterloo.ca")))
2020 gnus-message-archive-group "nnimap+amin:Sent"
2021 gnus-parameters
2022 '(("gnu.*"
2023 (gcc-self . t)))
2024 gnus-large-newsgroup 50
2025 gnus-home-directory (no-littering-expand-var-file-name "gnus/")
2026 gnus-directory (concat gnus-home-directory "news/")
2027 message-directory (concat gnus-home-directory "mail/")
2028 nndraft-directory (concat gnus-home-directory "drafts/")
2029 gnus-save-newsrc-file nil
2030 gnus-read-newsrc-file nil
2031 gnus-interactive-exit nil
2032 gnus-gcc-mark-as-read t))
2033
2034(use-package gnus-art
2035 :config
2036 (setq
2037 gnus-visible-headers
2038 (concat gnus-visible-headers "\\|^List-Id:\\|^X-RT-Originator:\\|^User-Agent:")
2039 gnus-sorted-header-list
2040 '("^From:" "^Subject:" "^Summary:" "^Keywords:"
2041 "^Followup-To:" "^To:" "^Cc:" "X-RT-Originator"
2042 "^Newsgroups:" "List-Id:" "^Organization:"
2043 "^User-Agent:" "^Date:")
2044 ;; local-lapsed article dates
2045 ;; from https://www.emacswiki.org/emacs/GnusFormatting#toc11
2046 gnus-article-date-headers '(user-defined)
2047 gnus-article-time-format
2048 (lambda (time)
2049 (let* ((date (format-time-string "%a, %d %b %Y %T %z" time))
2050 (local (article-make-date-line date 'local))
2051 (combined-lapsed (article-make-date-line date
2052 'combined-lapsed))
2053 (lapsed (progn
2054 (string-match " (.+" combined-lapsed)
2055 (match-string 0 combined-lapsed))))
2056 (concat local lapsed))))
2057 (bind-keys
2058 :map gnus-article-mode-map
2059 ("r" . gnus-article-reply-with-original)
2060 ("R" . gnus-article-wide-reply-with-original)
2061 ("M-L" . org-store-link)))
2062
2063(use-package gnus-sum
2064 :bind (:map gnus-summary-mode-map
2065 :prefix-map amin--gnus-summary-prefix-map
2066 :prefix "v"
2067 ("r" . gnus-summary-reply)
2068 ("w" . gnus-summary-wide-reply)
2069 ("v" . gnus-summary-show-raw-article))
2070 :config
2071 (bind-keys
2072 :map gnus-summary-mode-map
2073 ("r" . gnus-summary-reply-with-original)
2074 ("R" . gnus-summary-wide-reply-with-original)
2075 ("M-L" . org-store-link))
2076 :hook (gnus-summary-mode . amin--no-mouse-autoselect-window))
2077
2078(use-package gnus-msg
2079 :config
2080 (setq gnus-posting-styles
2081 '((".*"
2082 (address "amin@aminb.org")
2083 (body "\nBest,\namin\n")
2084 (eval (setq amin--message-cite-say-hi t)))
2085 ("gnu.*"
2086 (address "bandali@gnu.org"))
2087 ((header "subject" "ThankCRM")
2088 (to "webmasters-comment@gnu.org")
2089 (body "\nAdded to 2018supporters.html.\n\nMoving to campaigns.\n\n-amin\n")
2090 (eval (setq amin--message-cite-say-hi nil)))
2091 ("nnimap\\+uwaterloo:.*"
2092 (address "abandali@uwaterloo.ca")
2093 (gcc "\"nnimap+uwaterloo:Sent Items\"")))))
2094
2095(use-package gnus-topic
2096 :hook (gnus-group-mode . gnus-topic-mode))
2097
2098(use-package gnus-agent
2099 :config
2100 (setq gnus-agent-synchronize-flags 'ask)
2101 :hook (gnus-group-mode . gnus-agent-mode))
2102
2103(use-package gnus-group
2104 :config
2105 (setq gnus-permanently-visible-groups "\\((INBOX\\|gnu$\\)"))
2106
2107(use-package mm-decode
2108 :config
2109 (setq mm-discouraged-alternatives '("text/html" "text/richtext")))
2110#+end_src
2111
2112** sendmail
2113
2114#+begin_src emacs-lisp
2115(use-package sendmail
2116 :config
2117 (setq sendmail-program "/usr/bin/msmtp"
2118 ;; message-sendmail-extra-arguments '("-v" "-d")
2119 mail-specify-envelope-from t
2120 mail-envelope-from 'header))
2121#+end_src
2122
2123** message
2124
2125#+begin_src emacs-lisp
2126(use-package message
2127 :config
2128 (defconst amin--message-cite-style-format "On %Y-%m-%d %l:%M %p, %N wrote:")
2129 (defconst message-cite-style-bandali
2130 '((message-cite-function 'message-cite-original)
2131 (message-citation-line-function 'message-insert-formatted-citation-line)
2132 (message-cite-reply-position 'traditional)
2133 (message-yank-prefix "> ")
2134 (message-yank-cited-prefix ">")
2135 (message-yank-empty-prefix ">")
2136 (message-citation-line-format
2137 (if amin--message-cite-say-hi
2138 (concat "Hi %F,\n\n" amin--message-cite-style-format)
2139 amin--message-cite-style-format)))
2140 "Citation style based on Mozilla Thunderbird's. Use with message-cite-style.")
2141 (setq message-cite-style 'message-cite-style-bandali
2142 message-kill-buffer-on-exit t
2143 message-send-mail-function 'message-send-mail-with-sendmail
2144 message-sendmail-envelope-from 'header
2145 message-dont-reply-to-names
2146 "\\(\\(.*@aminb\\.org\\)\\|\\(amin@bandali\\.me\\)\\|\\(\\(aminb?\\|mab\\|bandali\\)@gnu\\.org\\)\\|\\(\\(m\\|a\\(min\\.\\)?\\)bandali@uwaterloo\\.ca\\)\\)"
2147 message-user-fqdn "aminb.org")
2148 :hook (;; (message-setup . mml-secure-message-sign-pgpmime)
2149 (message-mode . flyspell-mode)
2150 (message-mode . (lambda ()
2151 ;; (setq fill-column 65
2152 ;; message-fill-column 65)
2153 (make-local-variable 'company-idle-delay)
2154 (setq company-idle-delay 0.2))))
2155 ;; :custom-face
2156 ;; (message-header-subject ((t (:foreground "#111" :weight semi-bold))))
2157 ;; (message-header-to ((t (:foreground "#111" :weight normal))))
2158 ;; (message-header-cc ((t (:foreground "#333" :weight normal))))
2159 )
2160
2161(after! mml-sec
2162 (setq mml-secure-openpgp-encrypt-to-self t
2163 mml-secure-openpgp-sign-with-sender t))
2164#+end_src
2165
2166** footnote
2167
2168Convenient footnotes in =message-mode=.
2169
2170#+begin_src emacs-lisp
2171(use-package footnote
2172 :after message
2173 :bind
2174 (:map message-mode-map
2175 :prefix-map amin--footnote-prefix-map
2176 :prefix "C-c f"
2177 ("a" . footnote-add-footnote)
2178 ("b" . footnote-back-to-message)
2179 ("c" . footnote-cycle-style)
2180 ("d" . footnote-delete-footnote)
2181 ("g" . footnote-goto-footnote)
2182 ("r" . footnote-renumber-footnotes)
2183 ("s" . footnote-set-style))
2184 :config
2185 (setq footnote-start-tag ""
2186 footnote-end-tag ""
2187 footnote-style 'unicode))
2188#+end_src
2189
2190** bbdb
2191
2192Manually install bbdb (=lisp/bbdb= copied from an ELPA-based setup),
2193because installing it from source on Emacs 27 using the following
2194submodule configuration for some reason doesn’t work and results in
2195very strange errors when using any of the functions.
2196
2197#+begin_src conf :tangle no
2198[submodule "bbdb"]
2199 path = lib/bbdb
2200 url = https://git.savannah.nongnu.org/git/bbdb.git
2201 load-path = lisp
2202 info-path = doc
2203 build-step = ./autogen.sh
2204 build-step = ./configure
2205 build-step = make
2206 build-step = make install
2207#+end_src
2208
2209I tried using =borg-elpa= instead of doing it like this, but it added
22102 seconds to my startup time, which is unacceptable to me.
2211
2212#+begin_src emacs-lisp
2213(use-package bbdb
2214 :load-path "lisp/bbdb"
2215 :init
2216 (load (expand-file-name "lisp/bbdb/bbdb-autoloads.el" user-emacs-directory))
2217 ;; (bbdb-mua-auto-update-init 'message)
2218 (setq bbdb-mua-auto-update-p 'query
2219 bbdb-complete-mail nil)
2220 (bbdb-initialize 'gnus 'message))
2221#+end_src
2222
2223** COMMENT message-x
2224
2225#+begin_src emacs-lisp
2226(use-package message-x
2227 :custom
2228 (message-x-completion-alist
2229 (quote
2230 (("\\([rR]esent-\\|[rR]eply-\\)?[tT]o:\\|[bB]?[cC][cC]:" . gnus-harvest-find-address)
2231 ((if
2232 (boundp
2233 (quote message-newgroups-header-regexp))
2234 message-newgroups-header-regexp message-newsgroups-header-regexp)
2235 . message-expand-group)))))
2236#+end_src
2237
2238** COMMENT gnus-harvest
2239
2240#+begin_src emacs-lisp
2241(use-package gnus-harvest
2242 :commands gnus-harvest-install
2243 :demand t
2244 :config
2245 (if (featurep 'message-x)
2246 (gnus-harvest-install 'message-x)
2247 (gnus-harvest-install)))
2248#+end_src
2249
2250* Blogging
2251** [[https://ox-hugo.scripter.co][ox-hugo]]
2252
2253#+begin_src emacs-lisp
2254(use-package ox-hugo
2255 :after ox)
2256
2257(use-package ox-hugo-auto-export
2258 :load-path "lib/ox-hugo")
2259#+end_src
2260
2261* Post initialization
2262:PROPERTIES:
2263:CUSTOM_ID: post-initialization
2264:END:
2265
2266Display how long it took to load the init file.
2267
2268#+begin_src emacs-lisp
2269(message "Loading %s...done (%.3fs)" user-init-file
2270 (float-time (time-subtract (current-time)
2271 amin--before-user-init-time)))
2272#+end_src
2273
2274* Footer
2275:PROPERTIES:
2276:CUSTOM_ID: footer
2277:END:
2278
2279#+begin_src emacs-lisp :comments none
2280;;; init.el ends here
2281#+end_src
2282
2283* COMMENT Local Variables :ARCHIVE:
2284# Local Variables:
2285# eval: (add-hook 'after-save-hook #'amin/async-babel-tangle 'append 'local)
2286# End: