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\\*$"))))))
1224 '((mark modified read-only locked " "
1225 (name 18 18 :left :elide)
1227 (size-h 9 -1 :right)
1229 (mode 16 16 :left :elide)
1230 " " filename-and-process)
1234 :hook (ibuffer . (lambda () (ibuffer-switch-to-saved-filter-groups "default"))))
1239 #+begin_src emacs-lisp
1240 (use-feature outline
1241 :hook (prog-mode . outline-minor-mode)
1244 outline-minor-mode-map
1245 ("<s-tab>" . outline-toggle-children)
1246 ("M-p" . outline-previous-visible-heading)
1247 ("M-n" . outline-next-visible-heading)
1248 :prefix-map a/outline-prefix-map
1250 ("TAB" . outline-toggle-children)
1251 ("a" . outline-hide-body)
1252 ("H" . outline-hide-body)
1253 ("S" . outline-show-all)
1254 ("h" . outline-hide-subtree)
1255 ("s" . outline-show-subtree)))
1260 #+begin_src emacs-lisp
1261 (use-feature ls-lisp
1262 :custom (ls-lisp-dirs-first t))
1266 (setq dired-listing-switches "-alh"
1267 ls-lisp-use-insert-directory-program nil)
1269 ;; easily diff 2 marked files
1270 ;; https://oremacs.com/2017/03/18/dired-ediff/
1271 (defun dired-ediff-files ()
1273 (require 'dired-aux)
1274 (defvar ediff-after-quit-hook-internal)
1275 (let ((files (dired-get-marked-files))
1276 (wnd (current-window-configuration)))
1277 (if (<= (length files) 2)
1278 (let ((file1 (car files))
1279 (file2 (if (cdr files)
1283 (dired-dwim-target-directory)))))
1284 (if (file-newer-than-file-p file1 file2)
1285 (ediff-files file2 file1)
1286 (ediff-files file1 file2))
1287 (add-hook 'ediff-after-quit-hook-internal
1289 (setq ediff-after-quit-hook-internal nil)
1290 (set-window-configuration wnd))))
1291 (error "no more than 2 files should be marked"))))
1292 :bind (:map dired-mode-map
1293 ("b" . dired-up-directory)
1294 ("e" . dired-ediff-files)
1295 ("E" . dired-toggle-read-only)
1296 ("\\" . dired-hide-details-mode)
1299 (a/dired-start-process "zathura"))))
1300 :hook (dired-mode . dired-hide-details-mode))
1305 #+begin_src emacs-lisp
1308 (temp-buffer-resize-mode)
1309 (setq help-window-select t))
1314 #+begin_src emacs-lisp
1317 (add-to-list 'tramp-default-proxies-alist '(nil "\\`root\\'" "/ssh:%h:"))
1318 (add-to-list 'tramp-default-proxies-alist '("localhost" nil nil))
1319 (add-to-list 'tramp-default-proxies-alist
1320 (list (regexp-quote (system-name)) nil nil)))
1325 #+begin_src emacs-lisp
1327 :config (dash-enable-font-lock))
1337 Highlight uncommitted changes in the left fringe.
1339 #+begin_src emacs-lisp
1340 (use-package diff-hl
1342 (setq diff-hl-draw-borders nil)
1343 (global-diff-hl-mode)
1344 :hook (magit-post-refresh . diff-hl-magit-post-refresh))
1349 Display Lisp objects at point in the echo area.
1351 #+begin_src emacs-lisp
1353 :when (version< "25" emacs-version)
1354 :config (global-eldoc-mode))
1359 Highlight matching parens.
1361 #+begin_src emacs-lisp
1364 :config (show-paren-mode))
1367 ** simple (for column numbers)
1369 #+begin_src emacs-lisp
1371 :config (column-number-mode))
1376 Save minibuffer history.
1378 #+begin_src emacs-lisp
1379 (use-feature savehist
1380 :config (savehist-mode))
1385 Automatically save place in each file.
1387 #+begin_src emacs-lisp
1388 (use-feature saveplace
1389 :when (version< "25" emacs-version)
1390 :config (save-place-mode))
1395 #+begin_src emacs-lisp
1396 (use-feature prog-mode
1397 :config (global-prettify-symbols-mode)
1398 (defun indicate-buffer-boundaries-left ()
1399 (setq indicate-buffer-boundaries 'left))
1400 (add-hook 'prog-mode-hook #'indicate-buffer-boundaries-left))
1405 #+begin_src emacs-lisp
1406 (use-feature text-mode
1407 :hook ((text-mode . indicate-buffer-boundaries-left)
1408 (text-mode . abbrev-mode)))
1413 #+begin_src emacs-lisp
1414 (use-package company
1417 (:map company-active-map
1418 ([tab] . company-complete-common-or-cycle)
1419 ([escape] . company-abort))
1421 (company-minimum-prefix-length 1)
1422 (company-selection-wrap-around t)
1423 (company-dabbrev-char-regexp "\\sw\\|\\s_\\|[-_]")
1424 (company-dabbrev-downcase nil)
1425 (company-dabbrev-ignore-case nil)
1427 (global-company-mode t))
1432 #+begin_src emacs-lisp
1433 (use-package flycheck
1435 :hook (prog-mode . flycheck-mode)
1437 (:map flycheck-mode-map
1438 ("M-P" . flycheck-previous-error)
1439 ("M-N" . flycheck-next-error))
1441 ;; Use the load-path from running Emacs when checking elisp files
1442 (setq flycheck-emacs-lisp-load-path 'inherit)
1444 ;; Only flycheck when I actually save the buffer
1445 (setq flycheck-check-syntax-automatically '(mode-enabled save)))
1447 ;; http://endlessparentheses.com/ispell-and-apostrophes.html
1451 ;; ’ can be part of a word
1452 (setq ispell-local-dictionary-alist
1453 `((nil "[[:alpha:]]" "[^[:alpha:]]"
1454 "['\x2019]" nil ("-B") nil utf-8)))
1455 ;; don't send ’ to the subprocess
1456 (defun endless/replace-apostrophe (args)
1457 (cons (replace-regexp-in-string
1460 (advice-add #'ispell-send-string :filter-args
1461 #'endless/replace-apostrophe)
1463 ;; convert ' back to ’ from the subprocess
1464 (defun endless/replace-quote (args)
1465 (if (not (derived-mode-p 'org-mode))
1467 (cons (replace-regexp-in-string
1470 (advice-add #'ispell-parse-output :filter-args
1471 #'endless/replace-quote))
1476 :CUSTOM_ID: programming-modes
1481 #+begin_src emacs-lisp
1482 (use-feature lisp-mode
1484 (add-hook 'emacs-lisp-mode-hook 'outline-minor-mode)
1485 (add-hook 'emacs-lisp-mode-hook 'reveal-mode)
1486 (defun indent-spaces-mode ()
1487 (setq indent-tabs-mode nil))
1488 (add-hook 'lisp-interaction-mode-hook #'indent-spaces-mode))
1491 ** [[http://alloytools.org][Alloy]] (with [[https://github.com/dwwmmn/alloy-mode][alloy-mode]])
1493 #+begin_src emacs-lisp
1494 (use-package alloy-mode
1495 :straight (:host github :repo "dwwmmn/alloy-mode")
1496 :config (setq alloy-basic-offset 2))
1499 ** [[https://coq.inria.fr][Coq]] (with [[https://github.com/ProofGeneral/PG][Proof General]])
1501 #+begin_src emacs-lisp
1502 (use-package proof-site ; Proof General
1503 :straight proof-general)
1506 ** [[https://leanprover.github.io][Lean]] (with [[https://github.com/leanprover/lean-mode][lean-mode]])
1508 #+begin_src emacs-lisp
1509 (eval-when-compile (defvar lean-mode-map))
1510 (use-package lean-mode
1512 :bind (:map lean-mode-map
1513 ("S-SPC" . company-complete))
1515 (require 'lean-input)
1516 (setq default-input-method "Lean"
1517 lean-input-tweak-all '(lean-input-compose
1518 (lean-input-prepend "/")
1519 (lean-input-nonempty))
1520 lean-input-user-translations '(("/" "/")))
1526 *** [[https://github.com/haskell/haskell-mode][haskell-mode]]
1528 #+begin_src emacs-lisp
1529 (use-package haskell-mode
1531 (setq haskell-indentation-layout-offset 4
1532 haskell-indentation-left-offset 4
1533 flycheck-checker 'haskell-hlint
1534 flycheck-disabled-checkers '(haskell-stack-ghc haskell-ghc)))
1537 *** [[https://github.com/jyp/dante][dante]]
1539 #+begin_src emacs-lisp
1542 :commands dante-mode
1543 :hook (haskell-mode . dante-mode))
1546 *** [[https://github.com/mpickering/hlint-refactor-mode][hlint-refactor]]
1548 Emacs bindings for [[https://github.com/ndmitchell/hlint][hlint]]'s refactor option. This requires the refact
1549 executable from [[https://github.com/mpickering/apply-refact][apply-refact]].
1551 #+begin_src emacs-lisp
1552 (use-package hlint-refactor
1554 :bind (:map hlint-refactor-mode-map
1555 ("C-c l b" . hlint-refactor-refactor-buffer)
1556 ("C-c l r" . hlint-refactor-refactor-at-point))
1557 :hook (haskell-mode . hlint-refactor-mode))
1560 *** [[https://github.com/flycheck/flycheck-haskell][flycheck-haskell]]
1562 #+begin_src emacs-lisp
1563 (use-package flycheck-haskell
1564 :after haskell-mode)
1567 *** [[https://github.com/ndmitchell/hlint/blob/20e116a043f2073c57b17b24ae6364b5e433ba7e/data/hs-lint.el][hs-lint.el]]
1569 :header-args+: :tangle lisp/hs-lint.el :mkdirp yes
1572 Currently using =flycheck-haskell= with the =haskell-hlint= checker
1575 #+begin_src emacs-lisp :tangle no
1576 ;;; hs-lint.el --- minor mode for HLint code checking
1578 ;; Copyright 2009 (C) Alex Ott
1580 ;; Author: Alex Ott <alexott@gmail.com>
1581 ;; Keywords: haskell, lint, HLint
1583 ;; Status: distributed under terms of GPL2 or above
1585 ;; Typical message from HLint looks like:
1587 ;; /Users/ott/projects/lang-exp/haskell/test.hs:52:1: Eta reduce
1589 ;; count1 p l = length (filter p l)
1591 ;; count1 p = length . filter p
1596 (defgroup hs-lint nil
1597 "Run HLint as inferior of Emacs, parse error messages."
1601 (defcustom hs-lint-command "hlint"
1602 "The default hs-lint command for \\[hlint]."
1606 (defcustom hs-lint-save-files t
1607 "Save modified files when run HLint or no (ask user)"
1611 (defcustom hs-lint-replace-with-suggestions nil
1612 "Replace user's code with suggested replacements"
1616 (defcustom hs-lint-replace-without-ask nil
1617 "Replace user's code with suggested replacements automatically"
1621 (defun hs-lint-process-setup ()
1622 "Setup compilation variables and buffer for `hlint'."
1623 (run-hooks 'hs-lint-setup-hook))
1625 ;; regex for replace suggestions
1627 ;; ^\(.*?\):\([0-9]+\):\([0-9]+\): .*
1633 (defvar hs-lint-regex
1634 "^\\(.*?\\):\\([0-9]+\\):\\([0-9]+\\): .*[\n\C-m]Found:[\n\C-m]\\s +\\(.*\\)[\n\C-m]Why not:[\n\C-m]\\s +\\(.*\\)[\n\C-m]"
1635 "Regex for HLint messages")
1637 (defun make-short-string (str maxlen)
1638 (if (< (length str) maxlen)
1640 (concat (substring str 0 (- maxlen 3)) "...")))
1642 (defun hs-lint-replace-suggestions ()
1643 "Perform actual replacement of suggestions"
1644 (goto-char (point-min))
1645 (while (re-search-forward hs-lint-regex nil t)
1646 (let* ((fname (match-string 1))
1647 (fline (string-to-number (match-string 2)))
1648 (old-code (match-string 4))
1649 (new-code (match-string 5))
1650 (msg (concat "Replace '" (make-short-string old-code 30)
1651 "' with '" (make-short-string new-code 30) "'"))
1657 (switch-to-buffer (get-file-buffer fname))
1658 (goto-char (point-min))
1659 (forward-line (1- fline))
1661 (setf bline (point))
1662 (when (or hs-lint-replace-without-ask
1665 (setf eline (point))
1667 (setf old-code (regexp-quote old-code))
1668 (while (string-match "\\\\ " old-code spos)
1669 (setf new-old-code (concat new-old-code
1670 (substring old-code spos (match-beginning 0))
1672 (setf spos (match-end 0)))
1673 (setf new-old-code (concat new-old-code (substring old-code spos)))
1674 (remove-text-properties bline eline '(composition nil))
1675 (when (re-search-forward new-old-code eline t)
1676 (replace-match new-code nil t)))))))
1678 (defun hs-lint-finish-hook (buf msg)
1679 "Function, that is executed at the end of HLint execution"
1680 (if hs-lint-replace-with-suggestions
1681 (hs-lint-replace-suggestions)
1684 (define-compilation-mode hs-lint-mode "HLint"
1685 "Mode for check Haskell source code."
1686 (set (make-local-variable 'compilation-process-setup-function)
1687 'hs-lint-process-setup)
1688 (set (make-local-variable 'compilation-disable-input) t)
1689 (set (make-local-variable 'compilation-scroll-output) nil)
1690 (set (make-local-variable 'compilation-finish-functions)
1691 (list 'hs-lint-finish-hook))
1695 "Run HLint for current buffer with haskell source"
1697 (save-some-buffers hs-lint-save-files)
1698 (compilation-start (concat hs-lint-command " \"" buffer-file-name "\"")
1702 ;;; hs-lint.el ends here
1705 #+begin_src emacs-lisp :tangle no
1706 (use-package hs-lint
1708 :bind (:map haskell-mode-map
1709 ("C-c l l" . hs-lint)))
1716 #+begin_src emacs-lisp
1717 (use-package sgml-mode
1719 (setq sgml-basic-offset 2))
1724 #+begin_src emacs-lisp
1725 (use-package css-mode
1727 (setq css-indent-offset 2))
1732 #+begin_src emacs-lisp
1733 (use-package web-mode
1737 web-mode-code-indent-offset
1738 web-mode-css-indent-offset
1739 web-mode-markup-indent-offset))
1744 #+begin_src emacs-lisp
1745 (use-package emmet-mode
1746 :after (:any web-mode css-mode sgml-mode)
1747 :bind* (("C-)" . emmet-next-edit-point)
1748 ("C-(" . emmet-prev-edit-point))
1750 (unbind-key "C-j" emmet-mode-keymap)
1751 (setq emmet-move-cursor-between-quotes t)
1752 :hook (web-mode css-mode html-mode sgml-mode))
1757 *** COMMENT meghanada
1759 #+begin_src emacs-lisp
1760 (use-package meghanada
1762 (:map meghanada-mode-map
1763 (("C-M-o" . meghanada-optimize-import)
1764 ("C-M-t" . meghanada-import-all)))
1765 :hook (java-mode . meghanada-mode))
1768 *** COMMENT lsp-java
1786 #+begin_src emacs-lisp
1787 (use-package treemacs
1788 :config (setq treemacs-never-persist t))
1790 (use-package yasnippet
1792 ;; (yas-global-mode)
1795 (use-package lsp-mode
1796 :init (setq lsp-eldoc-render-all nil
1797 lsp-highlight-symbol-at-point nil)
1802 (use-package company-lsp
1805 (setq company-lsp-cache-candidates t
1806 company-lsp-async t))
1810 (setq lsp-ui-sideline-update-mode 'point))
1812 (use-package lsp-java
1814 (add-hook 'java-mode-hook
1816 (setq-local company-backends (list 'company-lsp))))
1818 (add-hook 'java-mode-hook 'lsp-java-enable)
1819 (add-hook 'java-mode-hook 'flycheck-mode)
1820 (add-hook 'java-mode-hook 'company-mode)
1821 (add-hook 'java-mode-hook 'lsp-ui-mode))
1823 (use-package dap-mode
1829 (use-package dap-java
1832 (use-package lsp-java-treemacs
1838 #+begin_src emacs-lisp
1840 :bind (:map eclim-mode-map ("S-SPC" . company-complete))
1841 :hook ((java-mode . eclim-mode)
1842 (eclim-mode . (lambda ()
1843 (make-local-variable 'company-idle-delay)
1844 (defvar company-idle-delay)
1845 ;; (setq company-idle-delay 0.7)
1846 (setq company-idle-delay nil))))
1848 (eclim-auto-save nil)
1849 ;; (eclimd-default-workspace "~/src/eclipse-workspace-exp")
1850 (eclim-executable "~/.p2/pool/plugins/org.eclim_2.8.0/bin/eclim")
1851 (eclim-eclipse-dirs '("~/usr/eclipse/dsl-2018-09/eclipse")))
1856 #+begin_src emacs-lisp
1857 (use-package geiser)
1859 (use-feature geiser-guile
1861 (setq geiser-guile-load-path "~/src/git/guix"))
1866 #+begin_src emacs-lisp
1870 * Emacs enhancements
1872 :CUSTOM_ID: emacs-enhancements
1877 #+begin_src emacs-lisp
1879 :config (setq Man-width 80))
1882 ** [[https://github.com/justbur/emacs-which-key][which-key]]
1885 Emacs package that displays available keybindings in popup
1888 #+begin_src emacs-lisp
1889 (use-package which-key
1892 (which-key-add-key-based-replacements
1893 ;; prefixes for global prefixes and minor modes
1897 "C-c 8 -" "typo/dashes"
1898 "C-c 8 <" "typo/left-brackets"
1899 "C-c 8 >" "typo/right-brackets"
1901 "C-x a" "abbrev/expand"
1902 "C-x r" "rectangle/register/bookmark"
1903 "C-x v" "version control"
1904 ;; prefixes for my personal bindings
1905 "C-c a" "applications"
1907 "C-c p" "package-management"
1908 "C-c p e" "package-management/epkg"
1909 "C-c p s" "straight.el"
1912 "C-c c" "compile-and-comments"
1917 "C-c m" "multiple-cursors"
1918 "C-c P" "projectile"
1919 "C-c P s" "projectile/search"
1920 "C-c P x" "projectile/execute"
1921 "C-c P 4" "projectile/other-window"
1927 ;; prefixes for major modes
1928 (which-key-add-major-mode-key-based-replacements 'message-mode
1930 (which-key-add-major-mode-key-based-replacements 'org-mode
1931 "C-c C-v" "org-babel")
1932 (which-key-add-major-mode-key-based-replacements 'web-mode
1933 "C-c C-a" "web/attributes"
1934 "C-c C-b" "web/blocks"
1936 "C-c C-e" "web/element"
1937 "C-c C-t" "web/tags")
1941 (which-key-add-column-padding 5)
1942 (which-key-max-description-length 32))
1947 #+begin_src emacs-lisp
1948 (add-to-list 'custom-theme-load-path "~/.emacs.d/lisp")
1949 (load-theme 'tangomod t)
1954 #+begin_src emacs-lisp
1955 (use-package smart-mode-line
1956 :commands (sml/apply-theme)
1964 #+begin_src emacs-lisp
1965 (use-package doom-themes)
1968 ** theme helper functions
1970 #+begin_src emacs-lisp
1971 (defvar a/org-mode-font-lock-keywords
1972 '(("[ \t]*\\(#\\+\\(BEGIN\\|END\\|begin\\|end\\)_\\(\\S-+\\)\\)[ \t]*\\([^\n:]*\\)"
1973 (1 '(:foreground "#5a5b5a" :background "#292b2b") t) ; directive
1974 (3 '(:foreground "#81a2be" :background "#292b2b") t) ; kind
1975 (4 '(:foreground "#c5c8c6") t)))) ; title
1977 (defun a/lights-on ()
1978 "Enable my favourite light theme."
1980 (mapc #'disable-theme custom-enabled-themes)
1981 (load-theme 'tangomod t)
1982 (sml/apply-theme 'automatic)
1983 (font-lock-remove-keywords
1984 'org-mode a/org-mode-font-lock-keywords))
1986 (defun a/lights-off ()
1989 (mapc #'disable-theme custom-enabled-themes)
1990 (load-theme 'doom-tomorrow-night t)
1991 (sml/apply-theme 'automatic)
1992 (font-lock-add-keywords
1993 'org-mode a/org-mode-font-lock-keywords t))
1996 ("s-t d" . a/lights-off)
1997 ("s-t l" . a/lights-on))
2000 ** [[https://github.com/bbatsov/crux][crux]]
2002 #+begin_src emacs-lisp
2003 (use-package crux ; results in Waiting for git... [2 times]
2005 :bind (("C-c b k" . crux-kill-other-buffers)
2006 ("C-c d" . crux-duplicate-current-line-or-region)
2007 ("C-c D" . crux-duplicate-and-comment-current-line-or-region)
2008 ("C-c f c" . crux-copy-file-preserve-attributes)
2009 ("C-c f d" . crux-delete-file-and-buffer)
2010 ("C-c f r" . crux-rename-file-and-buffer)
2011 ("C-c j" . crux-top-join-line)
2012 ("C-S-j" . crux-top-join-line)))
2015 ** [[https://github.com/alezost/mwim.el][mwim]]
2017 #+begin_src emacs-lisp
2019 :bind (("C-a" . mwim-beginning-of-code-or-line)
2020 ("C-e" . mwim-end-of-code-or-line)
2021 ("<home>" . mwim-beginning-of-line-or-code)
2022 ("<end>" . mwim-end-of-line-or-code)))
2027 #+begin_src emacs-lisp
2028 (use-package projectile
2029 :bind-keymap ("C-c P" . projectile-command-map)
2033 (defun my-projectile-invalidate-cache (&rest _args)
2034 ;; ignore the args to `magit-checkout'
2035 (projectile-invalidate-cache nil))
2037 (eval-after-load 'magit-branch
2039 (advice-add 'magit-checkout
2040 :after #'my-projectile-invalidate-cache)
2041 (advice-add 'magit-branch-and-checkout
2042 :after #'my-projectile-invalidate-cache)))
2043 :custom (projectile-completion-system 'ivy))
2046 ** [[https://github.com/Wilfred/helpful][helpful]]
2048 #+begin_src emacs-lisp
2049 (use-package helpful
2052 (("C-S-h c" . helpful-command)
2053 ("C-S-h f" . helpful-callable) ; helpful-function
2054 ("C-S-h v" . helpful-variable)
2055 ("C-S-h k" . helpful-key)
2056 ("C-S-h p" . helpful-at-point)))
2059 ** [[https://github.com/EricCrosson/unkillable-scratch][unkillable-scratch]]
2061 Make =*scratch*= and =*Messages*= unkillable.
2063 #+begin_src emacs-lisp
2064 (use-package unkillable-scratch
2067 (unkillable-scratch 1)
2069 (unkillable-buffers '("^\\*scratch\\*$" "^\\*Messages\\*$")))
2072 ** [[https://github.com/davep/boxquote.el][boxquote.el]]
2076 | make pretty boxed quotes like this
2080 #+begin_src emacs-lisp
2081 (use-package boxquote
2084 (:prefix-map a/boxquote-prefix-map
2086 ("b" . boxquote-buffer)
2087 ("B" . boxquote-insert-buffer)
2088 ("d" . boxquote-defun)
2089 ("F" . boxquote-insert-file)
2090 ("hf" . boxquote-describe-function)
2091 ("hk" . boxquote-describe-key)
2092 ("hv" . boxquote-describe-variable)
2093 ("hw" . boxquote-where-is)
2094 ("k" . boxquote-kill)
2095 ("p" . boxquote-paragraph)
2096 ("q" . boxquote-boxquote)
2097 ("r" . boxquote-region)
2098 ("s" . boxquote-shell-command)
2099 ("t" . boxquote-text)
2100 ("T" . boxquote-title)
2101 ("u" . boxquote-unbox)
2102 ("U" . boxquote-unbox-region)
2103 ("y" . boxquote-yank)
2104 ("M-q" . boxquote-fill-paragraph)
2105 ("M-w" . boxquote-kill-ring-save)))
2108 Also see [[https://www.emacswiki.org/emacs/rebox2][rebox2]].
2112 #+begin_src emacs-lisp
2113 (use-package orgalist
2115 :hook (message-mode . orgalist-mode))
2120 #+begin_src emacs-lisp
2124 (typo-global-mode 1)
2125 :hook (text-mode . typo-mode))
2130 #+begin_src emacs-lisp
2131 (use-package hl-todo
2134 (global-hl-todo-mode))
2139 #+begin_src emacs-lisp
2140 (use-package shrink-path
2144 (defun +eshell/prompt ()
2145 (let ((base/dir (shrink-path-prompt default-directory)))
2146 (concat (propertize (car base/dir)
2147 'face 'font-lock-comment-face)
2148 (propertize (cdr base/dir)
2149 'face 'font-lock-constant-face)
2150 (propertize (+eshell--current-git-branch)
2151 'face 'font-lock-function-name-face)
2154 ;; (propertize "λ" 'face 'eshell-prompt)
2155 ;; needed for the input text to not have prompt face
2156 (propertize "λ " 'face 'default))))
2158 (defun +eshell--current-git-branch ()
2159 (let ((branch (car (loop for match in (split-string (shell-command-to-string "git branch") "\n")
2160 when (string-match "^\*" match)
2162 (if (not (eq branch nil))
2163 (concat " " (substring branch 2))
2165 (setq eshell-prompt-regexp "\\(.*\n\\)*λ "
2166 eshell-prompt-function #'+eshell/prompt))
2169 ** [[https://github.com/peterwvj/eshell-up][eshell-up]]
2171 #+begin_src emacs-lisp
2172 (use-package eshell-up
2174 :commands eshell-up)
2179 #+begin_src emacs-lisp
2180 (use-package multi-term
2182 :bind (("C-c a s m" . multi-term-dedicated-toggle)
2184 ("C-c C-j" . term-char-mode)
2186 ("C-c C-j" . term-line-mode))
2188 (setq multi-term-program "/bin/screen"
2189 ;; TODO: add separate bindings for connecting to existing
2190 ;; session vs. always creating a new one
2191 multi-term-dedicated-select-after-open-p t
2192 multi-term-dedicated-window-height 20
2193 multi-term-dedicated-max-window-height 30
2195 '(("C-c C-c" . term-interrupt-subjob)
2196 ("C-c C-e" . term-send-esc)
2198 ("C-y" . term-paste)
2199 ("M-f" . term-send-forward-word)
2200 ("M-b" . term-send-backward-word)
2201 ("M-p" . term-send-up)
2202 ("M-n" . term-send-down)
2203 ("<C-backspace>" . term-send-backward-kill-word)
2204 ("<M-DEL>" . term-send-backward-kill-word)
2205 ("M-d" . term-send-delete-word)
2206 ("M-," . term-send-raw)
2207 ("M-." . comint-dynamic-complete))
2208 term-unbind-key-alist
2209 '("C-z" "C-x" "C-c" "C-h" "C-y" "<ESC>")))
2214 #+begin_src emacs-lisp
2215 (use-package page-break-lines
2217 (global-page-break-lines-mode))
2222 #+begin_src emacs-lisp
2223 (use-package expand-region
2224 :bind ("C-=" . er/expand-region))
2229 #+begin_src emacs-lisp
2230 (use-package multiple-cursors
2232 (("C-S-<mouse-1>" . mc/add-cursor-on-click)
2233 (:prefix-map a/mc-prefix-map
2235 ("c" . mc/edit-lines)
2236 ("n" . mc/mark-next-like-this)
2237 ("p" . mc/mark-previous-like-this)
2238 ("a" . mc/mark-all-like-this))))
2243 #+begin_src emacs-lisp
2251 #+begin_src emacs-lisp
2252 (use-package yasnippet
2255 (defconst yas-verbosity-cur yas-verbosity)
2256 (setq yas-verbosity 2)
2257 (add-to-list 'yas-snippet-dirs "~/src/git/guix/etc/snippets")
2259 (setq yas-verbosity yas-verbosity-cur)
2261 (text-mode . yas-minor-mode))
2269 #+begin_src emacs-lisp
2270 (defvar a/maildir (expand-file-name "~/mail/"))
2271 (with-eval-after-load 'recentf
2272 (add-to-list 'recentf-exclude a/maildir))
2277 #+begin_src emacs-lisp
2279 a/gnus-init-file (no-littering-expand-etc-file-name "gnus")
2280 mail-user-agent 'gnus-user-agent
2281 read-mail-command 'gnus)
2284 :bind (("s-m" . gnus)
2285 ("s-M" . gnus-unplugged))
2288 gnus-select-method '(nnnil "")
2289 gnus-secondary-select-methods
2291 (nnimap-stream plain)
2292 (nnimap-address "127.0.0.1")
2293 (nnimap-server-port 143)
2294 (nnimap-authenticator plain)
2295 (nnimap-user "amin@bndl.org"))
2297 (nnimap-stream plain)
2298 (nnimap-address "127.0.0.1")
2299 (nnimap-server-port 143)
2300 (nnimap-authenticator plain)
2301 (nnimap-user "abandali@uwaterloo.ca"))
2303 (nnimap-stream plain)
2304 (nnimap-address "127.0.0.1")
2305 (nnimap-server-port 143)
2306 (nnimap-authenticator plain)
2307 (nnimap-user "abandali@csclub.uw")))
2308 gnus-message-archive-group "nnimap+amin:Sent"
2312 gnus-large-newsgroup 50
2313 gnus-home-directory (no-littering-expand-var-file-name "gnus/")
2314 gnus-directory (concat gnus-home-directory "news/")
2315 message-directory (concat gnus-home-directory "mail/")
2316 nndraft-directory (concat gnus-home-directory "drafts/")
2317 gnus-save-newsrc-file nil
2318 gnus-read-newsrc-file nil
2319 gnus-interactive-exit nil
2320 gnus-gcc-mark-as-read t)
2324 (require 'ebdb-gnus))
2326 (use-feature gnus-art
2329 gnus-visible-headers
2330 (concat gnus-visible-headers "\\|^List-Id:\\|^X-RT-Originator:\\|^User-Agent:")
2331 gnus-sorted-header-list
2332 '("^From:" "^Subject:" "^Summary:" "^Keywords:"
2333 "^Followup-To:" "^To:" "^Cc:" "X-RT-Originator"
2334 "^Newsgroups:" "List-Id:" "^Organization:"
2335 "^User-Agent:" "^Date:")
2336 ;; local-lapsed article dates
2337 ;; from https://www.emacswiki.org/emacs/GnusFormatting#toc11
2338 gnus-article-date-headers '(user-defined)
2339 gnus-article-time-format
2341 (let* ((date (format-time-string "%a, %d %b %Y %T %z" time))
2342 (local (article-make-date-line date 'local))
2343 (combined-lapsed (article-make-date-line date
2346 (string-match " (.+" combined-lapsed)
2347 (match-string 0 combined-lapsed))))
2348 (concat local lapsed))))
2350 :map gnus-article-mode-map
2351 ("r" . gnus-article-reply-with-original)
2352 ("R" . gnus-article-wide-reply-with-original)
2353 ("M-L" . org-store-link)))
2355 (use-feature gnus-sum
2356 :bind (:map gnus-summary-mode-map
2357 :prefix-map a/gnus-summary-prefix-map
2359 ("r" . gnus-summary-reply)
2360 ("w" . gnus-summary-wide-reply)
2361 ("v" . gnus-summary-show-raw-article))
2364 :map gnus-summary-mode-map
2365 ("r" . gnus-summary-reply-with-original)
2366 ("R" . gnus-summary-wide-reply-with-original)
2367 ("M-L" . org-store-link))
2368 :hook (gnus-summary-mode . a/no-mouse-autoselect-window))
2370 (use-feature gnus-msg
2372 (setq gnus-posting-styles
2374 (address "amin@bndl.org")
2375 (body "\nBest,\namin\n")
2376 (eval (setq a/message-cite-say-hi t)))
2378 (address "bandali@gnu.org")
2379 (eval (set (make-local-variable 'message-user-fqdn) "fencepost.gnu.org")))
2380 ((header "subject" "ThankCRM")
2381 (to "webmasters-comment@gnu.org")
2382 (body "\nAdded to 2019supporters.html.\n\nMoving to campaigns.\n\n-amin\n")
2383 (eval (setq a/message-cite-say-hi nil)))
2384 ("nnimap\\+uwaterloo:.*"
2385 (address "abandali@uwaterloo.ca")
2386 (gcc "\"nnimap+uwaterloo:Sent Items\""))
2387 ("nnimap\\+csclub:.*"
2388 (address "abandali@csclub.uwaterloo.ca")
2389 (gcc "nnimap+csclub:Sent")))))
2391 (use-feature gnus-topic
2392 :hook (gnus-group-mode . gnus-topic-mode)
2393 :config (setq gnus-topic-line-format "%i[ %A: %(%{%n%}%) ]%v\n"))
2395 (use-feature gnus-agent
2397 (setq gnus-agent-synchronize-flags 'ask)
2398 :hook (gnus-group-mode . gnus-agent-mode))
2400 (use-feature gnus-group
2402 (setq gnus-permanently-visible-groups "\\(:INBOX$\\|:gnu$\\)"))
2404 (use-feature mm-decode
2406 (setq mm-discouraged-alternatives '("text/html" "text/richtext")))
2411 #+begin_src emacs-lisp
2412 (use-feature sendmail
2414 (setq sendmail-program "/usr/bin/msmtp"
2415 ;; message-sendmail-extra-arguments '("-v" "-d")
2416 mail-specify-envelope-from t
2417 mail-envelope-from 'header))
2422 #+begin_src emacs-lisp
2423 (use-feature message
2425 (defconst a/message-cite-style-format "On %Y-%m-%d %l:%M %p, %N wrote:")
2426 (defconst message-cite-style-bandali
2427 '((message-cite-function 'message-cite-original)
2428 (message-citation-line-function 'message-insert-formatted-citation-line)
2429 (message-cite-reply-position 'traditional)
2430 (message-yank-prefix "> ")
2431 (message-yank-cited-prefix ">")
2432 (message-yank-empty-prefix ">")
2433 (message-citation-line-format
2434 (if a/message-cite-say-hi
2435 (concat "Hi %F,\n\n" a/message-cite-style-format)
2436 a/message-cite-style-format)))
2437 "Citation style based on Mozilla Thunderbird's. Use with message-cite-style.")
2438 (setq message-cite-style 'message-cite-style-bandali
2439 message-kill-buffer-on-exit t
2440 message-send-mail-function 'message-send-mail-with-sendmail
2441 message-sendmail-envelope-from 'header
2442 message-dont-reply-to-names
2443 "\\(\\(amin@bndl\\.org\\)\\|\\(.*@\\(aminb\\|amin\\.bndl\\)\\.org\\)\\|\\(\\(bandali\\|aminb?\\|mab\\)@gnu\\.org\\)\\|\\(a\\(min\\.\\)?bandali@uwaterloo\\.ca\\)\\|\\(abandali@csclub\\.uwaterloo\\.ca\\)\\)")
2444 (require 'company-ebdb)
2445 :hook (;; (message-setup . mml-secure-message-sign-pgpmime)
2446 (message-mode . flyspell-mode)
2447 (message-mode . (lambda ()
2448 ;; (setq fill-column 65
2449 ;; message-fill-column 65)
2450 (make-local-variable 'company-idle-delay)
2451 (setq company-idle-delay 0.2))))
2453 ;; (message-header-subject ((t (:foreground "#111" :weight semi-bold))))
2454 ;; (message-header-to ((t (:foreground "#111" :weight normal))))
2455 ;; (message-header-cc ((t (:foreground "#333" :weight normal))))
2458 (with-eval-after-load 'mml-sec
2459 (setq mml-secure-openpgp-encrypt-to-self t
2460 mml-secure-openpgp-sign-with-sender t))
2465 Convenient footnotes in =message-mode=.
2467 #+begin_src emacs-lisp
2468 (use-feature footnote
2471 ;; (setq footnote-start-tag ""
2472 ;; footnote-end-tag ""
2473 ;; footnote-style 'unicode)
2475 (:map message-mode-map
2476 :prefix-map a/footnote-prefix-map
2478 ("a" . footnote-add-footnote)
2479 ("b" . footnote-back-to-message)
2480 ("c" . footnote-cycle-style)
2481 ("d" . footnote-delete-footnote)
2482 ("g" . footnote-goto-footnote)
2483 ("r" . footnote-renumber-footnotes)
2484 ("s" . footnote-set-style)))
2489 #+begin_src emacs-lisp
2491 :straight (:host github :repo "girzel/ebdb")
2493 :bind (:map gnus-group-mode-map ("e" . ebdb))
2495 (setq ebdb-sources (no-littering-expand-var-file-name "ebdb"))
2496 (with-eval-after-load 'swiper
2497 (add-to-list 'swiper-font-lock-exclude 'ebdb-mode t)))
2499 (use-feature ebdb-com
2502 ;; (use-package ebdb-complete
2505 ;; (ebdb-complete-enable))
2507 (use-package company-ebdb
2509 (defun company-ebdb--post-complete (_) nil))
2511 (use-feature ebdb-gnus
2514 (ebdb-gnus-window-configuration
2517 (summary 0.25 point)
2520 (ebdb-gnus 0.3))))))
2522 (use-feature ebdb-mua
2524 ;; :custom (ebdb-mua-pop-up nil)
2527 ;; (use-package ebdb-message
2531 ;; (use-package ebdb-vcard
2537 #+begin_src emacs-lisp
2538 (use-package message-x)
2541 #+begin_src emacs-lisp :tangle no
2542 (use-package message-x
2544 (message-x-completion-alist
2546 (("\\([rR]esent-\\|[rR]eply-\\)?[tT]o:\\|[bB]?[cC][cC]:" . gnus-harvest-find-address)
2549 (quote message-newgroups-header-regexp))
2550 message-newgroups-header-regexp message-newsgroups-header-regexp)
2551 . message-expand-group)))))
2554 ** COMMENT gnus-harvest
2556 #+begin_src emacs-lisp
2557 (use-package gnus-harvest
2558 :commands gnus-harvest-install
2561 (if (featurep 'message-x)
2562 (gnus-harvest-install 'message-x)
2563 (gnus-harvest-install)))
2568 :CUSTOM_ID: blogging
2571 ** [[https://ox-hugo.scripter.co][ox-hugo]]
2573 #+begin_src emacs-lisp
2574 (use-package ox-hugo
2579 * Post initialization
2581 :CUSTOM_ID: post-initialization
2584 Display how long it took to load the init file.
2586 #+begin_src emacs-lisp
2587 (message "Loading %s...done (%.3fs)" user-init-file
2588 (float-time (time-subtract (current-time)
2589 a/before-user-init-time)))
2597 #+begin_src emacs-lisp :comments none
2598 ;;; init.el ends here
2601 * COMMENT Local Variables :ARCHIVE:
2603 # eval: (add-hook 'after-save-hook #'a/async-babel-tangle 'append 'local)
2604 # eval: (when (featurep 'typo (typo-mode -1)))