1 #+title: Literate Emacs Configuration of Amin Bandali
4 #+property: header-args :tangle yes
11 This org file is my literate configuration for GNU Emacs, and is
12 tangled to [[./init.el][init.el]]. Packages are installed and managed using
13 [[https://github.com/raxod502/straight.el][straight.el]]. Over the years, I've taken inspiration from
14 configurations of many different people. Some of the configurations
15 that I can remember off the top of my head are:
17 - [[https://github.com/dieggsy/dotfiles][dieggsy/dotfiles]]: literate Emacs and dotfiles configuration, uses
18 straight.el for managing packages
19 - [[https://github.com/dakra/dmacs][dakra/dmacs]]: literate Emacs configuration, using Borg for managing
21 - [[http://pages.sachachua.com/.emacs.d/Sacha.html][Sacha Chua's literate Emacs configuration]]
22 - [[https://github.com/dakrone/eos][dakrone/eos]]
23 - Ryan Rix's [[http://doc.rix.si/cce/cce.html][Complete Computing Environment]] ([[http://doc.rix.si/projects/fsem.html][about cce]])
24 - [[https://github.com/jwiegley/dot-emacs][jwiegley/dot-emacs]]: nix-based configuration
25 - [[https://github.com/wasamasa/dotemacs][wasamasa/dotemacs]]
26 - [[https://github.com/hlissner/doom-emacs][Doom Emacs]]
28 I'd like to have a fully reproducible Emacs setup (part of the reason
29 why I store my configuration in this repository) but unfortunately out
30 of the box, that's not achievable with =package.el=, not currently
31 anyway. So, I've opted to use =straight.el=. I also used Borg for a
32 few months, but decided to try =straight.el= which allows direct use
33 of the various package archives.
37 :CUSTOM_ID: installation
40 To use this config for your Emacs, first you need to clone this repo,
41 then tangle =init.org= into =init.el=, and optionally byte-compile
44 First, clone the repository and =cd= into it:
46 #+begin_src sh :tangle no
47 git clone https://git.sr.ht/~bandali/dotfiles ~/.emacs.d
51 Then, decide if you would like to use a byte-compiled init file, and
52 set the [[#byte-compiled-init][a/byte-compiled-init]] variable accordingly.
54 Now, first tangle =init.org=, and only if you chose to have a
55 byte-compiled init, build init as well:
57 #+begin_src sh :tangle no
62 If you'd like to use a byte-compiled init, it's important that it be
63 recompiled whenever =init.el= is generated from an updated =init.org=.
64 Not only does my setup automatically and asynchronously tangle
65 =init.org= to =init.el= every time you edit and save =init.org= in GNU
66 Emacs, it will also invoke =make build-init= if you set
67 =a/byte-compiled-init= to =t= above, so you wouldn't have to worry
68 about manually tangling and compiling your init file whenever you
69 change it. The output of the last byte-compilation in the current
70 session is kept in a =*compilation*= buffer, which will automatically
71 be displayed if compilation fails.
73 * Contents :toc_1:noexport:
77 - [[#initial-setup][Initial setup]]
79 - [[#borg-essentials][Borg's =layer/essentials=]]
80 - [[#editing][Editing]]
81 - [[#syntax-spell-checking][Syntax and spell checking]]
82 - [[#programming-modes][Programming modes]]
83 - [[#emacs-enhancements][Emacs enhancements]]
85 - [[#blogging][Blogging]]
86 - [[#post-initialization][Post initialization]]
96 #+begin_src emacs-lisp :comments none
97 ;;; init.el --- Amin Bandali's Emacs config -*- lexical-binding: t; eval: (view-mode 1) -*-
100 Enable =view-mode=, which both makes the file read-only (as a reminder
101 that =init.el= is an auto-generated file, not supposed to be edited),
102 and provides some convenient key bindings for browsing through the
107 #+begin_src emacs-lisp :comments none
108 ;; Copyright (C) 2018-2019 Amin Bandali <bandali@gnu.org>
110 ;; This program is free software: you can redistribute it and/or modify
111 ;; it under the terms of the GNU General Public License as published by
112 ;; the Free Software Foundation, either version 3 of the License, or
113 ;; (at your option) any later version.
115 ;; This program is distributed in the hope that it will be useful,
116 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
117 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
118 ;; GNU General Public License for more details.
120 ;; You should have received a copy of the GNU General Public License
121 ;; along with this program. If not, see <https://www.gnu.org/licenses/>.
126 #+begin_src emacs-lisp :comments none
129 ;; Emacs configuration of Amin Bandali, computer scientist, functional
130 ;; programmer, and free software advocate.
132 ;; THIS FILE IS AUTO-GENERATED FROM `init.org'.
137 :CUSTOM_ID: initial-setup
140 ** Byte-compiled init preference
142 :CUSTOM_ID: byte-compiled-init
145 If you would like a byte-compiled init file, set the following
146 variable to ~t~, otherwise set it to ~nil~.
148 #+begin_src emacs-lisp
149 (defvar a/byte-compiled-init t
150 "If non-nil, byte-(re)compile init.el on successful tangles.")
153 You can click on [[#installation][Installation]] to jump back up there if you like :)
155 ** Emacs initialization
157 I'd like to do a couple of measurements of Emacs' startup time. First,
158 let's see how long Emacs takes to start up, before even loading
159 =init.el=, i.e. =user-init-file=:
161 #+begin_src emacs-lisp
162 (defvar a/before-user-init-time (current-time)
163 "Value of `current-time' when Emacs begins loading `user-init-file'.")
164 (message "Loading Emacs...done (%.3fs)"
165 (float-time (time-subtract a/before-user-init-time
169 Also, temporarily increase ~gc-cons-threshhold~ and
170 ~gc-cons-percentage~ during startup to reduce garbage collection
171 frequency. Clearing the ~file-name-handler-alist~ seems to help reduce
172 startup time as well.
174 #+begin_src emacs-lisp
175 (defvar a/gc-cons-threshold gc-cons-threshold)
176 (defvar a/gc-cons-percentage gc-cons-percentage)
177 (defvar a/file-name-handler-alist file-name-handler-alist)
178 (setq gc-cons-threshold (* 400 1024 1024) ; 400 MiB
179 gc-cons-percentage 0.6
180 file-name-handler-alist nil
181 ;; sidesteps a bug when profiling with esup
182 esup-child-profile-require-level 0)
185 Of course, we'd like to set them back to their defaults once we're
188 #+begin_src emacs-lisp
192 (setq gc-cons-threshold a/gc-cons-threshold
193 gc-cons-percentage a/gc-cons-percentage
194 file-name-handler-alist a/file-name-handler-alist)))
197 Increase the number of lines kept in message logs (the =*Messages*=
200 #+begin_src emacs-lisp
201 (setq message-log-max 20000)
204 Optionally, we could suppress some byte compiler warnings like below,
205 but for now I've decided to keep them enabled. See documentation for
206 ~byte-compile-warnings~ for more details.
208 #+begin_src emacs-lisp
209 ;; (setq byte-compile-warnings
210 ;; '(not free-vars unresolved noruntime lexical make-local))
215 #+begin_src emacs-lisp
216 (setq user-full-name "Amin Bandali"
217 user-mail-address "amin@bndl.org")
220 ** Package management
224 I can do all my package management things with =straight.el=, and
225 don't need Emacs' built-in =package.el=. Emacs 27 lets us disable
226 =package.el= in the =early-init-file= (see [[https://git.savannah.gnu.org/cgit/emacs.git/commit/?id=24acb31c04b4048b85311d794e600ecd7ce60d3b][here]]).
228 #+begin_src emacs-lisp :tangle early-init.el
229 (setq package-enable-at-startup nil)
232 But since Emacs 27 isn't out yet (Emacs 26 is just around the corner
233 right now), and even when released it'll be long before most distros
234 ship in their repos, I'll still put the old workaround with the
235 commented call to ~package-initialize~ here anyway.
237 #+begin_src emacs-lisp :tangle no
238 (setq package-enable-at-startup nil)
239 ;; (package-initialize)
242 Update: the above is not necessary, since =straight.el= automatically
243 does that (and more). See =straight-package-neutering-mode=.
248 Next-generation, purely functional package manager for the Emacs
252 =straight.el= allows me to have a fully reproducible Emacs setup.
254 #+begin_src emacs-lisp
255 ;; Main engine start...
257 (setq straight-repository-branch "develop"
258 straight-check-for-modifications '(check-on-save find-when-checking))
260 (defun a/bootstrap-straight ()
261 (defvar bootstrap-version)
262 (let ((bootstrap-file
263 (expand-file-name "straight/repos/straight.el/bootstrap.el" user-emacs-directory))
264 (bootstrap-version 5))
265 (unless (file-exists-p bootstrap-file)
267 (url-retrieve-synchronously
268 "https://raw.githubusercontent.com/raxod502/straight.el/develop/install.el"
269 'silent 'inhibit-cookies)
270 (goto-char (point-max))
271 (eval-print-last-sexp)))
272 (load bootstrap-file nil 'nomessage)))
274 ;; Solid rocket booster ignition...
276 (defun a/build-init ()
277 (a/bootstrap-straight)
278 (byte-compile-file "init.el"))
280 (a/bootstrap-straight)
284 (setq straight-use-package-by-default t)
287 Since we enable =straight.el='s =straight-use-package-by-default=
288 integration, we will define a =use-feature= for plain ole
289 =use-package= without any of the =straight.el= stuff.
291 #+begin_src emacs-lisp
292 (defmacro use-feature (name &rest args)
293 "Like `use-package', but with `straight-use-package-by-default' disabled."
294 (declare (indent defun))
303 A use-package declaration for simplifying your .emacs
306 [[https://github.com/jwiegley/use-package][use-package]] is an awesome utility for managing and configuring
307 packages (in our case especially the latter) in a neatly organized way
308 and without compromising on performance.
310 #+begin_src emacs-lisp
311 (straight-use-package 'use-package)
312 (if nil ; set to t when need to debug init
314 (setq use-package-verbose t
315 use-package-expand-minimally nil
316 use-package-compute-statistics t
318 (require 'use-package))
319 (setq use-package-verbose nil
320 use-package-expand-minimally t))
322 (setq use-package-always-defer t)
328 Browse the Emacsmirror package database
331 Epkg provides access to a local copy of the [[https://emacsmirror.net][Emacsmirror]] package
332 database, low-level functions for querying the database, and a
333 =package.el=-like user interface for browsing the available packages.
335 #+begin_src emacs-lisp
338 :commands (epkg-list-packages epkg-describe-package)
340 (("C-c p e d" . epkg-describe-package)
341 ("C-c p e p" . epkg-list-packages))
343 (setq epkg-repository "~/.emacs.d/straight/repos/epkgs/")
344 (eval-when-compile (defvar ivy-initial-inputs-alist))
345 (with-eval-after-load 'ivy
347 'ivy-initial-inputs-alist '(epkg-describe-package . "^") t)))
350 ** No littering in =~/.emacs.d=
353 Help keeping ~/.emacs.d clean
356 By default, even for Emacs' built-in packages, the configuration files
357 and persistent data are all over the place. Use =no-littering= to help
360 #+begin_src emacs-lisp
361 (use-package no-littering
365 (add-to-list 'savehist-additional-variables 'kill-ring)
367 (setq auto-save-file-name-transforms
368 `((".*" ,(no-littering-expand-var-file-name "auto-save/") t))))
371 ** Custom file (=custom.el=)
373 I'm not planning on using the custom file much, but even so, I
374 definitely don't want it mixing with =init.el=. So, here; let's give
375 it it's own file. While at it, treat themes as safe.
377 #+begin_src emacs-lisp
381 (setq custom-file (no-littering-expand-etc-file-name "custom.el"))
382 (when (file-exists-p custom-file)
384 (setf custom-safe-themes t))
389 Load the secrets file if it exists, otherwise show a warning.
391 #+begin_src emacs-lisp
393 (load (no-littering-expand-etc-file-name "secrets")))
396 ** Better =$PATH= handling
398 Let's use [[https://github.com/purcell/exec-path-from-shell][exec-path-from-shell]] to make Emacs use the =$PATH= as set up
401 #+begin_src emacs-lisp
402 (use-package exec-path-from-shell
405 (setq exec-path-from-shell-arguments nil
406 exec-path-from-shell-check-startup-files nil)
408 (exec-path-from-shell-initialize)
409 ;; while we're at it, let's fix access to our running ssh-agent
410 (exec-path-from-shell-copy-env "SSH_AGENT_PID")
411 (exec-path-from-shell-copy-env "SSH_AUTH_SOCK"))
414 ** COMMENT Only one custom theme at a time
416 #+begin_src emacs-lisp
417 (defadvice load-theme (before clear-previous-themes activate)
418 "Clear existing theme settings instead of layering them"
419 (mapc #'disable-theme custom-enabled-themes))
424 Start server if not already running. Alternatively, can be done by
425 issuing =emacs --daemon= in the terminal, which can be automated with
426 a systemd service or using =brew services start emacs= on macOS. I use
427 Emacs as my window manager (via EXWM), so I always start Emacs on
428 login; so starting the server from inside Emacs is good enough for me.
430 See [[https://www.gnu.org/software/emacs/manual/html_node/emacs/Emacs-Server.html#Emacs-Server][Using Emacs as a Server]].
432 #+begin_src emacs-lisp
435 :config (or (server-running-p) (server-mode)))
438 ** COMMENT Unicode support
440 Font stack with better unicode support, around =Ubuntu Mono= and
443 #+begin_src emacs-lisp
444 (dolist (ft (fontset-list))
448 (font-spec :name "Source Code Pro" :size 14))
452 (font-spec :name "DejaVu Sans Mono")
459 ;; :name "Symbola monospacified for DejaVu Sans Mono")
465 ;; (font-spec :name "DejaVu Sans Mono")
471 (font-spec :name "DejaVu Sans Mono" :size 14)
476 ** Gentler font resizing
478 #+begin_src emacs-lisp
479 (setq text-scale-mode-step 1.05)
482 ** Focus follows mouse
484 I’d like focus to follow the mouse when I move the cursor from one
487 #+begin_src emacs-lisp
488 (setq mouse-autoselect-window t)
491 Let’s define a function to conveniently disable this for certain
492 buffers and/or modes.
494 #+begin_src emacs-lisp
495 (defun a/no-mouse-autoselect-window ()
496 (make-local-variable 'mouse-autoselect-window)
497 (setq mouse-autoselect-window nil))
500 ** Better scrolling (arguably)
502 #+begin_src emacs-lisp
503 (setq ;; scroll-margin 1
504 ;; scroll-conservatively 10000
506 scroll-conservatively 10
507 scroll-preserve-screen-position 1)
512 (setq mouse-wheel-scroll-amount '(1 ((shift) . 1)) ; one line at a time
513 mouse-wheel-progressive-speed nil ; don't accelerate scrolling
514 mouse-wheel-follow-mouse t)) ; scroll window under mouse
516 (use-feature pixel-scroll
518 :config (pixel-scroll-mode 1))
521 ** Ask for GPG passphrase in minibuffer
523 #+begin_src emacs-lisp
524 (setq epg-pinentry-mode 'loopback)
529 #+begin_src emacs-lisp
536 Convenience macro for =setq='ing multiple variables to the same value:
538 #+begin_src emacs-lisp
539 (defmacro a/setq-every (value &rest vars)
540 "Set all the variables from VARS to value VALUE."
541 (declare (indent defun) (debug t))
542 `(progn ,@(mapcar (lambda (x) (list 'setq x value)) vars)))
545 The following process-related stuff from [[https://github.com/alezost/emacs-config][alezost's emacs-config]].
547 #+begin_src emacs-lisp
548 (defun a/start-process (program &rest args)
549 "Same as `start-process', but doesn't bother about name and buffer."
550 (let ((process-name (concat program "_process"))
551 (buffer-name (generate-new-buffer-name
552 (concat program "_output"))))
553 (apply #'start-process
554 process-name buffer-name program args)))
556 (defun a/dired-start-process (program &optional args)
557 "Open current file with a PROGRAM."
558 ;; Shell command looks like this: "program [ARGS]... FILE" (ARGS can
559 ;; be nil, so remove it).
560 (apply #'a/start-process
562 (remove nil (list args (dired-get-file-for-visit)))))
572 *** Time and battery in mode-line
574 Enable displaying time and battery in the mode-line, since I'm not
575 using the Xfce panel anymore. Also, I don't need to see the load
576 average on a regular basis, so disable that.
578 Note: using =i3status= on sway at the moment, so disabling this.
580 #+begin_src emacs-lisp :tangle no
583 (setq display-time-default-load-average nil)
589 (display-battery-mode))
594 Might want to set the fringe to a smaller value, especially if using
595 EXWM. I'm fine with the default for now.
597 #+begin_src emacs-lisp
598 ;; (fringe-mode '(3 . 1))
602 *** Disable disabled commands
604 Emacs disables some commands by default that could persumably be
605 confusing for novice users. Let's disable that.
607 #+begin_src emacs-lisp
608 (setq disabled-command-function nil)
613 Save what I copy into clipboard from other applications into Emacs'
614 kill-ring, which would allow me to still be able to easily access it
615 in case I kill (cut or copy) something else inside Emacs before
616 yanking (pasting) what I'd originally intended to.
618 #+begin_src emacs-lisp
619 (setq save-interprogram-paste-before-kill t)
624 #+begin_src emacs-lisp
625 (setq enable-recursive-minibuffers t
626 resize-mini-windows t)
629 *** Lazy-person-friendly yes/no prompts
631 Lazy people would prefer to type fewer keystrokes, especially for yes
632 or no questions. I'm lazy.
634 #+begin_src emacs-lisp
635 (defalias 'yes-or-no-p #'y-or-n-p)
638 *** Startup screen and =*scratch*=
640 Firstly, let Emacs know that I'd like to have =*scratch*= as my
643 #+begin_src emacs-lisp
644 (setq initial-buffer-choice t)
647 Now let's customize the =*scratch*= buffer a bit. First off, I don't
648 need the default hint.
650 #+begin_src emacs-lisp
651 (setq initial-scratch-message nil)
654 Also, let's use Text mode as the major mode, in case I want to
655 customize it (=*scratch*='s default major mode, Fundamental mode,
656 can't really be customized).
658 #+begin_src emacs-lisp
659 (setq initial-major-mode 'text-mode)
662 Inhibit the buffer list when more than 2 files are loaded.
664 #+begin_src emacs-lisp
665 (setq inhibit-startup-buffer-menu t)
668 I don't really need to see the startup screen or echo area message
671 #+begin_src emacs-lisp
672 (advice-add #'display-startup-echo-area-message :override #'ignore)
673 (setq inhibit-startup-screen t
674 inhibit-startup-echo-area-message user-login-name)
677 *** More useful frame titles
679 Show either the file name or the buffer name (in case the buffer isn't
680 visiting a file). Borrowed from Emacs Prelude.
682 #+begin_src emacs-lisp
683 (setq frame-title-format
684 '("" invocation-name " - "
685 (:eval (if (buffer-file-name)
686 (abbreviate-file-name (buffer-file-name))
692 Emacs' default backup settings aren't that great. Let's use more
693 sensible options. See documentation for the ~make-backup-file~
696 #+begin_src emacs-lisp
697 (setq backup-by-copying t
699 delete-old-versions t)
704 Enable automatic reloading of changed buffers and files.
706 #+begin_src emacs-lisp
707 (global-auto-revert-mode 1)
708 (setq auto-revert-verbose nil
709 global-auto-revert-non-file-buffers nil)
712 *** Always use space for indentation
714 #+begin_src emacs-lisp
717 require-final-newline t
723 Enable =winner-mode=.
725 #+begin_src emacs-lisp
729 *** Don’t display =*compilation*= on success
731 Based on https://stackoverflow.com/a/17788551, with changes to use
732 =cl-letf= instead of the now obsolete =flet=.
734 #+begin_src emacs-lisp
735 (with-eval-after-load 'compile
736 (defun a/compilation-finish-function (buffer outstr)
737 (unless (string-match "finished" outstr)
738 (switch-to-buffer-other-window buffer))
741 (setq compilation-finish-functions #'a/compilation-finish-function)
745 (defadvice compilation-start
746 (around inhibit-display
747 (command &optional mode name-function highlight-regexp))
748 (if (not (string-match "^\\(find\\|grep\\)" command))
749 (cl-letf (((symbol-function 'display-buffer) #'ignore))
750 (save-window-excursion ad-do-it))
752 (ad-activate 'compilation-start))
755 *** Search for non-ASCII characters
757 I’d like non-ASCII characters such as ‘’“”«»‹›áⓐ𝒶 to be selected when
758 I search for their ASCII counterpart. Shoutout to [[http://endlessparentheses.com/new-in-emacs-25-1-easily-search-non-ascii-characters.html][endlessparentheses]]
761 #+begin_src emacs-lisp
762 (setq search-default-mode #'char-fold-to-regexp)
764 ;; uncomment to extend this behaviour to query-replace
765 ;; (setq replace-char-fold t)
770 #+begin_src emacs-lisp
771 (setq-default cursor-type 'bar)
774 *** Allow scrolling in Isearch
776 #+begin_src emacs-lisp
777 (setq isearch-allow-scroll t)
782 Some bindings for functions from built-in GNU Emacs packages:
784 #+begin_src emacs-lisp
788 ("C-c e b" . eval-buffer)
789 ("C-c e r" . eval-region)
791 ("C-c e i" . emacs-init-time)
792 ("C-c e u" . emacs-uptime)
794 ("C-c F m" . make-frame-command)
795 ("C-c F d" . delete-frame)
796 ("C-c F D" . delete-other-frames)
798 ("C-c o" . other-window)
800 ("C-S-h C" . describe-char)
801 ("C-S-h F" . describe-face)
803 ("C-x k" . kill-this-buffer)
804 ("C-x K" . kill-buffer)
806 ("s-p" . beginning-of-buffer)
807 ("s-n" . end-of-buffer))
809 (when (display-graphic-p)
810 (unbind-key "C-z" global-map))
813 While at it, let's bind a few for some =straight-*= functions too:
815 #+begin_src emacs-lisp
817 :prefix-map a/straight-prefix-map
819 ("u" . straight-use-package)
820 ("f" . straight-freeze-versions)
821 ("t" . straight-thaw-versions)
822 ("P" . straight-prune-build)
823 ("r" . straight-get-recipe)
824 ;; M-x ^straight-.*-all$
825 ("a c" . straight-check-all)
826 ("a f" . straight-fetch-all)
827 ("a m" . straight-merge-all)
828 ("a n" . straight-normalize-all)
829 ("a F" . straight-pull-all)
830 ("a P" . straight-push-all)
831 ("a r" . straight-rebuild-all)
832 ;; M-x ^straight-.*-package$
833 ("p c" . straight-check-package)
834 ("p f" . straight-fetch-package)
835 ("p m" . straight-merge-package)
836 ("p n" . straight-normalize-package)
837 ("p F" . straight-pull-package)
838 ("p P" . straight-push-package)
839 ("p r" . straight-rebuild-package))
844 The packages in this section are absolutely essential to my everyday
845 workflow, and they play key roles in how I do my computing. They
846 immensely enhance the Emacs experience for me; both using Emacs, and
849 *** [[https://github.com/emacscollective/auto-compile][auto-compile]]
851 #+begin_src emacs-lisp
852 (use-package auto-compile
855 (auto-compile-on-load-mode)
856 (auto-compile-on-save-mode)
857 (setq auto-compile-display-buffer nil
858 auto-compile-mode-line-counter t
859 auto-compile-source-recreate-deletes-dest t
860 auto-compile-toggle-deletes-nonlib-dest t
861 auto-compile-update-autoloads t)
862 (add-hook 'auto-compile-inhibit-compile-hook
863 'auto-compile-inhibit-compile-detached-git-head))
866 *** [[https://orgmode.org/][Org]]
869 Org mode is for keeping notes, maintaining TODO lists, planning
870 projects, and authoring documents with a fast and effective plain-text
874 In short, my favourite way of life.
876 First, we have to resort to a [[https://github.com/raxod502/straight.el#installing-org-with-straightel][hack]] to be able to use the correct
877 latest version of Org from upstream.
879 #+begin_src emacs-lisp
882 (defun org-git-version ()
883 "The Git version of org-mode.
884 Inserted by installing org-mode or when a release is made."
886 (let ((git-repo (expand-file-name
887 "straight/repos/org/" user-emacs-directory)))
894 (defun org-release ()
895 "The release version of org-mode.
896 Inserted by installing org-mode or when a release is made."
898 (let ((git-repo (expand-file-name
899 "straight/repos/org/" user-emacs-directory)))
901 (string-remove-prefix
908 (provide 'org-version)
911 We will use the =org-plus-contrib= package to get the whole deal:
913 #+begin_src emacs-lisp
914 (straight-use-package 'org-plus-contrib)
917 And here's where my actual Org configurations begin:
919 #+begin_src emacs-lisp
923 (setq org-src-tab-acts-natively t
924 org-src-preserve-indentation nil
925 org-edit-src-content-indentation 0
926 org-email-link-description-format "Email %c: %s" ; %.30s
927 org-highlight-latex-and-related '(entities)
928 org-use-speed-commands t
929 org-startup-folded 'content
930 org-catch-invisible-edits 'show-and-error
932 (add-to-list 'org-structure-template-alist '("L" . "src emacs-lisp") t)
933 :bind (:map org-mode-map ("M-L" . org-insert-last-stored-link))
934 :hook ((org-mode . org-indent-mode)
935 (org-mode . auto-fill-mode)
936 (org-mode . flyspell-mode))
938 (org-latex-packages-alist '(("" "listings") ("" "color")))
940 '(org-block-begin-line ((t (:foreground "#5a5b5a" :background "#1d1f21"))))
941 '(org-block ((t (:background "#1d1f21"))))
942 '(org-latex-and-related ((t (:foreground "#b294bb")))))
944 (use-feature ox-latex
947 (setq org-latex-listings 'listings
948 ;; org-latex-prefer-user-labels t
950 (add-to-list 'org-latex-classes
951 '("IEEEtran" "\\documentclass[11pt]{IEEEtran}"
952 ("\\section{%s}" . "\\section*{%s}")
953 ("\\subsection{%s}" . "\\subsection*{%s}")
954 ("\\subsubsection{%s}" . "\\subsubsection*{%s}")
955 ("\\paragraph{%s}" . "\\paragraph*{%s}")
956 ("\\subparagraph{%s}" . "\\subparagraph*{%s}"))
958 (require 'ox-beamer))
961 **** asynchronous tangle
963 =a/async-babel-tangle= is a function closely inspired by [[https://github.com/dieggsy/dotfiles/tree/cc10edf7701958eff1cd94d4081da544d882a28c/emacs.d#dotfiles][dieggsy's
964 d/async-babel-tangle]] which uses [[https://github.com/jwiegley/emacs-async][async]] to asynchronously tangle an org
967 #+begin_src emacs-lisp
968 (with-eval-after-load 'org
969 (defvar a/show-async-tangle-results nil
970 "Keep *emacs* async buffers around for later inspection.")
972 (defvar a/show-async-tangle-time nil
973 "Show the time spent tangling the file.")
975 (defvar a/async-tangle-post-compile
976 (when a/byte-compiled-init "make build-init")
977 "If non-nil, pass to `compile' after successful tangle.")
979 ;; TODO: look into why directly byte-compiling init.el causes a
980 ;; number of problems, including magit-status not loading (busy
982 (defvar a/async-tangle-byte-recompile nil
983 "If non-nil, byte-recompile the file on successful tangle.")
985 (defun a/async-babel-tangle ()
986 "Tangle org file asynchronously."
988 (let* ((file-tangle-start-time (current-time))
989 (file (buffer-file-name))
990 (file-nodir (file-name-nondirectory file))
991 ;; (async-quiet-switch "-q")
992 (file-noext (file-name-sans-extension file)))
996 (org-babel-tangle-file ,file))
997 (unless a/show-async-tangle-results
1001 ;; (setq byte-compile-warnings '(not noruntime unresolved))
1002 (message "Tangled %s%s"
1004 (if a/show-async-tangle-time
1006 (float-time (time-subtract (current-time)
1007 ',file-tangle-start-time)))
1009 (when a/async-tangle-post-compile
1010 (compile a/async-tangle-post-compile))
1011 (when a/async-tangle-byte-recompile
1012 (byte-recompile-file (concat ,file-noext ".el"))))
1013 (message "Tangling %s failed" ,file-nodir))))))))
1016 'safe-local-variable-values
1017 '(eval add-hook 'after-save-hook #'a/async-babel-tangle 'append 'local))
1020 *** [[https://magit.vc/][Magit]]
1023 It's Magit! A Git porcelain inside Emacs.
1026 Not just how I do git, but /the/ way to do git.
1028 #+begin_src emacs-lisp
1031 :bind (("C-x g" . magit-status)
1032 ("s-g s" . magit-status)
1033 ("s-g l" . magit-log-buffer-file))
1035 (magit-add-section-hook 'magit-status-sections-hook
1036 'magit-insert-modules
1037 'magit-insert-stashes
1040 magit-repository-directories '(("~/.emacs.d/" . 0)
1041 ("~/src/git/" . 1)))
1042 (nconc magit-section-initial-visibility-alist
1043 '(([unpulled status] . show)
1044 ([unpushed status] . show)))
1045 :custom-face (magit-diff-file-heading ((t (:weight normal)))))
1050 Recently opened files.
1052 #+begin_src emacs-lisp
1053 (use-feature recentf
1056 (add-to-list 'recentf-exclude "^/\\(?:ssh\\|su\\|sudo\\)?:")
1057 (setq recentf-max-saved-items 40))
1063 A smart M-x enhancement for Emacs.
1066 Mostly because =counsel= needs it to remember history.
1068 #+begin_src emacs-lisp
1072 *** [[https://github.com/abo-abo/swiper][Ivy]] (and friends)
1075 Ivy - a generic completion frontend for Emacs, Swiper - isearch with
1076 an overview, and more. Oh, man!
1079 There's no way I could top that, so I won't attempt to.
1083 #+begin_src emacs-lisp
1087 (:map ivy-minibuffer-map
1088 ([escape] . keyboard-escape-quit)
1089 ([S-up] . ivy-previous-history-element)
1090 ([S-down] . ivy-next-history-element)
1091 ("DEL" . ivy-backward-delete-char))
1096 ;; (ivy-minibuffer-match-face-2 ((t (:background "#e99ce8" :weight semi-bold))))
1097 ;; (ivy-minibuffer-match-face-3 ((t (:background "#bbbbff" :weight semi-bold))))
1098 ;; (ivy-minibuffer-match-face-4 ((t (:background "#ffbbff" :weight semi-bold))))
1104 #+begin_src emacs-lisp
1107 :bind (("C-s" . swiper)
1109 ("C-S-s" . isearch-forward)))
1114 #+begin_src emacs-lisp
1115 (use-package counsel
1117 :bind (([remap execute-extended-command] . counsel-M-x)
1118 ([remap find-file] . counsel-find-file)
1119 ("s-r" . counsel-recentf)
1120 ("C-c x" . counsel-M-x)
1121 ("C-c f ." . counsel-find-file)
1122 :map minibuffer-local-map
1123 ("C-r" . counsel-minibuffer-history))
1126 (defalias 'locate #'counsel-locate))
1131 #+begin_src emacs-lisp
1135 :bind ("C-c a s e" . eshell)
1137 (eval-when-compile (defvar eshell-prompt-regexp))
1138 (defun a/eshell-quit-or-delete-char (arg)
1140 (if (and (eolp) (looking-back eshell-prompt-regexp nil))
1141 (eshell-life-is-too-much)
1144 (defun a/eshell-clear ()
1146 (let ((inhibit-read-only t))
1148 (eshell-send-input))
1150 (defun a/eshell-setup ()
1151 (make-local-variable 'company-idle-delay)
1152 (defvar company-idle-delay)
1153 (setq company-idle-delay nil)
1154 (bind-keys :map eshell-mode-map
1155 ("C-d" . a/eshell-quit-or-delete-char)
1156 ("C-S-l" . a/eshell-clear)
1157 ("M-r" . counsel-esh-history)
1158 ([tab] . company-complete)))
1160 :hook (eshell-mode . a/eshell-setup)
1162 (eshell-hist-ignoredups t)
1163 (eshell-input-filter 'eshell-input-filter-initial-space))
1168 #+begin_src emacs-lisp
1169 (use-feature ibuffer
1171 (("C-x C-b" . ibuffer-other-window)
1172 :map ibuffer-mode-map
1173 ("P" . ibuffer-backward-filter-group)
1174 ("N" . ibuffer-forward-filter-group)
1175 ("M-p" . ibuffer-do-print)
1176 ("M-n" . ibuffer-do-shell-command-pipe-replace))
1178 ;; Use human readable Size column instead of original one
1179 (define-ibuffer-column size-h
1180 (:name "Size" :inline t)
1182 ((> (buffer-size) 1000000) (format "%7.1fM" (/ (buffer-size) 1000000.0)))
1183 ((> (buffer-size) 100000) (format "%7.0fk" (/ (buffer-size) 1000.0)))
1184 ((> (buffer-size) 1000) (format "%7.1fk" (/ (buffer-size) 1000.0)))
1185 (t (format "%8d" (buffer-size)))))
1187 (ibuffer-saved-filter-groups
1189 ("dired" (mode . dired-mode))
1190 ("org" (mode . org-mode))
1193 (mode . gnus-group-mode)
1194 (mode . gnus-summary-mode)
1195 (mode . gnus-article-mode)
1196 ;; not really, but...
1197 (mode . message-mode)))
1206 (mode . eshell-mode)
1208 (mode . term-mode)))
1211 (mode . python-mode)
1215 (mode . emacs-lisp-mode)
1216 (mode . scheme-mode)
1217 (mode . haskell-mode)
1218 (mode . lean-mode)))
1221 (name . "^\\*scratch\\*$")
1222 (name . "^\\*Messages\\*$")))
1223 ("erc" (mode . erc-mode)))))
1225 '((mark modified read-only locked " "
1226 (name 18 18 :left :elide)
1228 (size-h 9 -1 :right)
1230 (mode 16 16 :left :elide)
1231 " " filename-and-process)
1235 :hook (ibuffer . (lambda () (ibuffer-switch-to-saved-filter-groups "default"))))
1240 #+begin_src emacs-lisp
1241 (use-feature outline
1242 :hook (prog-mode . outline-minor-mode)
1245 outline-minor-mode-map
1246 ("<s-tab>" . outline-toggle-children)
1247 ("M-p" . outline-previous-visible-heading)
1248 ("M-n" . outline-next-visible-heading)
1249 :prefix-map a/outline-prefix-map
1251 ("TAB" . outline-toggle-children)
1252 ("a" . outline-hide-body)
1253 ("H" . outline-hide-body)
1254 ("S" . outline-show-all)
1255 ("h" . outline-hide-subtree)
1256 ("s" . outline-show-subtree)))
1261 #+begin_src emacs-lisp
1262 (use-feature ls-lisp
1263 :custom (ls-lisp-dirs-first t))
1267 (setq dired-listing-switches "-alh"
1268 ls-lisp-use-insert-directory-program nil)
1270 ;; easily diff 2 marked files
1271 ;; https://oremacs.com/2017/03/18/dired-ediff/
1272 (defun dired-ediff-files ()
1274 (require 'dired-aux)
1275 (defvar ediff-after-quit-hook-internal)
1276 (let ((files (dired-get-marked-files))
1277 (wnd (current-window-configuration)))
1278 (if (<= (length files) 2)
1279 (let ((file1 (car files))
1280 (file2 (if (cdr files)
1284 (dired-dwim-target-directory)))))
1285 (if (file-newer-than-file-p file1 file2)
1286 (ediff-files file2 file1)
1287 (ediff-files file1 file2))
1288 (add-hook 'ediff-after-quit-hook-internal
1290 (setq ediff-after-quit-hook-internal nil)
1291 (set-window-configuration wnd))))
1292 (error "no more than 2 files should be marked"))))
1293 :bind (:map dired-mode-map
1294 ("b" . dired-up-directory)
1295 ("e" . dired-ediff-files)
1296 ("E" . dired-toggle-read-only)
1297 ("\\" . dired-hide-details-mode)
1300 (a/dired-start-process "zathura"))))
1301 :hook (dired-mode . dired-hide-details-mode))
1306 #+begin_src emacs-lisp
1309 (temp-buffer-resize-mode)
1310 (setq help-window-select t))
1315 #+begin_src emacs-lisp
1318 (add-to-list 'tramp-default-proxies-alist '(nil "\\`root\\'" "/ssh:%h:"))
1319 (add-to-list 'tramp-default-proxies-alist '("localhost" nil nil))
1320 (add-to-list 'tramp-default-proxies-alist
1321 (list (regexp-quote (system-name)) nil nil)))
1326 #+begin_src emacs-lisp
1328 :config (dash-enable-font-lock))
1338 Highlight uncommitted changes in the left fringe.
1340 #+begin_src emacs-lisp
1341 (use-package diff-hl
1343 (setq diff-hl-draw-borders nil)
1344 (global-diff-hl-mode)
1345 :hook (magit-post-refresh . diff-hl-magit-post-refresh))
1350 Display Lisp objects at point in the echo area.
1352 #+begin_src emacs-lisp
1354 :when (version< "25" emacs-version)
1355 :config (global-eldoc-mode))
1360 Highlight matching parens.
1362 #+begin_src emacs-lisp
1365 :config (show-paren-mode))
1368 ** simple (for column numbers)
1370 #+begin_src emacs-lisp
1372 :config (column-number-mode))
1377 Save minibuffer history.
1379 #+begin_src emacs-lisp
1380 (use-feature savehist
1381 :config (savehist-mode))
1386 Automatically save place in each file.
1388 #+begin_src emacs-lisp
1389 (use-feature saveplace
1390 :when (version< "25" emacs-version)
1391 :config (save-place-mode))
1396 #+begin_src emacs-lisp
1397 (use-feature prog-mode
1398 :config (global-prettify-symbols-mode)
1399 (defun indicate-buffer-boundaries-left ()
1400 (setq indicate-buffer-boundaries 'left))
1401 (add-hook 'prog-mode-hook #'indicate-buffer-boundaries-left))
1406 #+begin_src emacs-lisp
1407 (use-feature text-mode
1408 :hook ((text-mode . indicate-buffer-boundaries-left)
1409 (text-mode . abbrev-mode)))
1414 #+begin_src emacs-lisp
1415 (use-package company
1418 (:map company-active-map
1419 ([tab] . company-complete-common-or-cycle)
1420 ([escape] . company-abort))
1422 (company-minimum-prefix-length 1)
1423 (company-selection-wrap-around t)
1424 (company-dabbrev-char-regexp "\\sw\\|\\s_\\|[-_]")
1425 (company-dabbrev-downcase nil)
1426 (company-dabbrev-ignore-case nil)
1428 (global-company-mode t))
1433 #+begin_src emacs-lisp
1434 (use-package flycheck
1436 :hook (prog-mode . flycheck-mode)
1438 (:map flycheck-mode-map
1439 ("M-P" . flycheck-previous-error)
1440 ("M-N" . flycheck-next-error))
1442 ;; Use the load-path from running Emacs when checking elisp files
1443 (setq flycheck-emacs-lisp-load-path 'inherit)
1445 ;; Only flycheck when I actually save the buffer
1446 (setq flycheck-check-syntax-automatically '(mode-enabled save)))
1448 ;; http://endlessparentheses.com/ispell-and-apostrophes.html
1452 ;; ’ can be part of a word
1453 (setq ispell-local-dictionary-alist
1454 `((nil "[[:alpha:]]" "[^[:alpha:]]"
1455 "['\x2019]" nil ("-B") nil utf-8)))
1456 ;; don't send ’ to the subprocess
1457 (defun endless/replace-apostrophe (args)
1458 (cons (replace-regexp-in-string
1461 (advice-add #'ispell-send-string :filter-args
1462 #'endless/replace-apostrophe)
1464 ;; convert ' back to ’ from the subprocess
1465 (defun endless/replace-quote (args)
1466 (if (not (derived-mode-p 'org-mode))
1468 (cons (replace-regexp-in-string
1471 (advice-add #'ispell-parse-output :filter-args
1472 #'endless/replace-quote))
1477 :CUSTOM_ID: programming-modes
1482 #+begin_src emacs-lisp
1483 (use-feature lisp-mode
1485 (add-hook 'emacs-lisp-mode-hook 'outline-minor-mode)
1486 (add-hook 'emacs-lisp-mode-hook 'reveal-mode)
1487 (defun indent-spaces-mode ()
1488 (setq indent-tabs-mode nil))
1489 (add-hook 'lisp-interaction-mode-hook #'indent-spaces-mode))
1492 ** [[http://alloytools.org][Alloy]] (with [[https://github.com/dwwmmn/alloy-mode][alloy-mode]])
1494 #+begin_src emacs-lisp
1495 (use-package alloy-mode
1496 :straight (:host github :repo "dwwmmn/alloy-mode")
1497 :config (setq alloy-basic-offset 2))
1500 ** [[https://coq.inria.fr][Coq]] (with [[https://github.com/ProofGeneral/PG][Proof General]])
1502 #+begin_src emacs-lisp
1503 (use-package proof-site ; Proof General
1504 :straight proof-general)
1507 ** [[https://leanprover.github.io][Lean]] (with [[https://github.com/leanprover/lean-mode][lean-mode]])
1509 #+begin_src emacs-lisp
1510 (eval-when-compile (defvar lean-mode-map))
1511 (use-package lean-mode
1513 :bind (:map lean-mode-map
1514 ("S-SPC" . company-complete))
1516 (require 'lean-input)
1517 (setq default-input-method "Lean"
1518 lean-input-tweak-all '(lean-input-compose
1519 (lean-input-prepend "/")
1520 (lean-input-nonempty))
1521 lean-input-user-translations '(("/" "/")))
1527 *** [[https://github.com/haskell/haskell-mode][haskell-mode]]
1529 #+begin_src emacs-lisp
1530 (use-package haskell-mode
1532 (setq haskell-indentation-layout-offset 4
1533 haskell-indentation-left-offset 4
1534 flycheck-checker 'haskell-hlint
1535 flycheck-disabled-checkers '(haskell-stack-ghc haskell-ghc)))
1538 *** [[https://github.com/jyp/dante][dante]]
1540 #+begin_src emacs-lisp
1543 :commands dante-mode
1544 :hook (haskell-mode . dante-mode))
1547 *** [[https://github.com/mpickering/hlint-refactor-mode][hlint-refactor]]
1549 Emacs bindings for [[https://github.com/ndmitchell/hlint][hlint]]'s refactor option. This requires the refact
1550 executable from [[https://github.com/mpickering/apply-refact][apply-refact]].
1552 #+begin_src emacs-lisp
1553 (use-package hlint-refactor
1555 :bind (:map hlint-refactor-mode-map
1556 ("C-c l b" . hlint-refactor-refactor-buffer)
1557 ("C-c l r" . hlint-refactor-refactor-at-point))
1558 :hook (haskell-mode . hlint-refactor-mode))
1561 *** [[https://github.com/flycheck/flycheck-haskell][flycheck-haskell]]
1563 #+begin_src emacs-lisp
1564 (use-package flycheck-haskell
1565 :after haskell-mode)
1568 *** [[https://github.com/ndmitchell/hlint/blob/20e116a043f2073c57b17b24ae6364b5e433ba7e/data/hs-lint.el][hs-lint.el]]
1570 :header-args+: :tangle lisp/hs-lint.el :mkdirp yes
1573 Currently using =flycheck-haskell= with the =haskell-hlint= checker
1576 #+begin_src emacs-lisp :tangle no
1577 ;;; hs-lint.el --- minor mode for HLint code checking
1579 ;; Copyright 2009 (C) Alex Ott
1581 ;; Author: Alex Ott <alexott@gmail.com>
1582 ;; Keywords: haskell, lint, HLint
1584 ;; Status: distributed under terms of GPL2 or above
1586 ;; Typical message from HLint looks like:
1588 ;; /Users/ott/projects/lang-exp/haskell/test.hs:52:1: Eta reduce
1590 ;; count1 p l = length (filter p l)
1592 ;; count1 p = length . filter p
1597 (defgroup hs-lint nil
1598 "Run HLint as inferior of Emacs, parse error messages."
1602 (defcustom hs-lint-command "hlint"
1603 "The default hs-lint command for \\[hlint]."
1607 (defcustom hs-lint-save-files t
1608 "Save modified files when run HLint or no (ask user)"
1612 (defcustom hs-lint-replace-with-suggestions nil
1613 "Replace user's code with suggested replacements"
1617 (defcustom hs-lint-replace-without-ask nil
1618 "Replace user's code with suggested replacements automatically"
1622 (defun hs-lint-process-setup ()
1623 "Setup compilation variables and buffer for `hlint'."
1624 (run-hooks 'hs-lint-setup-hook))
1626 ;; regex for replace suggestions
1628 ;; ^\(.*?\):\([0-9]+\):\([0-9]+\): .*
1634 (defvar hs-lint-regex
1635 "^\\(.*?\\):\\([0-9]+\\):\\([0-9]+\\): .*[\n\C-m]Found:[\n\C-m]\\s +\\(.*\\)[\n\C-m]Why not:[\n\C-m]\\s +\\(.*\\)[\n\C-m]"
1636 "Regex for HLint messages")
1638 (defun make-short-string (str maxlen)
1639 (if (< (length str) maxlen)
1641 (concat (substring str 0 (- maxlen 3)) "...")))
1643 (defun hs-lint-replace-suggestions ()
1644 "Perform actual replacement of suggestions"
1645 (goto-char (point-min))
1646 (while (re-search-forward hs-lint-regex nil t)
1647 (let* ((fname (match-string 1))
1648 (fline (string-to-number (match-string 2)))
1649 (old-code (match-string 4))
1650 (new-code (match-string 5))
1651 (msg (concat "Replace '" (make-short-string old-code 30)
1652 "' with '" (make-short-string new-code 30) "'"))
1658 (switch-to-buffer (get-file-buffer fname))
1659 (goto-char (point-min))
1660 (forward-line (1- fline))
1662 (setf bline (point))
1663 (when (or hs-lint-replace-without-ask
1666 (setf eline (point))
1668 (setf old-code (regexp-quote old-code))
1669 (while (string-match "\\\\ " old-code spos)
1670 (setf new-old-code (concat new-old-code
1671 (substring old-code spos (match-beginning 0))
1673 (setf spos (match-end 0)))
1674 (setf new-old-code (concat new-old-code (substring old-code spos)))
1675 (remove-text-properties bline eline '(composition nil))
1676 (when (re-search-forward new-old-code eline t)
1677 (replace-match new-code nil t)))))))
1679 (defun hs-lint-finish-hook (buf msg)
1680 "Function, that is executed at the end of HLint execution"
1681 (if hs-lint-replace-with-suggestions
1682 (hs-lint-replace-suggestions)
1685 (define-compilation-mode hs-lint-mode "HLint"
1686 "Mode for check Haskell source code."
1687 (set (make-local-variable 'compilation-process-setup-function)
1688 'hs-lint-process-setup)
1689 (set (make-local-variable 'compilation-disable-input) t)
1690 (set (make-local-variable 'compilation-scroll-output) nil)
1691 (set (make-local-variable 'compilation-finish-functions)
1692 (list 'hs-lint-finish-hook))
1696 "Run HLint for current buffer with haskell source"
1698 (save-some-buffers hs-lint-save-files)
1699 (compilation-start (concat hs-lint-command " \"" buffer-file-name "\"")
1703 ;;; hs-lint.el ends here
1706 #+begin_src emacs-lisp :tangle no
1707 (use-package hs-lint
1709 :bind (:map haskell-mode-map
1710 ("C-c l l" . hs-lint)))
1717 #+begin_src emacs-lisp
1718 (use-package sgml-mode
1720 (setq sgml-basic-offset 2))
1725 #+begin_src emacs-lisp
1726 (use-package css-mode
1728 (setq css-indent-offset 2))
1733 #+begin_src emacs-lisp
1734 (use-package web-mode
1738 web-mode-code-indent-offset
1739 web-mode-css-indent-offset
1740 web-mode-markup-indent-offset))
1745 #+begin_src emacs-lisp
1746 (use-package emmet-mode
1747 :after (:any web-mode css-mode sgml-mode)
1748 :bind* (("C-)" . emmet-next-edit-point)
1749 ("C-(" . emmet-prev-edit-point))
1751 (unbind-key "C-j" emmet-mode-keymap)
1752 (setq emmet-move-cursor-between-quotes t)
1753 :hook (web-mode css-mode html-mode sgml-mode))
1758 *** COMMENT meghanada
1760 #+begin_src emacs-lisp
1761 (use-package meghanada
1763 (:map meghanada-mode-map
1764 (("C-M-o" . meghanada-optimize-import)
1765 ("C-M-t" . meghanada-import-all)))
1766 :hook (java-mode . meghanada-mode))
1769 *** COMMENT lsp-java
1787 #+begin_src emacs-lisp
1788 (use-package treemacs
1789 :config (setq treemacs-never-persist t))
1791 (use-package yasnippet
1793 ;; (yas-global-mode)
1796 (use-package lsp-mode
1797 :init (setq lsp-eldoc-render-all nil
1798 lsp-highlight-symbol-at-point nil)
1803 (use-package company-lsp
1806 (setq company-lsp-cache-candidates t
1807 company-lsp-async t))
1811 (setq lsp-ui-sideline-update-mode 'point))
1813 (use-package lsp-java
1815 (add-hook 'java-mode-hook
1817 (setq-local company-backends (list 'company-lsp))))
1819 (add-hook 'java-mode-hook 'lsp-java-enable)
1820 (add-hook 'java-mode-hook 'flycheck-mode)
1821 (add-hook 'java-mode-hook 'company-mode)
1822 (add-hook 'java-mode-hook 'lsp-ui-mode))
1824 (use-package dap-mode
1830 (use-package dap-java
1833 (use-package lsp-java-treemacs
1839 #+begin_src emacs-lisp
1841 :bind (:map eclim-mode-map ("S-SPC" . company-complete))
1842 :hook ((java-mode . eclim-mode)
1843 (eclim-mode . (lambda ()
1844 (make-local-variable 'company-idle-delay)
1845 (defvar company-idle-delay)
1846 ;; (setq company-idle-delay 0.7)
1847 (setq company-idle-delay nil))))
1849 (eclim-auto-save nil)
1850 ;; (eclimd-default-workspace "~/src/eclipse-workspace-exp")
1851 (eclim-executable "~/.p2/pool/plugins/org.eclim_2.8.0/bin/eclim")
1852 (eclim-eclipse-dirs '("~/usr/eclipse/dsl-2018-09/eclipse")))
1857 #+begin_src emacs-lisp
1858 (use-package geiser)
1860 (use-feature geiser-guile
1862 (setq geiser-guile-load-path "~/src/git/guix"))
1867 #+begin_src emacs-lisp
1871 * Emacs enhancements
1873 :CUSTOM_ID: emacs-enhancements
1878 #+begin_src emacs-lisp
1880 :config (setq Man-width 80))
1883 ** [[https://github.com/justbur/emacs-which-key][which-key]]
1886 Emacs package that displays available keybindings in popup
1889 #+begin_src emacs-lisp
1890 (use-package which-key
1893 (which-key-add-key-based-replacements
1894 ;; prefixes for global prefixes and minor modes
1898 "C-c 8 -" "typo/dashes"
1899 "C-c 8 <" "typo/left-brackets"
1900 "C-c 8 >" "typo/right-brackets"
1902 "C-x a" "abbrev/expand"
1903 "C-x r" "rectangle/register/bookmark"
1904 "C-x v" "version control"
1905 ;; prefixes for my personal bindings
1906 "C-c a" "applications"
1909 "C-c p" "package-management"
1910 "C-c p e" "package-management/epkg"
1911 "C-c p s" "straight.el"
1914 "C-c c" "compile-and-comments"
1919 "C-c m" "multiple-cursors"
1920 "C-c P" "projectile"
1921 "C-c P s" "projectile/search"
1922 "C-c P x" "projectile/execute"
1923 "C-c P 4" "projectile/other-window"
1929 ;; prefixes for major modes
1930 (which-key-add-major-mode-key-based-replacements 'message-mode
1932 (which-key-add-major-mode-key-based-replacements 'org-mode
1933 "C-c C-v" "org-babel")
1934 (which-key-add-major-mode-key-based-replacements 'web-mode
1935 "C-c C-a" "web/attributes"
1936 "C-c C-b" "web/blocks"
1938 "C-c C-e" "web/element"
1939 "C-c C-t" "web/tags")
1943 (which-key-add-column-padding 5)
1944 (which-key-max-description-length 32))
1949 #+begin_src emacs-lisp
1950 (add-to-list 'custom-theme-load-path "~/.emacs.d/lisp")
1951 (load-theme 'tangomod t)
1956 #+begin_src emacs-lisp
1957 (use-package smart-mode-line
1958 :commands (sml/apply-theme)
1966 #+begin_src emacs-lisp
1967 (use-package doom-themes)
1970 ** theme helper functions
1972 #+begin_src emacs-lisp
1973 (defvar a/org-mode-font-lock-keywords
1974 '(("[ \t]*\\(#\\+\\(BEGIN\\|END\\|begin\\|end\\)_\\(\\S-+\\)\\)[ \t]*\\([^\n:]*\\)"
1975 (1 '(:foreground "#5a5b5a" :background "#292b2b") t) ; directive
1976 (3 '(:foreground "#81a2be" :background "#292b2b") t) ; kind
1977 (4 '(:foreground "#c5c8c6") t)))) ; title
1979 (defun a/lights-on ()
1980 "Enable my favourite light theme."
1982 (mapc #'disable-theme custom-enabled-themes)
1983 (load-theme 'tangomod t)
1984 (sml/apply-theme 'automatic)
1985 (font-lock-remove-keywords
1986 'org-mode a/org-mode-font-lock-keywords))
1988 (defun a/lights-off ()
1991 (mapc #'disable-theme custom-enabled-themes)
1992 (load-theme 'doom-tomorrow-night t)
1993 (sml/apply-theme 'automatic)
1994 (font-lock-add-keywords
1995 'org-mode a/org-mode-font-lock-keywords t))
1998 ("s-t d" . a/lights-off)
1999 ("s-t l" . a/lights-on))
2002 ** [[https://github.com/bbatsov/crux][crux]]
2004 #+begin_src emacs-lisp
2005 (use-package crux ; results in Waiting for git... [2 times]
2007 :bind (("C-c b k" . crux-kill-other-buffers)
2008 ("C-c d" . crux-duplicate-current-line-or-region)
2009 ("C-c D" . crux-duplicate-and-comment-current-line-or-region)
2010 ("C-c f c" . crux-copy-file-preserve-attributes)
2011 ("C-c f d" . crux-delete-file-and-buffer)
2012 ("C-c f r" . crux-rename-file-and-buffer)
2013 ("C-c j" . crux-top-join-line)
2014 ("C-S-j" . crux-top-join-line)))
2017 ** [[https://github.com/alezost/mwim.el][mwim]]
2019 #+begin_src emacs-lisp
2021 :bind (("C-a" . mwim-beginning-of-code-or-line)
2022 ("C-e" . mwim-end-of-code-or-line)
2023 ("<home>" . mwim-beginning-of-line-or-code)
2024 ("<end>" . mwim-end-of-line-or-code)))
2029 #+begin_src emacs-lisp
2030 (use-package projectile
2031 :bind-keymap ("C-c P" . projectile-command-map)
2035 (defun my-projectile-invalidate-cache (&rest _args)
2036 ;; ignore the args to `magit-checkout'
2037 (projectile-invalidate-cache nil))
2039 (eval-after-load 'magit-branch
2041 (advice-add 'magit-checkout
2042 :after #'my-projectile-invalidate-cache)
2043 (advice-add 'magit-branch-and-checkout
2044 :after #'my-projectile-invalidate-cache)))
2045 :custom (projectile-completion-system 'ivy))
2048 ** [[https://github.com/Wilfred/helpful][helpful]]
2050 #+begin_src emacs-lisp
2051 (use-package helpful
2054 (("C-S-h c" . helpful-command)
2055 ("C-S-h f" . helpful-callable) ; helpful-function
2056 ("C-S-h v" . helpful-variable)
2057 ("C-S-h k" . helpful-key)
2058 ("C-S-h p" . helpful-at-point)))
2061 ** [[https://github.com/EricCrosson/unkillable-scratch][unkillable-scratch]]
2063 Make =*scratch*= and =*Messages*= unkillable.
2065 #+begin_src emacs-lisp
2066 (use-package unkillable-scratch
2069 (unkillable-scratch 1)
2071 (unkillable-buffers '("^\\*scratch\\*$" "^\\*Messages\\*$")))
2074 ** [[https://github.com/davep/boxquote.el][boxquote.el]]
2078 | make pretty boxed quotes like this
2082 #+begin_src emacs-lisp
2083 (use-package boxquote
2086 (:prefix-map a/boxquote-prefix-map
2088 ("b" . boxquote-buffer)
2089 ("B" . boxquote-insert-buffer)
2090 ("d" . boxquote-defun)
2091 ("F" . boxquote-insert-file)
2092 ("hf" . boxquote-describe-function)
2093 ("hk" . boxquote-describe-key)
2094 ("hv" . boxquote-describe-variable)
2095 ("hw" . boxquote-where-is)
2096 ("k" . boxquote-kill)
2097 ("p" . boxquote-paragraph)
2098 ("q" . boxquote-boxquote)
2099 ("r" . boxquote-region)
2100 ("s" . boxquote-shell-command)
2101 ("t" . boxquote-text)
2102 ("T" . boxquote-title)
2103 ("u" . boxquote-unbox)
2104 ("U" . boxquote-unbox-region)
2105 ("y" . boxquote-yank)
2106 ("M-q" . boxquote-fill-paragraph)
2107 ("M-w" . boxquote-kill-ring-save)))
2110 Also see [[https://www.emacswiki.org/emacs/rebox2][rebox2]].
2114 #+begin_src emacs-lisp
2115 (use-package orgalist
2117 :hook (message-mode . orgalist-mode))
2122 #+begin_src emacs-lisp
2126 (typo-global-mode 1)
2127 :hook (text-mode . typo-mode))
2132 #+begin_src emacs-lisp
2133 (use-package hl-todo
2136 (global-hl-todo-mode))
2141 #+begin_src emacs-lisp
2142 (use-package shrink-path
2146 (defun +eshell/prompt ()
2147 (let ((base/dir (shrink-path-prompt default-directory)))
2148 (concat (propertize (car base/dir)
2149 'face 'font-lock-comment-face)
2150 (propertize (cdr base/dir)
2151 'face 'font-lock-constant-face)
2152 (propertize (+eshell--current-git-branch)
2153 'face 'font-lock-function-name-face)
2156 ;; (propertize "λ" 'face 'eshell-prompt)
2157 ;; needed for the input text to not have prompt face
2158 (propertize "λ " 'face 'default))))
2160 (defun +eshell--current-git-branch ()
2161 (let ((branch (car (loop for match in (split-string (shell-command-to-string "git branch") "\n")
2162 when (string-match "^\*" match)
2164 (if (not (eq branch nil))
2165 (concat " " (substring branch 2))
2167 (setq eshell-prompt-regexp "\\(.*\n\\)*λ "
2168 eshell-prompt-function #'+eshell/prompt))
2171 ** [[https://github.com/peterwvj/eshell-up][eshell-up]]
2173 #+begin_src emacs-lisp
2174 (use-package eshell-up
2176 :commands eshell-up)
2181 #+begin_src emacs-lisp
2182 (use-package multi-term
2184 :bind (("C-c a s m" . multi-term-dedicated-toggle)
2186 ("C-c C-j" . term-char-mode)
2188 ("C-c C-j" . term-line-mode))
2190 (setq multi-term-program "/bin/screen"
2191 ;; TODO: add separate bindings for connecting to existing
2192 ;; session vs. always creating a new one
2193 multi-term-dedicated-select-after-open-p t
2194 multi-term-dedicated-window-height 20
2195 multi-term-dedicated-max-window-height 30
2197 '(("C-c C-c" . term-interrupt-subjob)
2198 ("C-c C-e" . term-send-esc)
2200 ("C-y" . term-paste)
2201 ("M-f" . term-send-forward-word)
2202 ("M-b" . term-send-backward-word)
2203 ("M-p" . term-send-up)
2204 ("M-n" . term-send-down)
2205 ("<C-backspace>" . term-send-backward-kill-word)
2206 ("<M-DEL>" . term-send-backward-kill-word)
2207 ("M-d" . term-send-delete-word)
2208 ("M-," . term-send-raw)
2209 ("M-." . comint-dynamic-complete))
2210 term-unbind-key-alist
2211 '("C-z" "C-x" "C-c" "C-h" "C-y" "<ESC>")))
2216 #+begin_src emacs-lisp
2217 (use-package page-break-lines
2219 (global-page-break-lines-mode))
2224 #+begin_src emacs-lisp
2225 (use-package expand-region
2226 :bind ("C-=" . er/expand-region))
2231 #+begin_src emacs-lisp
2232 (use-package multiple-cursors
2234 (("C-S-<mouse-1>" . mc/add-cursor-on-click)
2235 (:prefix-map a/mc-prefix-map
2237 ("c" . mc/edit-lines)
2238 ("n" . mc/mark-next-like-this)
2239 ("p" . mc/mark-previous-like-this)
2240 ("a" . mc/mark-all-like-this))))
2245 #+begin_src emacs-lisp
2253 #+begin_src emacs-lisp
2254 (use-package yasnippet
2257 (defconst yas-verbosity-cur yas-verbosity)
2258 (setq yas-verbosity 2)
2259 (add-to-list 'yas-snippet-dirs "~/src/git/guix/etc/snippets")
2261 (setq yas-verbosity yas-verbosity-cur)
2263 (text-mode . yas-minor-mode))
2271 #+begin_src emacs-lisp
2272 (defvar a/maildir (expand-file-name "~/mail/"))
2273 (with-eval-after-load 'recentf
2274 (add-to-list 'recentf-exclude a/maildir))
2279 #+begin_src emacs-lisp
2281 a/gnus-init-file (no-littering-expand-etc-file-name "gnus")
2282 mail-user-agent 'gnus-user-agent
2283 read-mail-command 'gnus)
2286 :bind (("s-m" . gnus)
2287 ("s-M" . gnus-unplugged))
2290 gnus-select-method '(nnnil "")
2291 gnus-secondary-select-methods
2293 (nnimap-stream plain)
2294 (nnimap-address "127.0.0.1")
2295 (nnimap-server-port 143)
2296 (nnimap-authenticator plain)
2297 (nnimap-user "amin@bndl.org"))
2299 (nnimap-stream plain)
2300 (nnimap-address "127.0.0.1")
2301 (nnimap-server-port 143)
2302 (nnimap-authenticator plain)
2303 (nnimap-user "abandali@uwaterloo.ca"))
2305 (nnimap-stream plain)
2306 (nnimap-address "127.0.0.1")
2307 (nnimap-server-port 143)
2308 (nnimap-authenticator plain)
2309 (nnimap-user "abandali@csclub.uw")))
2310 gnus-message-archive-group "nnimap+amin:Sent"
2314 gnus-large-newsgroup 50
2315 gnus-home-directory (no-littering-expand-var-file-name "gnus/")
2316 gnus-directory (concat gnus-home-directory "news/")
2317 message-directory (concat gnus-home-directory "mail/")
2318 nndraft-directory (concat gnus-home-directory "drafts/")
2319 gnus-save-newsrc-file nil
2320 gnus-read-newsrc-file nil
2321 gnus-interactive-exit nil
2322 gnus-gcc-mark-as-read t)
2326 (require 'ebdb-gnus))
2328 (use-feature gnus-art
2331 gnus-visible-headers
2332 (concat gnus-visible-headers "\\|^List-Id:\\|^X-RT-Originator:\\|^User-Agent:")
2333 gnus-sorted-header-list
2334 '("^From:" "^Subject:" "^Summary:" "^Keywords:"
2335 "^Followup-To:" "^To:" "^Cc:" "X-RT-Originator"
2336 "^Newsgroups:" "List-Id:" "^Organization:"
2337 "^User-Agent:" "^Date:")
2338 ;; local-lapsed article dates
2339 ;; from https://www.emacswiki.org/emacs/GnusFormatting#toc11
2340 gnus-article-date-headers '(user-defined)
2341 gnus-article-time-format
2343 (let* ((date (format-time-string "%a, %d %b %Y %T %z" time))
2344 (local (article-make-date-line date 'local))
2345 (combined-lapsed (article-make-date-line date
2348 (string-match " (.+" combined-lapsed)
2349 (match-string 0 combined-lapsed))))
2350 (concat local lapsed))))
2352 :map gnus-article-mode-map
2353 ("r" . gnus-article-reply-with-original)
2354 ("R" . gnus-article-wide-reply-with-original)
2355 ("M-L" . org-store-link)))
2357 (use-feature gnus-sum
2358 :bind (:map gnus-summary-mode-map
2359 :prefix-map a/gnus-summary-prefix-map
2361 ("r" . gnus-summary-reply)
2362 ("w" . gnus-summary-wide-reply)
2363 ("v" . gnus-summary-show-raw-article))
2366 :map gnus-summary-mode-map
2367 ("r" . gnus-summary-reply-with-original)
2368 ("R" . gnus-summary-wide-reply-with-original)
2369 ("M-L" . org-store-link))
2370 :hook (gnus-summary-mode . a/no-mouse-autoselect-window))
2372 (use-feature gnus-msg
2374 (setq gnus-posting-styles
2376 (address "amin@bndl.org")
2377 (body "\nBest,\namin\n")
2378 (eval (setq a/message-cite-say-hi t)))
2380 (address "bandali@gnu.org")
2381 (eval (set (make-local-variable 'message-user-fqdn) "fencepost.gnu.org")))
2382 ((header "subject" "ThankCRM")
2383 (to "webmasters-comment@gnu.org")
2384 (body "\nAdded to 2019supporters.html.\n\nMoving to campaigns.\n\n-amin\n")
2385 (eval (setq a/message-cite-say-hi nil)))
2386 ("nnimap\\+uwaterloo:.*"
2387 (address "abandali@uwaterloo.ca")
2388 (gcc "\"nnimap+uwaterloo:Sent Items\""))
2389 ("nnimap\\+csclub:.*"
2390 (address "abandali@csclub.uwaterloo.ca")
2391 (gcc "nnimap+csclub:Sent")))))
2393 (use-feature gnus-topic
2394 :hook (gnus-group-mode . gnus-topic-mode)
2395 :config (setq gnus-topic-line-format "%i[ %A: %(%{%n%}%) ]%v\n"))
2397 (use-feature gnus-agent
2399 (setq gnus-agent-synchronize-flags 'ask)
2400 :hook (gnus-group-mode . gnus-agent-mode))
2402 (use-feature gnus-group
2404 (setq gnus-permanently-visible-groups "\\(:INBOX$\\|:gnu$\\)"))
2406 (use-feature mm-decode
2408 (setq mm-discouraged-alternatives '("text/html" "text/richtext")))
2413 #+begin_src emacs-lisp
2414 (use-feature sendmail
2416 (setq sendmail-program "/usr/bin/msmtp"
2417 ;; message-sendmail-extra-arguments '("-v" "-d")
2418 mail-specify-envelope-from t
2419 mail-envelope-from 'header))
2424 #+begin_src emacs-lisp
2425 (use-feature message
2427 ;; redefine for a simplified In-Reply-To header
2428 ;; (see https://todo.sr.ht/~sircmpwn/lists.sr.ht/67)
2429 (defun message-make-in-reply-to ()
2430 "Return the In-Reply-To header for this message."
2431 (when message-reply-headers
2432 (let ((from (mail-header-from message-reply-headers))
2433 (msg-id (mail-header-id message-reply-headers)))
2437 (defconst a/message-cite-style-format "On %Y-%m-%d %l:%M %p, %N wrote:")
2438 (defconst message-cite-style-bandali
2439 '((message-cite-function 'message-cite-original)
2440 (message-citation-line-function 'message-insert-formatted-citation-line)
2441 (message-cite-reply-position 'traditional)
2442 (message-yank-prefix "> ")
2443 (message-yank-cited-prefix ">")
2444 (message-yank-empty-prefix ">")
2445 (message-citation-line-format
2446 (if a/message-cite-say-hi
2447 (concat "Hi %F,\n\n" a/message-cite-style-format)
2448 a/message-cite-style-format)))
2449 "Citation style based on Mozilla Thunderbird's. Use with message-cite-style.")
2450 (setq message-cite-style 'message-cite-style-bandali
2451 message-kill-buffer-on-exit t
2452 message-send-mail-function 'message-send-mail-with-sendmail
2453 message-sendmail-envelope-from 'header
2454 message-dont-reply-to-names
2455 "\\(\\(amin@bndl\\.org\\)\\|\\(.*@\\(aminb\\|amin\\.bndl\\)\\.org\\)\\|\\(\\(bandali\\|aminb?\\|mab\\)@gnu\\.org\\)\\|\\(a\\(min\\.\\)?bandali@uwaterloo\\.ca\\)\\|\\(abandali@csclub\\.uwaterloo\\.ca\\)\\)")
2456 (require 'company-ebdb)
2457 :hook (;; (message-setup . mml-secure-message-sign-pgpmime)
2458 (message-mode . flyspell-mode)
2459 (message-mode . (lambda ()
2460 ;; (setq fill-column 65
2461 ;; message-fill-column 65)
2462 (make-local-variable 'company-idle-delay)
2463 (setq company-idle-delay 0.2))))
2465 ;; (message-header-subject ((t (:foreground "#111" :weight semi-bold))))
2466 ;; (message-header-to ((t (:foreground "#111" :weight normal))))
2467 ;; (message-header-cc ((t (:foreground "#333" :weight normal))))
2470 (with-eval-after-load 'mml-sec
2471 (setq mml-secure-openpgp-encrypt-to-self t
2472 mml-secure-openpgp-sign-with-sender t))
2477 Convenient footnotes in =message-mode=.
2479 #+begin_src emacs-lisp
2480 (use-feature footnote
2483 ;; (setq footnote-start-tag ""
2484 ;; footnote-end-tag ""
2485 ;; footnote-style 'unicode)
2487 (:map message-mode-map
2488 :prefix-map a/footnote-prefix-map
2490 ("a" . footnote-add-footnote)
2491 ("b" . footnote-back-to-message)
2492 ("c" . footnote-cycle-style)
2493 ("d" . footnote-delete-footnote)
2494 ("g" . footnote-goto-footnote)
2495 ("r" . footnote-renumber-footnotes)
2496 ("s" . footnote-set-style)))
2501 #+begin_src emacs-lisp
2503 :straight (:host github :repo "girzel/ebdb")
2505 :bind (:map gnus-group-mode-map ("e" . ebdb))
2507 (setq ebdb-sources (no-littering-expand-var-file-name "ebdb"))
2508 (with-eval-after-load 'swiper
2509 (add-to-list 'swiper-font-lock-exclude 'ebdb-mode t)))
2511 (use-feature ebdb-com
2514 ;; (use-package ebdb-complete
2517 ;; (ebdb-complete-enable))
2519 (use-package company-ebdb
2521 (defun company-ebdb--post-complete (_) nil))
2523 (use-feature ebdb-gnus
2526 (ebdb-gnus-window-configuration
2529 (summary 0.25 point)
2532 (ebdb-gnus 0.3))))))
2534 (use-feature ebdb-mua
2536 ;; :custom (ebdb-mua-pop-up nil)
2539 ;; (use-package ebdb-message
2543 ;; (use-package ebdb-vcard
2549 #+begin_src emacs-lisp
2550 (use-package message-x)
2553 #+begin_src emacs-lisp :tangle no
2554 (use-package message-x
2556 (message-x-completion-alist
2558 (("\\([rR]esent-\\|[rR]eply-\\)?[tT]o:\\|[bB]?[cC][cC]:" . gnus-harvest-find-address)
2561 (quote message-newgroups-header-regexp))
2562 message-newgroups-header-regexp message-newsgroups-header-regexp)
2563 . message-expand-group)))))
2566 ** COMMENT gnus-harvest
2568 #+begin_src emacs-lisp
2569 (use-package gnus-harvest
2570 :commands gnus-harvest-install
2573 (if (featurep 'message-x)
2574 (gnus-harvest-install 'message-x)
2575 (gnus-harvest-install)))
2585 #+begin_src emacs-lisp
2587 :straight (:host nil :repo "https://git.bndl.org/amin/znc.el")
2588 :bind (("C-c a e e" . znc-erc)
2589 ("C-c a e a" . znc-all))
2591 (let ((pwd (let ((auth (auth-source-search :host "znca")))
2593 ((null auth) (error "Couldn't find znca's authinfo"))
2594 (t (funcall (plist-get (car auth) :secret)))))))
2596 `(("znc.bndl.org" 1337 t
2597 ((freenode "amin/freenode" ,pwd)))
2598 ("znc.bndl.org" 1337 t
2599 ((moznet "amin/moznet" ,pwd)))))))
2604 :CUSTOM_ID: blogging
2607 ** [[https://ox-hugo.scripter.co][ox-hugo]]
2609 #+begin_src emacs-lisp
2610 (use-package ox-hugo
2615 * Post initialization
2617 :CUSTOM_ID: post-initialization
2620 Display how long it took to load the init file.
2622 #+begin_src emacs-lisp
2623 (message "Loading %s...done (%.3fs)" user-init-file
2624 (float-time (time-subtract (current-time)
2625 a/before-user-init-time)))
2633 #+begin_src emacs-lisp :comments none
2634 ;;; init.el ends here
2637 * COMMENT Local Variables :ARCHIVE:
2639 # eval: (add-hook 'after-save-hook #'a/async-babel-tangle 'append 'local)
2640 # eval: (when (featurep 'typo (typo-mode -1)))