;; init.el -- Marcel Fries ;; Configure emacs in a literate style from an org file. ;; The file sould be named "setup-emacs.org". It should be located in your ".emacs.d" directory. ;; Source at https://git.sr.ht/~luxick/emacs-config (require 'org) (org-babel-load-file (expand-file-name (concat user-emacs-directory "setup-emacs.org"))) ;; make PC keyboard's Win key or other to type Super or Hyper, for emacs running on Windows. (setq w32-pass-lwindow-to-system nil) (setq w32-lwindow-modifier 'super) ; Left Windows key (setq w32-pass-rwindow-to-system nil) (setq w32-rwindow-modifier 'super) ; Right Windows key (setq w32-pass-apps-to-system nil) (setq w32-apps-modifier 'hyper) ; Menu/App key (use-package all-the-icons) ;; Ensure that items in the PATH are made available to Emacs. This should ;; probably just come with the main distribution. ;; Recentf comes with Emacs but it should always be enabled. (use-package recentf :init (recentf-mode t) :config (add-to-list 'recentf-exclude "\\.emacs.d") (add-to-list 'recentf-exclude ".+tmp......\\.org")) (use-package gnu-elpa-keyring-update) ;; Magit is one of the best pieces of OSS I have ever used. It is truly esssential. (use-package magit :bind (("C-c g" . magit-status)) :diminish magit-auto-revert-mode :diminish auto-revert-mode :custom (magit-remote-set-if-missing t) (magit-diff-refine-hunk t) :config (magit-auto-revert-mode t) ;; Magit, and Emacs in general, has a nasty habit of prompting to save buffers ;; that are identical to those on disk. This is an attempt at remedying that, ;; one that I should probably attach to other functions like save-buffers-kill-emacs. (advice-add 'magit-refresh :before #'maybe-unset-buffer-modified) (advice-add 'magit-commit :before #'maybe-unset-buffer-modified) (setq magit-completing-read-function 'ivy-completing-read) (add-to-list 'magit-no-confirm 'stage-all-changes)) ;; Unclear whether this does anything at the moment. (use-package libgit :disabled :after magit) ;; Since I grew up on Textmate, I'm more-or-less reliant on snippets (use-package yasnippet :config (yas-global-mode 1) (setq yas-prompt-functions '(yas-completing-prompt)) :diminish yas-minor-mode) ;; Haskell and Elisp are made a lot easier when delimiters are nicely color-coded. (use-package rainbow-delimiters :disabled :hook (prog-mode . rainbow-delimiters-mode)) ;; multiple-cursors is better than cua-selection-mode. ;; TODO: learn ace-mc (use-package multiple-cursors :bind (("C-c M" . mc/edit-lines))) ;; The beauty of undo-tree is that it means that, once you've typed something into ;; a buffer, you'll always be able to get it back. That is crucial. (use-package undo-tree :bind (("C-x u" . undo-tree-visualize) ("C-z" . undo-tree-undo) ("C-S-z" . undo-tree-redo)) :config (global-undo-tree-mode +1) (unbind-key "M-_" undo-tree-map) :diminish) ;; Trying undo-propose, which seems to offer a better experience, as ;; undo tree is prone to losing data. (use-package undo-propose :disabled :bind (("C-c _" . undo-propose) :map undo-propose-mode-map ("" . undo-only))) ;; (use-package ansi-color ;; :config ;; (add-hook 'shell-mode-hook 'ansi-color-for-comint-mode-on)) ;; C stuff. (use-package cc-mode :disabled) ;; I do all of my writing in either org-mode or markdown-mode. (use-package markdown-mode :mode ("\\.md$" . gfm-mode) :config (when (executable-find "pandoc") (setq markdown-command "pandoc -f markdown -t html"))) ;; Quickly duplicate whatever's under the cursor. I'm shocked this requires a ;; third-party package; it should be standard. (use-package duplicate-thing :bind (("C-c u" . duplicate-thing))) ;; I can never remember the hierarchies of certain bindings, like C-x v for version control ;; stuff. Guide-key helps there. (TODO: figure out other places it'll help.) (use-package guide-key :diminish guide-key-mode :config (guide-key-mode t) (setq guide-key/guide-key-sequence '("C-x v" ;; version control "C-c a" ;; my mode-specific bindings "C-c l" ;; line-jumping "C-c o" ))) (defun em-dash () "Insert an em-dash." (interactive) (insert "—")) (defun ellipsis () "Insert an ellipsis." (interactive) (insert "…")) (defun lambduh () "Insert a lowercase lambda." (interactive) (insert "λ")) (defun open-link-list () "Open the list of captured links." (interactive) (find-file "~/Notes/links.org")) (defun open-semantic-notes () "Open my notes file." (interactive) (find-file "~/Notes/semantic.org")) (defun open-main-todo-file () "Open The List." (interactive) (find-file "~/Notes/todo.org")) (use-package org ;; Always get this from the GNU archive. :pin gnu :diminish org-indent-mode ;; Global functions that drop me into org-mode-related modes ;; are prefixed under C-c o. :bind (("C-c o c" . org-capture) ("C-c o n" . open-semantic-notes) ("C-c o t" . open-main-todo-file) ("C-c o l" . open-link-list) ("C-c o s" . org-store-link) ("C-c o a" . org-agenda) :map org-mode-map ("M-s-" . org-insert-todo-heading) ("M--" . em-dash) ("M-;" . ellipsis) ("C-c c" . org-mode-insert-code) ("C-c a s" . org-emphasize) ("C-c a r" . org-ref) ("C-c a e" . outline-show-all) ("C-c a l" . lambduh) ("C-c a t" . unindent-by-four)) :hook (org-mode . visual-line-mode) :config ;; TODO: build a real indentation solution (defun unindent-by-four () (interactive) (indent-rigidly (region-beginning) (region-end) -4)) ;; Org-mode conflicts with a lot of stuff I take for granted. (unbind-key "C-c ;" org-mode-map) (unbind-key "C-," org-mode-map) (unbind-key "M-" org-mode-map) (unbind-key "M-" org-mode-map) (let ((todo-path (expand-file-name "~/Seafile/Notes/Org/todo.org"))) (when (file-exists-p todo-path) (setq org-agenda-files (list todo-path) org-default-notes-file todo-path))) (setq org-footnote-section "" org-startup-with-inline-images t org-pretty-entities t org-ellipsis "…" org-footnote-section nil org-hide-leading-stars nil ) ;; This allegedly gets better behavior for delineating org-mode ;; emphasis. But (Monique voice) I would like to see it. (setcar (nthcdr 4 org-emphasis-regexp-components) 4) (defun org-mode-insert-code () (interactive) (org-emphasize ?~))) (setq org-capture-templates '(("l" "Link" entry (file "~/Notes/links.org") "* NEW %?\n:PROPERTIES:\n:CREATED: %U\n:END:\n%i\n"))) (defun my-elisp-mode-hook () "My elisp customizations." (electric-pair-mode 1) (add-hook 'before-save-hook 'check-parens nil t) (auto-composition-mode nil)) (add-hook 'emacs-lisp-mode-hook 'my-elisp-mode-hook) (defun open-init-file () "Open this very file." (interactive) (find-file user-init-file)) (defun kill-all-buffers () "Close all buffers." (interactive) (maybe-unset-buffer-modified) (save-some-buffers) (mapc 'kill-buffer-with-prejudice (buffer-list))) (defun split-right-and-enter () "Split the window to the right and enter it." (interactive) (split-window-right) (other-window 1)) (defun split-below-and-enter () "Split the window to the right and enter it." (interactive) (split-window-below) (other-window 1)) (defun switch-to-previous-buffer () "Switch to previously open buffer. Repeated invocations toggle between the two most recently open buffers." (interactive) (switch-to-buffer (other-buffer (current-buffer) 1))) (defun display-startup-echo-area-message () "Overrides the normally tedious error message." (message "Welcome back.")) (defun eol-then-newline () "Go to end of line then return." (interactive) (move-end-of-line nil) (newline) (indent-for-tab-command)) ;; There is an extant bug where magit-refresh prompts to save files that haven't ;; been modified. We work around this with some defadvice over maybe-unset-buffer-modified. SO: ;; https://emacs.stackexchange.com/questions/24011/make-emacs-diff-files-before-asking-to-save (autoload 'diff-no-select "diff") (defun current-buffer-matches-file-p () "Return t if the current buffer is identical to its associated file." (when (and buffer-file-name (buffer-modified-p)) (diff-no-select buffer-file-name (current-buffer) nil 'noasync) (with-current-buffer "*Diff*" (and (search-forward-regexp "^Diff finished \(no differences\)\." (point-max) 'noerror) t)))) (defun maybe-unset-buffer-modified (&optional _) "Clear modified bit on all unmodified buffers." (interactive) (dolist (buf (buffer-list)) (with-current-buffer buf (when (and buffer-file-name (buffer-modified-p) (current-buffer-matches-file-p)) (set-buffer-modified-p nil))))) ;; Don't prompt to save unmodified buffers on exit. (advice-add 'save-buffers-kill-emacs :before #'maybe-unset-buffer-modified) (defun kill-buffer-with-prejudice (&optional _) "Kill a buffer, eliding the save dialogue if there are no diffs." (interactive) (when (current-buffer-matches-file-p) (set-buffer-modified-p nil)) (kill-buffer)) ;; Define my main key bindings (bind-key "C-x k" 'kill-buffer-with-prejudice) (bind-key "C-c e" 'open-init-file) (bind-key "C-c k" 'kill-all-buffers) (bind-key "s-" 'eol-then-newline) (bind-key "C-c 5" 'query-replace-regexp) ;; stupid vestigial binding (bind-key "M-/" 'hippie-expand) (bind-key "C-c \\" 'align-regexp) (bind-key "C-c m" 'compile) (bind-key "C-c 3" 'split-right-and-enter) (bind-key "C-c 2" 'split-below-and-enter) (bind-key "C-c p" 'switch-to-previous-buffer) (bind-key "M-p" 'ace-window) (bind-key "C-c /" 'comment-or-uncomment-region) (bind-key "C-c x" 'ESC-prefix) (bind-key "M-i" 'delete-indentation) ;; When tracking down slowness, opening ivy to start these functions ;; throws off the traces. ;; (bind-key "C-c a p" 'profiler-start) ;; (bind-key "C-c a P" 'profiler-report) ;; macOS-style bindings, too (no cua-mode, it's nasty) (bind-key "C-+" 'text-scale-increase) (bind-key "C--" 'text-scale-decrease) (bind-key "s-s" 'save-buffer) (bind-key "s-c" 'kill-ring-save) (bind-key "s-v" 'yank) (bind-key "s-z" 'undo) (bind-key "s-a" 'mark-whole-buffer) (bind-key "C-<" 'beginning-of-buffer) (bind-key "s-x" 'kill-region) (bind-key "C->" 'end-of-buffer) (bind-key "M-_" 'em-dash) (bind-key "M-;" 'ellipsis) (bind-key "C-=" 'next-error) (bind-key "s-{" 'previous-buffer) (bind-key "s-}" 'next-buffer) (unbind-key "C-") ;; prevent switching to tab mode randomly (unbind-key "C-h n") ;; I have never wanted to see emacs news ever (unbind-key "C-h C-n") ;; why on earth is it bound to two keybindings?? (unbind-key "C-x C-d") ;; list-directory is utterly useless given the existence of dired (unbind-key "M-o") ;; facemenu mode is useless (unbind-key "C-x C-r") ;; as is find-file-read-only ;; I'm not made of time, I can't type "yes" for every choice (defalias 'yes-or-no-p 'y-or-n-p) (global-hl-line-mode t) ; Always highlight the current line. (show-paren-mode t) ; And point out matching parentheses. (delete-selection-mode t) ; Behave like any other sensible text editor would. (cua-mode t) ; Please let me copy/paste like everyone else. (save-place-mode) ; Remember where I was ;; Make sure that ligatures from fonts that offer them are enabled. ;; This isn't present on GNU Emacs and requires a tremendous amount ;; of munging of 'prettify-symbols-alist'. ;; (ignore-errors (mac-auto-operator-composition-mode)) (setq compilation-always-kill t ; Never prompt to kill a compilation session. compilation-scroll-output 'first-error ; Always scroll to the bottom. make-backup-files nil ; No backups, thanks. auto-save-default nil ; Or autosaves. What's the difference between autosaves and backups? create-lockfiles nil ; Emacs sure loves to put lockfiles everywhere. default-directory "~/Notes/" ; All my Notes are here. inhibit-startup-screen t ; No need to see GNU agitprop. kill-whole-line t ; Lets C-k delete the whole line require-final-newline t ; Auto-insert trailing newlines. ring-bell-function 'ignore ; Do not ding. Ever. use-dialog-box nil ; Dialogues always go in the modeline. frame-title-format "emacs – %b" ; Put something useful in the status bar. initial-scratch-message nil ; SHUT UP SHUT UP SHUT UP save-interprogram-paste-before-kill t ; preserve paste to system ring enable-recursive-minibuffers t ; don't fucking freak out if I use the minibuffer twice sentence-end-double-space nil ; are you fucking kidding me with this shit confirm-kill-processes nil ; don't whine at me when I'm quitting. mark-even-if-inactive nil ; prevent really unintuitive undo behavior user-full-name "Marcel Fries" ; it me ) (setq-default cursor-type 'bar indent-tabs-mode nil cursor-in-non-selected-windows nil) (set-fill-column 95) ;; Always trim trailing whitespace. (add-hook 'before-save-hook 'delete-trailing-whitespace) ;; (setq debug-on-error nil) ;; Show file full path in title bar (setq-default frame-title-format (list '((buffer-file-name " %f" (dired-directory dired-directory (revert-buffer-function " %b" ("%b - Dir: " default-directory))))))) ;; Make org-mode look pretty (use-package org-bullets :init (add-hook 'org-mode-hook (lambda () (org-bullets-mode 1)))) (use-package pretty-mode :init (global-pretty-mode t)) (add-hook 'org-mode-hook (lambda () "Beautify Org Checkbox Symbol" (push '("[ ]" . "☐") prettify-symbols-alist) (push '("[X]" . "☑" ) prettify-symbols-alist) (push '("[-]" . "❍" ) prettify-symbols-alist) (prettify-symbols-mode))) (defface org-checkbox-done-text '((t (:foreground "#71696A" :strike-through t))) "Face for the text part of a checked org-mode checkbox.") (font-lock-add-keywords 'org-mode `(("^[ \t]*\\(?:[-+*]\\|[0-9]+[).]\\)[ \t]+\\(\\(?:\\[@\\(?:start:\\)?[0-9]+\\][ \t]*\\)?\\[\\(?:X\\|\\([0-9]+\\)/\\2\\)\\][^\n]*\n\\)" 1 'org-checkbox-done-text prepend)) 'append) ;; Configure backup files to be in a dedicated directory (setq backup-directory-alist `(("." . "~/.emacs-backup"))) (setq backup-by-copying t) (setq delete-old-versions t kept-new-versions 6 kept-old-versions 2 version-control t) ;; put your css files there (defvar org-theme-css-dir "~/.emacs.d/org-css/") (defun toggle-org-custom-inline-style () (interactive) (let ((hook 'org-export-before-parsing-hook) (fun 'set-org-html-style)) (if (memq fun (eval hook)) (progn (remove-hook hook fun 'buffer-local) (message "Removed %s from %s" (symbol-name fun) (symbol-name hook))) (add-hook hook fun nil 'buffer-local) (message "Added %s to %s" (symbol-name fun) (symbol-name hook))))) (defun org-theme () (interactive) (let* ((cssdir org-theme-css-dir) (css-choices (directory-files cssdir nil ".css$")) (css (completing-read "theme: " css-choices nil t))) (concat cssdir css))) (defun set-org-html-style (&optional backend) (interactive) (when (or (null backend) (eq backend 'html)) (let ((f (or (and (boundp 'org-theme-css) org-theme-css) (org-theme)))) (if (file-exists-p f) (progn (set (make-local-variable 'org-theme-css) f) (set (make-local-variable 'org-html-head) (with-temp-buffer (insert "\n") (buffer-string))) (set (make-local-variable 'org-html-head-include-default-style) nil) (message "Set custom style from %s" f)) (message "Custom header file %s doesnt exist"))))) ;; Remove the brackets around timestamps in org-mode exports (add-to-list 'org-export-filter-timestamp-functions #'endless/filter-timestamp) (defun endless/filter-timestamp (trans back _comm) "Remove <> around time-stamps." (pcase back ((or `jekyll `html) (replace-regexp-in-string "&[lg]t;" "" trans)) (`latex (replace-regexp-in-string "[<>]" "" trans)))) (setq-default org-display-custom-times t) ;;; Before you ask: No, removing the <> here doesn't work. (setq org-time-stamp-custom-formats '("<%a %d.%m.%Y>" . "<%d.%m.%y %H:%M>")) ;; goodbye, thanks for reading (provide 'init) ;;; init.el ends here