1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
|
;;; +emacs.el --- measured defaults for Emacs -*- lexical-binding: t -*-
;;; Commentary:
;; I find myself copy-pasting a lot of "boilerplate" type code when
;; bankrupting my Emacs config and starting afresh. Instead of doing
;; that, I'm putting it here, where it'll be easier to include in my
;; config.
;; Of course, some might say I could just ... stop bankrupting my
;; Emacs. But like, why would I want to?
;; Other notable packages include
;; - https://git.sr.ht/~technomancy/better-defaults/
;; - https://github.com/susam/emfy
;;; Code:
(require 'early-init (locate-user-emacs-file "early-init.el"))
(defun +set-major-mode-from-buffer-name (&optional buf)
"Set the major mode for BUF from the buffer's name.
Do this only if the buffer is not visiting a file."
(unless buffer-file-name
(let ((buffer-file-name (buffer-name buf)))
(set-auto-mode))))
;;; General settings
(setq-default
apropos-do-all t
async-shell-command-buffer 'new-buffer
async-shell-command-display-buffer nil
auto-hscroll-mode 'current-line
auto-revert-verbose t
auto-save-default nil
auto-save-file-name-transforms `((".*" ,(.etc "auto-save/") ,(car (secure-hash-algorithms)))
(".*" ,(.etc "auto-save/") t))
auto-save-interval 30
auto-save-list-file-prefix (.etc "auto-save/.saves-" t)
auto-save-timeout 30
auto-save-visited-interval 5
auto-window-vscroll nil
backup-by-copying t
backup-directory-alist `((".*" . ,(.etc "backup/" t)))
blink-cursor-blinks 1
comp-deferred-compilation nil
completion-category-defaults nil
completion-category-overrides '((file (styles . (partial-completion))))
completion-ignore-case t
completion-styles '(substring partial-completion)
create-lockfiles nil
cursor-in-non-selected-windows 'hollow
cursor-type 'bar
custom-file (.etc "custom.el")
delete-old-versions t
echo-keystrokes 0.1
ediff-window-setup-function 'ediff-setup-windows-plain
eldoc-echo-area-use-multiline-p nil
eldoc-idle-delay 0.1
enable-recursive-minibuffers t
executable-prefix-env t
fast-but-imprecise-scrolling t
file-name-shadow-properties '(invisible t intangible t)
fill-column 80
find-file-visit-truename t
frame-resize-pixelwise t
global-auto-revert-non-file-buffers t
global-mark-ring-max 100
hscroll-margin 1
hscroll-step 1
imenu-auto-rescan t
image-use-external-converter (or (executable-find "convert")
(executable-find "gm")
(executable-find "ffmpeg"))
indent-tabs-mode nil
inhibit-startup-screen t
initial-buffer-choice t
kept-new-versions 6
kept-old-versions 2
kill-do-not-save-duplicates t
kill-read-only-ok t
kill-ring-max 500
kmacro-ring-max 20
load-prefer-newer noninteractive
major-mode '+set-major-mode-from-buffer-name
mark-ring-max 50
minibuffer-eldef-shorten-default t
minibuffer-prompt-properties (list 'read-only t
'cursor-intangible t
'face 'minibuffer-prompt)
mode-require-final-newline 'visit-save
mouse-drag-copy-region t
mouse-wheel-progressive-speed nil
mouse-yank-at-point t
native-comp-async-report-warnings-errors 'silent
native-comp-deferred-compilation nil
read-answer-short t
read-buffer-completion-ignore-case t
;; read-extended-command-predicate
;; (when (fboundp
;; 'command-completion-default-include-p)
;; 'command-completion-default-include-p)
read-process-output-max 1048576 ; We’re in the future man. Set that to at least a megabyte
recenter-positions '(top middle bottom)
regexp-search-ring-max 100
regexp-search-ring-max 200
save-interprogram-paste-before-kill t
save-some-buffers-default-predicate #'+save-some-buffers-p
scroll-conservatively 101
scroll-down-aggressively 0.01
scroll-margin 2
scroll-preserve-screen-position 1
scroll-step 1
scroll-up-aggressively 0.01
search-ring-max 200
search-ring-max 200
sentence-end-double-space t
set-mark-command-repeat-pop t
show-paren-delay 0
show-paren-style 'parenthesis
show-paren-when-point-in-periphery t
show-paren-when-point-inside-paren t
;;show-trailing-whitespace t
tab-bar-show 1
tab-width 8 ; so alignment expecting the default looks right
tramp-backup-directory-alist backup-directory-alist
undo-limit 100000000 ; 10 MB
use-dialog-box nil
use-file-dialog nil
use-short-answers t
vc-follow-symlinks t
vc-make-backup-files t
version-control t
view-read-only t
visible-bell nil
window-resize-pixelwise t
x-select-enable-clipboard t
x-select-enable-primary t
yank-pop-change-selection t
)
;; Programming language offsets.
;; Set these after the initial block so I can use `tab-width'
(setq-default
c-basic-offset tab-width)
;; Emacs 28 ships with an option, `use-short-answers', that makes this form
;; obsolete, but I still use 27 at work.
(when (version< emacs-version "28")
(fset 'yes-or-no-p 'y-or-n-p))
;;; Encodings
;; Allegedly, this is the only one you need...
(set-language-environment "UTF-8")
;; But I still set all of these, for fun.
(setq-default locale-coding-system 'utf-8-unix
coding-system-for-read 'utf-8-unix
coding-system-for-write 'utf-8-unix
buffer-file-coding-system 'utf-8-unix
default-process-coding-system '(utf-8-unix . utf-8-unix)
x-select-request-type '(UTF8_STRING
COMPOUND_TEXT
TEXT
STRING))
(set-charset-priority 'unicode)
(prefer-coding-system 'utf-8-unix)
(set-default-coding-systems 'utf-8-unix)
(set-terminal-coding-system 'utf-8-unix)
(set-keyboard-coding-system 'utf-8-unix)
(pcase system-type
((or 'ms-dos 'windows-nt)
(set-clipboard-coding-system 'utf-16-le)
(set-selection-coding-system 'utf-16-le))
(_
(set-selection-coding-system 'utf-8)
(set-clipboard-coding-system 'utf-8)))
;;; Modes
(dolist (enable-mode '(global-auto-revert-mode
blink-cursor-mode
electric-pair-mode
show-paren-mode
global-so-long-mode
minibuffer-depth-indicate-mode
file-name-shadow-mode
minibuffer-electric-default-mode
delete-selection-mode
auto-save-visited-mode
;; column-number-mode
))
(when (fboundp enable-mode)
(funcall enable-mode +1)))
(dolist (disable-mode '(tooltip-mode
tool-bar-mode
menu-bar-mode
scroll-bar-mode
horizontal-scroll-bar-mode))
(when (fboundp disable-mode)
(funcall disable-mode -1)))
;;; Hooks
(defun +auto-create-missing-dirs ()
"Automatically create missing directories when finding a file."
;; https://emacsredux.com/blog/2022/06/12/auto-create-missing-directories/
(let ((target-dir (file-name-directory buffer-file-name)))
(unless (file-exists-p target-dir)
(make-directory target-dir t))))
(defvar +save-some-buffers-debounce-time nil
"Last time `+save-some-buffers-debounce' was run.")
(defcustom +save-some-buffers-debounce-timeout 5
"Number of seconds to wait before saving buffers again.")
(defun +save-some-buffers-debounce (&rest _)
"Run `save-some-buffers', but only if it's been a while."
(unless (and +save-some-buffers-debounce-time
(< (- (time-convert nil 'integer) +save-some-buffers-debounce-time)
+save-some-buffers-debounce-timeout))
(save-some-buffers t)
(setf +save-some-buffers-debounce-time (time-convert nil 'integer))))
;;; Better-default functions ...
(defun +cycle-spacing (&optional n preserve-nl-back mode)
"Negate N argument on `cycle-spacing'.
That is, with a positive N, deletes newlines as well, leaving -N
spaces. If N is negative, it will not delete newlines and leave
N spaces. See docstring of `cycle-spacing' for the meaning of
PRESERVE-NL-BACK and MODE."
(interactive "*p")
(cycle-spacing (- n) preserve-nl-back mode))
(defun +save-buffers-quit (&optional arg)
"Silently save each buffer, then kill the current connection.
If the current frame has no client, kill Emacs itself using
`save-buffers-kill-emacs' after confirming with the user.
With prefix ARG, silently save all file-visiting buffers, then
kill without asking."
(interactive "P")
(save-some-buffers t)
(if (and (not (frame-parameter nil 'client))
(and (not arg)))
(when (yes-or-no-p "Sure you want to quit? ")
(save-buffers-kill-emacs))
(delete-frame nil :force)))
(defun +kill-word-backward-or-region (&optional arg backward-kill-word-fn)
"Kill active region or ARG words backward.
BACKWARD-KILL-WORD-FN is the function to call to kill a word
backward. It defaults to `backward-kill-word'."
(interactive "P")
(call-interactively (if (region-active-p)
#'kill-region
(or backward-kill-word-fn #'backward-kill-word))))
(defun +backward-kill-word-wrapper (fn &optional arg)
"Kill backward using FN until the beginning of a word, smartly.
If point is on at the beginning of a line, kill the previous new
line. If the only thing before point on the current line is
whitespace, kill that whitespace.
With argument ARG: if ARG is a number, just call FN
ARG times. Otherwise, just call FN."
;; I want this to be a wrapper so that I can call other word-killing functions
;; with it. It's *NOT* advice because those functions probably use
;; `backward-kill-word' under the hood (looking at you, paredit), so advice
;; will make things weird.
(if (null arg)
(cond
((looking-back "^" 1)
(let ((delete-active-region nil))
(delete-backward-char 1)))
((looking-back "^[ ]*")
(delete-horizontal-space :backward-only))
(t (call-interactively fn)))
(funcall fn (if (listp arg) 1 arg))))
(defun +backward-kill-word (&optional arg)
"Kill word backward using `backward-kill-word'.
ARG is passed to `backward-kill-word'."
(interactive "P")
(+backward-kill-word-wrapper #'backward-kill-word arg))
;;; ... and advice
;; Indent the region after a yank.
(defun +yank@indent (&rest _)
"Indent the current region."
(indent-region (min (point) (mark)) (max (point) (mark))))
;; (advice-add #'yank :after #'+yank@indent)
;; (advice-add #'yank-pop :after #'+yank@indent)
;; https://old.reddit.com/r/emacs/comments/y92y4b/tramp_users_slowness_got_you_down_check/it3a35r/
(defun +vc-off-when-remote ()
(when (file-remote-p (buffer-file-name))
(setq-local vc-handled-backends nil)))
;;; Extra functions
(defun +save-some-buffers-p ()
"Predicate for `save-some-buffers-default-predicate'.
It returns nil with remote files and those without attached files."
(and (buffer-file-name)
(not (file-remote-p (buffer-file-name)))))
;; https://www.wwwtech.de/articles/2013/may/emacs:-jump-to-matching-paren-beginning-of-block
(defun +goto-matching-paren (&optional arg)
"Go to the matching paren, similar to vi's %."
(interactive "p")
(or arg (setf arg 1))
(cond
;; Check for "outside of bracket" positions
((looking-at "[\[\(\{]") (forward-sexp arg))
((looking-back "[\]\)\}]" 1) (backward-sexp arg))
;; Otherwise, move from inside the bracket
((looking-at "[\]\)\}]") (forward-char) (backward-sexp arg))
((looking-back "[\[\(\{]" 1) (backward-char) (forward-sexp arg))
(t (up-list arg t t))))
(defun +delete-window-or-bury-buffer ()
"Delete the current window, or bury the current buffer.
If the current window is the only window, bury the buffer."
(interactive)
(condition-case e
(delete-window)
(t (bury-buffer))))
;;; Required libraries
(when (require 'abbrev nil :noerror)
(setq-default abbrev-file-name (sync/ "abbrev.el")
save-abbrevs 'silent))
(when (require 'autorevert nil :noerror)
(setq-default global-auto-revert-non-file-buffers t
auto-revert-verbose nil)
(global-auto-revert-mode +1))
(when (require 'uniquify nil :noerror)
(setq-default uniquify-buffer-name-style 'forward
uniquify-separator path-separator
uniquify-after-kill-buffer-p t
uniquify-ignore-buffers-re "^\\*"))
(when (require 'goto-addr)
(if (fboundp 'global-goto-address-mode)
(global-goto-address-mode +1)
(add-hook 'after-change-major-mode-hook 'goto-address-mode)))
(when (require 'recentf nil :noerror)
(setq-default recentf-save-file (.etc "recentf.el")
recentf-max-menu-items 100
recentf-max-saved-items nil
recentf-auto-cleanup 'mode)
(add-to-list 'recentf-exclude .etc)
(recentf-mode +1))
(when (require 'savehist nil :noerror)
(setq-default history-length t
history-delete-duplicates t
history-autosave-interval 60
savehist-file (.etc "savehist.el")
;; Other variables --- don't truncate any of these.
;; `add-to-history' uses the values of these variables unless
;; they're nil, in which case it falls back to `history-length'.
kill-ring-max 100
mark-ring-max 100
global-mark-ring-max 100
regexp-search-ring-max 100
search-ring-max 100
kmacro-ring-max 100
eww-history-limit 100)
(dolist (var '(extended-command-history
global-mark-ring
mark-ring
kill-ring
kmacro-ring
regexp-search-ring
search-ring))
(add-to-list 'savehist-additional-variables var))
(savehist-mode +1))
(when (require 'saveplace nil :noerror)
(setq-default save-place-file (.etc "places.el")
save-place-forget-unreadable-files (eq system-type 'gnu/linux))
(save-place-mode +1))
;; (when (require 'tramp)
;; ;; thanks Irreal! https://irreal.org/blog/?p=895
;; (add-to-list 'tramp-default-proxies-alist
;; '(nil "\\`root\\'" "/ssh:%h:"))
;; (add-to-list 'tramp-default-proxies-alist
;; '((regexp-quote (system-name)) nil nil)))
;;; Newer features
;; These aren't in older version of Emacs, but they're so nice.
(when (fboundp 'repeat-mode)
(setq-default repeat-exit-key "g"
repeat-exit-timeout 5)
(repeat-mode +1))
(when (fboundp 'pixel-scroll-precision-mode)
(pixel-scroll-precision-mode +1))
(provide '+emacs)
;;; +emacs.el ends here
|