;;; init.el --- Init -*- no-byte-compile: t; lexical-binding: t; -*- (add-to-list 'load-path "~/.emacs.d/elisp") (add-to-list 'load-path "~/.emacs.d/site-lisp") (setq package-list '(avy devdocs diminish direnv ednc elfeed elfeed-protocol enwc emms expand-region exwm geiser geiser-guile git-gutter git-gutter-fringe org-roam go-mode guix pdf-tools magit markdown-mode mastodon nix-mode nov pass pinentry tldr transpose-frame ws-butler)) (package-initialize) (unless package-archive-contents (package-refresh-contents)) ;; install the missing packages (dolist (package package-list) (unless (package-installed-p package) (package-install package))) (load-theme 'BOGO t) (use-package exwm :init (setq exwm-workspace-number 1 exwm-input-global-keys `(([?\s-r] . exwm-reset) ;; s-r: Reset (to line-mode). ;; s-N: Switch to certain workspace. ,@(mapcar (lambda (i) `(,(kbd (format "s-%d" i)) . (lambda () (interactive) (tab-bar-select-tab ,i)))) (number-sequence 0 9)))) (exwm-input-set-key (kbd "s-f") 'exwm-layout-toggle-fullscreen) (exwm-input-set-key (kbd "s-t") 'exwm-floating-toggle-floating) (exwm-input-set-key (kbd "") (lambda () (interactive) (shell-command "scrot -F '/home/mccd/screenshots/%Y-%m-%d_$wx$h.png'"))) (exwm-input-set-key (kbd "C-") (lambda () (interactive) (shell-command "scrot -s -F '/home/mccd/screenshots/%Y-%m-%d_$wx$h.png'"))) (exwm-input-set-key (kbd "") (lambda () (interactive) (let ((inhibit-message t)) (shell-command "amixer set Master toggle 1> /dev/null")))) (exwm-input-set-key (kbd "") (lambda () (interactive) (let ((inhibit-message t)) (shell-command "amixer set Master 10%+ 1> /dev/null")))) (exwm-input-set-key (kbd "") (lambda () (interactive) (let ((inhibit-message t)) (shell-command "amixer set Master 10%- 1> /dev/null")))) (exwm-input-set-key (kbd "") (lambda () (interactive) (let ((inhibit-message t)) (shell-command "brightnessctl set 5%-")))) (exwm-input-set-key (kbd "") (lambda () (interactive) (let ((inhibit-message t)) (shell-command "brightnessctl set +5%")))) (exwm-input-set-key (kbd "s-") (lambda (cmd) ;; s-&: Launch application. (interactive (list (read-shell-command "$ "))) (start-process-shell-command cmd nil cmd))) (push ?\C-\\ exwm-input-prefix-keys) (shell-command "xrdb ~/.Xresources") :custom (ring-bell-function 'ignore) (default-directory "/home/mccd/") (focus-follows-mouse t) (comment-multi-line t) (sentence-end-double-space nil) (require-final-newline t) (kill-do-not-save-duplicates t) (comment-empty-lines t) (lazy-highlight-initial-delay 0) (mouse-autoselect-window t) :hook (;; Make sure titles match X window (exwm-init . (lambda () (interactive) (load-library "/home/mccd/.emacs.d/auth.el.gpg"))) (exwm-update-title . (lambda () (exwm-workspace-rename-buffer (concat exwm-class-name ":" (if (<= (length exwm-title) 50) exwm-title (concat (substring exwm-title 0 49) "...")))))) (exwm-update-class . (lambda () (exwm-workspace-rename-buffer (concat exwm-class-name ":" (if (<= (length exwm-title) 50) exwm-title (concat (substring exwm-title 0 49) "...")))))))) (use-package diminish :ensure t) (use-package ediff :custom (ediff-window-setup-function #'ediff-setup-windows-plain) (ediff-split-window-function #'split-window-horizontally)) (use-package emacs :init (set-scroll-bar-mode nil) (fset 'yes-or-no-p 'y-or-n-p) (setq-default fill-column 80 line-spacing 10 cursor-type 'box cursor-in-non-selected-windows nil frame-title-format '("%b")) :custom (scroll-bar-mode nil) (highlight-nonselected-windows nil) (default-frame-alist '((font . "Iosevka SS13 Extended") (height . 150) (line-spacing . 0.5))) (undo-limit 10000000) (undo-outer-limit 20000000) (create-lockfiles nil) (idle-update-delay 1.0) (auto-save-list-file-prefix (expand-file-name "autosave/" user-emacs-directory)) (tramp-auto-save-directory (expand-file-name "tramp-autosave/" user-emacs-directory)) ;; Auto save options (kill-buffer-delete-auto-save-files t) ;; Revert other buffers (e.g, Dired) (global-auto-revert-non-file-buffers t) ;;; recentf ;; `recentf' is an Emacs package that maintains a list of recently ;; accessed files, making it easier to reopen files you have worked on ;; recently. (recentf-max-saved-items 300) ; default is 20 (recehvntf-auto-cleanup 'mode) ;;; saveplace ;; `save-place-mode` enables Emacs to remember the last location within a file ;; upon reopening. This feature is particularly beneficial for resuming work at ;; the precise point where you previously left off. (save-place-file (expand-file-name "saveplace" user-emacs-directory)) (save-place-limit 600) ;;; savehist ;; `savehist` is an Emacs feature that preserves the minibuffer history between ;; sessions. It saves the history of inputs in the minibuffer, such as commands, ;; search strings, and other prompts, to a file. This allows users to retain ;; their minibuffer history across Emacs restarts. (history-length 300) (savehist-save-minibuffer-history t) ;; Default ;; Resizing the Emacs frame can be costly when changing the font. Disable this ;; to improve startup times with fonts larger than the system default. (frame-resize-pixelwise t) ;; However, do not resize windows pixelwise, as this can cause crashes in some ;; cases when resizing too many windows at once or rapidly. (window-resize-pixelwise nil) (resize-mini-windows 'grow-only) (fast-but-imprecise-scrolling t) (scroll-step 1) (scroll-conservatively 100000) (scroll-preserve-screen-position 1) (scroll-margin 0) (blink-cursor-mode -1) (read-process-output-max (* 512 1024)) (x-stretch-cursor t) (tab-always-indent 'complete) (text-mode-ispell-word-completion nil) (read-extended-command-predicate #'command-completion-default-include-p )) (use-package vc-hooks :custom (vc-make-backup-files nil) (vc-follow-symlinks t)) (use-package tab-bar :bind (("C-" . nil)) :custom (tab-bar-auto-width t) (tab-bar-close-button-show nil) (tab-bar-format '(tab-bar-format-tabs tab-bar-format-align-right tab-bar-format-global)) :init (tab-bar-mode) (setq tab-bar-new-tab-choice "*scratch*")) (use-package project :ensure t :bind (:map project-prefix-map ("b" . project-list-buffers)) :config (add-to-list 'project-switch-commands '(project-list-buffers "Buffers")) (add-to-list 'project-switch-commands '(project-kill-buffers "Kill Buffers")) (add-to-list 'project-switch-commands '(project-shell "Shell"))) (use-package transpose-frame :ensure t :defer t :bind (("C-x 5 t" . 'transpose-frame))) (use-package ffap :defer t :bind (("C-x C-f" . find-file-at-point)) :init (ffap-bindings) :custom (ffap-machine-p-known 'reject) (ffap-require-prefix nil)) (use-package tldr :ensure t :defer t) (use-package mouse :custom (mouse-yank-at-point t)) (use-package time :init (display-time-mode) :custom (display-time-format "%a, %F %R")) (use-package geiser :ensure t :hook (geiser-mode . (lambda () (keymap-set geiser-mode-map "C-." nil)))) ;; (use-package corfu ;; :defer t ;; ;; Optional customizations ;; :custom ;; (corfu-cycle t) ;; Enable cycling for `corfu-next/previous' ;; ;; (corfu-auto t) ;; Enable auto completion ;; ;; (corfu-quit-at-boundary nil) ;; Never quit at completion boundary ;; (corfu-quit-no-match t) ;; Never quit, even if there is no match ;; (corfu-preview-current t) ;; Disable current candidate preview ;; (corfu-preselect 'prompt) ;; Preselect the prompt ;; (corfu-quit-no-match t) ;; (corfu-on-exact-match 'show) ;; Configure handling of exact matches ;; ;; ;; Recommended: Enable Corfu globally. This is recommended since Dabbrev can ;; ;; be used globally (M-/). See also the customization variable ;; ;; `global-corfu-modes' to exclude certain modes. ;; :init ;; (global-corfu-mode) ;; (corfu-popupinfo-mode)) (use-package dabbrev ;; Swap M-/ and C-M-/ :bind (("M-/" . dabbrev-completion) ("C-M-/" . dabbrev-expand)) :config (add-to-list 'dabbrev-ignored-buffer-regexps "\\` ") ;; Since 29.1, use `dabbrev-ignored-buffer-regexps' on older. (add-to-list 'dabbrev-ignored-buffer-modes 'doc-view-mode) (add-to-list 'dabbrev-ignored-buffer-modes 'pdf-view-mode) (add-to-list 'dabbrev-ignored-buffer-modes 'tags-table-mode)) (use-package delsel :init (delete-selection-mode)) (use-package frame :custom (window-divider-default-right-width 2) :init (window-divider-mode)) (use-package window :custom (switch-to-buffer-obey-display-actions t) :bind (("C-x C-}" . enlarge-window-horizontally) ("C-x C-o" . other-window) :repeat-map move-up-repeat-map (("v" . scroll-down-command)) :repeat-map move-down-repeat-map (("v" . scroll-up-command))) :config (add-to-list 'display-buffer-alist '("*Async Shell Command*" display-buffer-no-window (nil)))) (use-package files :custom (make-backup-files nil) (backup-directory-alist `(("." . ,(expand-file-name "backup" user-emacs-directory)))) (tramp-backup-directory-alist backup-directory-alist) (backup-by-copying-when-linked t) (backup-by-copying t) ; Backup by copying rather renaming (delete-old-versions t) ; Delete excess backup versions silently (version-control t) ; Use version numbers for backup files (kept-new-versions 10) (kept-old-versions 10) ;; acceptable since it will redirect you to the existing buffer regardless. (find-file-suppress-same-file-warnings t) ;; Resolve symlinks when opening files, so that any operations are conducted ;; from the file's true directory (like `find-file'). (find-file-visit-truename t) ;; Skip confirmation prompts when creating a new file or buffer (confirm-nonexistent-file-or-buffer nil)) (use-package uniquify :custom (uniquify-buffer-name-style 'forward)) (use-package org-roam :ensure t :config (org-roam-db-autosync-mode) :bind (("C-c n n" . org-roam-node-find) ("C-c n i" . org-roam-node-insert) ("C-c n c" . org-roam-capture)) :custom (org-roam-directory "~/org-roam") (org-roam-capture-templates '(("d" "default" plain "%?" :target (file+head "%<%Y%m%d%H%M%S>-${slug}.org" "* ${title}\n%U\n") :unnarrowed t)))) (use-package direnv :ensure t :custom (direnv-always-show-summary nil) :config (direnv-mode)) (use-package minibuffer :defer t :custom (completion-cycle-threshold 5) (enable-recursive-minibuffers t) (completion-auto-help 'always) (completions-max-height 10) (completions-format 'horizontal) (completion-auto-select 'second-tab) (completion-ignore-case t)) (use-package pass :defer t :ensure t) (defun elfeed-refresh () (interactive) (cl-loop for entry in (elfeed-search-selected) do (elfeed-untag-1 entry 'unread)) (elfeed-protocol-fever-reinit "https://freshrss@rss.mccd.space")) (use-package elfeed :ensure t :bind (:map elfeed-search-mode-map ("G" . elfeed-refresh))) (use-package elfeed-protocol :after elfeed :ensure t :demand t :custom (elfeed-use-curl t) (elfeed-curl-extra-arguments '("--insecure")) (elfeed-protocol-fever-update-unread-only t) (elfeed-log-level 'debug) (elfeed-protocol-fever-fetch-category-as-tag t) (elfeed-protocol-feeds '(("fever+https://freshrss@rss.mccd.space" :api-url "https://rss.mccd.space/api/fever.php" :password freshrss-password))) ;; enable elfeed-protocol (elfeed-protocol-enabled-protocols '(fever)) :config (elfeed-set-timeout 36000) (elfeed-protocol-enable)) (use-package select :config (setq x-select-enable-clipboard t x-select-enable-primary t)) (use-package pdf-tools :defer t :ensure t) (use-package imenu :defer t :bind (("M-i" . imenu)) :custom (imenu-auto-rescan t)) (use-package nov :defer t :ensure t :init (add-to-list 'auto-mode-alist '("\\.epub'" . nov-mode))) (use-package ednc :after exwm :diminish ednc-mode :ensure t :config (ednc-mode)) (use-package scheme :bind (:map scheme-mode-map ("C-." . nil))) (use-package repeat :demand t :bind (("C-z" . repeat) :repeat-map move-repeat-map ("n" . next-line) ("e" . move-end-of-line) ("a" . move-beginning-of-line) ("b" . backward-char) ("f" . forward-char) ("p" . previous-line) ("o" . avy-goto-char-timer) :repeat-map sexp-move-repeat-map ("n" . forward-list) ("p" . backward-list) ("e" . end-of-defun) ("a" . beginning-of-defun) ("d" . down-list) ("u" . backward-up-list) ("f" . forward-sexp) ("b" . backward-sexp) :repeat-map forward-words-repeat-map ("f" . forward-word) ("e" . forward-sentence) ("a" . backward-sentence) ("b" . backward-word) :repeat-map undo-redo-repeat-map ("/" . undo) ("?" . undo-redo) :repeat-map kill-lines-repeat-map ("k" . kill-line) :repeat-map kill-lines-org-repeat-map ("k" . org-kill-line) :repeat-map backspace-repeat-map ("" . backward-kill-word) :repeat-map delete-repeat-map ("d" . delete-char) :repeat-map kill-word-repeat-map ("d" . kill-word)) :custom (repeat-on-final-keystroke t) ;; When you have repeat maps while in the minibuffer, repeat-echo-message ;; becomes distracting (repeat-echo-function 'repeat-echo-mode-line) (set-mark-command-repeat-pop t) (repeat-exit-key (kbd "i")) :config (repeat-mode 1)) (use-package battery :init (display-battery-mode t)) (use-package comint :bind ((:repeat-map repeat-hist-line-map ("n" . comint-next-input) ("p" . comint-previous-input)))) (use-package eww :defer t :custom (eww-auto-rename-buffer t) (eww-buffer-name-length 20) :bind ((:map eww-mode-map ("+" . image-increase-size)))) (use-package enwc :demand t :ensure t :custom (enwc-default-backend 'nm) (enwc-wired-device "lo") (enwc-wireless-device "wlp61s0") :init (defun list-notifications () (mapconcat #'ednc-format-notification (ednc-notifications) "")) (nconc global-mode-string '((:eval (list-notifications)))) ; or stack (add-hook 'ednc-notification-presentation-functions (lambda (&rest _) (force-mode-line-update t)))) (use-package git-gutter :defer t :diminish git-gutter-mode :hook (go-mode . git-gutter-mode) (prog-mode . git-gutter-mode) (nix-mode . git-gutter-mode) :bind (("C-c g p" . git-gutter:previous-hunk) ("C-c g n" . git-gutter:next-hunk) ("C-c g s" . git-gutter:stage-hunk) ("C-c g r" . git-gutter:revert-hunk) ("C-c g =" . git-gutter:popup-hunk) :repeat-map git-gutter-repeat-map ("n" . git-gutter:next-hunk) ("=" . git-gutter:popup-hunk) ("s" . git-gutter:stage-hunk) ("r" . git-gutter:revert-hunk) ("p" . git-gutter:previous-hunk)) :ensure t) (use-package git-gutter-fringe :defer t :ensure t) (use-package grep :defer t :custom (grep-template (string-join '("ugrep" "--color=never" "--ignore-binary" "--ignore-case" "--include=" "--line-number" "--null" "--recursive" "--regexp=") " "))) (use-package pinentry :ensure t :init (pinentry-start)) (use-package org-crypt :ensure nil :custom (auth-source-debug t) :init (setenv "GPG_AGENT_INFO" nil) (org-crypt-use-before-save-magic)) (use-package epa-file :custom (epa-pinentry-mode 'loopback) (epa-file-select-keys nil) (epg-gpg-program "gpg") :init (epa-file-enable)) (use-package exwm-mff :demand t :load-path "elisp" :hook ((server-after-make-frame . exwm-mff-mode))) (use-package isearch :defer t :bind ((:repeat-map repeat-isearch-map ("o" . avy-isearch) ("s" . isearch-repeat-forward) ("r" . isearch-repeat-backward)))) (use-package avy :defer t :ensure t :custom (avy-timeout-seconds 0.2) :bind ("M-o" . avy-goto-char-timer) :config (set-face-attribute 'avy-lead-face nil :inherit 'fixed-pitch) (set-face-attribute 'avy-lead-face-0 nil :inherit 'fixed-pitch) (set-face-attribute 'avy-lead-face-1 nil :inherit 'fixed-pitch) (set-face-attribute 'avy-lead-face-2 nil :inherit 'fixed-pitch) (setf (alist-get ? avy-dispatch-alist) 'avy-action-mark-to-char) (define-key isearch-mode-map (kbd "M-o") 'avy-isearch) (defun avy-action-kill-whole-line (pt) (save-excursion (goto-char pt) (kill-whole-line)) (select-window (cdr (ring-ref avy-ring 0))) t) (setf (alist-get ?k avy-dispatch-alist) 'avy-action-kill-stay (alist-get ?K avy-dispatch-alist) 'avy-action-kill-whole-line) (defun avy-action-copy-whole-line (pt) (save-excursion (goto-char pt) (cl-destructuring-bind (start . end) (bounds-of-thing-at-point 'line) (copy-region-as-kill start end))) (select-window (cdr (ring-ref avy-ring 0))) t) (defun avy-action-yank-whole-line (pt) (avy-action-copy-whole-line pt) (save-excursion (yank)) t) (setf (alist-get ?y avy-dispatch-alist) 'avy-action-yank (alist-get ?w avy-dispatch-alist) 'avy-action-copy (alist-get ?W avy-dispatch-alist) 'avy-action-copy-whole-line (alist-get ?Y avy-dispatch-alist) 'avy-action-yank-whole-line) (defun avy-action-copy-whole-line (pt) (save-excursion (goto-char pt) (cl-destructuring-bind (start . end) (bounds-of-thing-at-point 'line) (copy-region-as-kill start end))) (select-window (cdr (ring-ref avy-ring 0))) t) (defun avy-action-yank-whole-line (pt) (avy-action-copy-whole-line pt) (save-excursion (yank)) t) (setf (alist-get ?y avy-dispatch-alist) 'avy-action-yank (alist-get ?w avy-dispatch-alist) 'avy-action-copy (alist-get ?W avy-dispatch-alist) 'avy-action-copy-whole-line (alist-get ?Y avy-dispatch-alist) 'avy-action-yank-whole-line) (defun avy-action-teleport-whole-line (pt) (avy-action-kill-whole-line pt) (save-excursion (yank)) t) (setf (alist-get ?t avy-dispatch-alist) 'avy-action-teleport (alist-get ?T avy-dispatch-alist) 'avy-action-teleport-whole-line) (defun avy-action-mark-to-char (pt) (activate-mark) (goto-char pt)) (setf (alist-get ? avy-dispatch-alist) 'avy-action-mark-to-char)) (use-package flymake :defer t :bind (:repeat-map flymake-repeat-map ("]" . flymake-goto-next-error) ("[" . flymake-goto-prev-error)) :hook ((flymake-mode . (lambda () (define-key flymake-mode-map (kbd "C-c ]") 'flymake-goto-next-error) (define-key flymake-mode-map (kbd "C-c [") 'flymake-goto-prev-error))))) (use-package expand-region :ensure t :bind ("C-." . er/expand-region) ("C--" . er/contract-region)) (use-package devdocs :ensure t :defer t :bind ("C-h D" . devdocs-lookup)) (use-package guix :ensure t :bind (("C-c C-g g" . guix) ("C-c C-g o" . (lambda () (interactive) (sudo-shell-command "guix system reconfigure /home/mccd/system/os/config.scm"))) ("C-c C-g h" . (lambda () (interactive) (async-shell-command "guix home reconfigure ~/system/home/config.scm"))))) (use-package ws-butler :ensure t :diminish ws-butler-mode :hook ((prog-mode . ws-butler-mode) (css-mode . ws-butler-mode) (javascript-mode . ws-butler-mode) (html-mode . ws-butler-mode))) (use-package sgml-mode :init (add-to-list 'auto-mode-alist '("\\.njk\\'" . html-mode)) (add-to-list 'auto-mode-alist '("\\.tmpl\\'" . html-mode)) (add-to-list 'auto-mode-alist '("\\.js.tmpl\\'" . javascript-mode)) :bind (:map html-mode-map ("M-" . nil) ("M-o" . nil) ("M-i" . nil)) :hook (javascript-mode . (lambda ((interactive) (setq-local tab-width 2) (setq sgml-basic-offset 2 indent-tabs-mode t)))) (css-mode . (lambda ((interactive) (setq-local tab-width 2) (setq sgml-basic-offset 2 indent-tabs-mode t)))) (html-mode . (lambda () (interactive) (setq-local tab-width 2) (setq sgml-basic-offset 2 indent-tabs-mode t)))) (use-package eldoc :ensure t :demand t :diminish eldoc-mode :init (defun kill-help-and-eldoc-buffers () "Removes info windows." (interactive) (kill-matching-buffers "^\\*compilation\\*" nil t) (kill-matching-buffers "^\\*Help\\*" nil t) (kill-matching-buffers "^\\*eldoc" nil t)) (global-eldoc-mode -1) :bind ("C-h ." . eldoc) ("C-h ," . kill-help-and-eldoc-buffers) :custom (eldoc-print-after-edit nil) (eldoc-echo-area-display-truncation-message t) (eldoc-echo-area-prefer-doc-buffer t) (eldoc-documentation-strategy 'eldoc-documentation-compose-eagerly) (eldoc-echo-area-use-multiline-p 'truncate-sym-name-if-fit)) (use-package markdown-mode :defer t :ensure t :custom (markdown-list-item-bullets '("-")) :config (set-face-attribute 'markdown-code-face nil :inherit 'fixed-pitch) :hook ((markdown-mode . variable-pitch-mode) (markdown-mode . markdown-toggle-url-hiding) (markdown-mode . markdown-toggle-markup-hiding) (markdown-mode . visual-line-mode))) (use-package elec-pair :init (electric-pair-mode 1) :bind ("M-{" . insert-pair) ("M-\"" . insert-pair) ("M-\'" . insert-pair) ("M-<" . insert-pair) ("M-[" . insert-pair)) (use-package dbus) (use-package ido :defer t :init (ido-mode t) (ido-everywhere) :bind (:repeat-map ido-completion-mode-map ("s" . ido-next-match) ("r" . ido-prev-match)) :custom (ido-enable-flex-matching nil) (ido-enable-regexp nil) (ido-max-prospects 12) (ido-max-window-height 1) (ido-decorations '(" " " " ", " " , ..." " [" "] " " [No match]" " [Matched]" " [Not readable]" " [Too big]" " [Confirm]"))) (use-package completion :custom (read-file-name-completion-ignore-case t) (read-buffer-completion-ignore-case t) (completion-styles '(basic initials partial-completion substring)) (completion-category-defaults nil) ; ehh (completion-show-help 't) (completion-category-overrides '((file (styles basic partial-completion))))) (use-package calendar :defer t :bind (("C-c d" . calendar)) :custom (calendar-week-start-day 1) (org-agenda-include-diary t) :config (calendar-set-date-style 'european) (setq calendar-week-start-day 1)) (use-package nix-mode :ensure t :defer t :hook ((nix-mode . (lambda () (add-hook 'before-save-hook #'nix-format-buffer nil t))) (nix-mode . nix-prettify-mode))) (use-package eglot :config ;; Ensure `nil` is in your PATH. (add-to-list 'eglot-server-programs '(nix-mode . ("nil"))) (fset #'jsonrpc--log-event #'ignore) :bind (:map eglot-mode-map ("C-c r" . eglot-rename) ("C-c o" . eglot-code-actions)) :custom (setq jsonrpc-event-hook nil) :hook (eglot-managed-mode . (lambda () (eldoc-mode -1))) (css-mode . eglot-ensure) (html-mode . eglot-ensure) (javascript-mode . eglot-ensure) (nix-mode . eglot-ensure) (go-mode . eglot-ensure)) (use-package tab-line :custom (tab-line-close-button nil) (tab-line-new-button-show nil) :bind (:map eww-mode-map ("M-" . eww-open-in-new-buffer)) :hook (eww-mode . tab-line-mode)) (use-package go-mode :ensure t :demand t :hook ((go-mode . (lambda () (add-hook 'before-save-hook 'eglot-format-buffer nil t))) (go-mode . (lambda () (add-hook 'before-save-hook 'eglot-format-buffer nil t))) (go-mode . (lambda () (setq-local compile-command "go build ."))))) (use-package erc :ensure t :custom (erc-autojoin-channels-alist '(("Libera.Chat" "#emacs" "#permacomputing" "#oldcomputerchallenge" "#go-nuts" "#netbsd" "#guile" "#nonguix" "#guix"))) :bind ("\C-ceo" . (lambda () "erc - join osm" (interactive) (erc :server "irc.oftc.net" :port "6667" :nick "mccd"))) ("\C-cel" . (lambda () "erc - join libera chat" (interactive) (erc :server "irc.libera.chat" :port "6667" :nick "mccd" :password libera-chat-pass)))) ;; Org (use-package org :ensure t :init (defun sync-org () (interactive) (async-shell-command "org-sync")) :custom (org-complete-tags-always-offer-all-agenda-tags nil) (org-use-fast-tag-selection (quote auto)) (org-refile-targets '((nil :maxlevel . 4) (org-agenda-files :maxlevel . 4))) (org-default-notes-file "~/personal-db/notes/lore.org") (org-hide-emphasis-markers t) (org-link-keep-stored-after-insertion t) (org-startup-folded t) (org-hierarchical-todo-statistics t) (org-refile-allow-creating-parent-nodes 'confirm) (org-outline-path-complete-in-steps nil) (org-refile-use-outline-path 'file) (org-log-done 'time) (org-indent-line 2) (org-tags-column 0) (org-pretty-entities t) (org-todo-keywords '((sequence "TODO(t)" "|" "DONE(d)") (sequence "UNREAD(u)" "READING(e)" "|" "READ(r)") (sequence "|" "CANCELED(c)"))) (org-agenda-custom-commands '(("t" "Regular" todo "TODO") ("b" "Books" todo "UNREAD") ("n" "Agenda and all TODOs" ((agenda "today") (todo "TODO"))) ("p" "Pivå and Agenda" ((tags-todo "+piva") (agenda ""))))) (org-priority-lowest ?E) (org-priority-faces '((?A . (:weight bold :inherit org-priority)) (?D . (:foreground "gray50" :inherit org-priority)) (?E . (:foreground "gray60" :inherit org-priority)))) :init (add-to-list 'auto-mode-alist '("\\.org\\'" . org-mode)) (org-babel-do-load-languages 'org-babel-load-languages '((shell . t) (rec .t))) :config (set-face-attribute 'org-block nil :inherit 'fixed-pitch) (set-face-attribute 'org-code nil :inherit 'fixed-pitch) (set-face-attribute 'org-table nil :foreground "black" :inherit 'fixed-pitch) (set-face-attribute 'org-date nil :inherit 'fixed-pitch) (set-face-attribute 'org-tag nil :inherit 'fixed-pitch) (set-face-attribute 'org-drawer nil :inherit 'fixed-pitch) (set-face-attribute 'org-property-value nil :inherit 'fixed-pitch) (set-face-attribute 'org-special-keyword nil :inherit 'fixed-pitch) (set-face-attribute 'org-meta-line nil :inherit 'fixed-pitch) :hook ((org-mode . org-indent-mode) (org-capture-after-finalize . sync-org) (org-mode . flyspell-mode) (org-mode . (lambda () (auto-revert-mode 1))) (org-mode . visual-line-mode) (org-mode . variable-pitch-mode))) (use-package org-id :after org :init (defun me/org-id-insert-link () "Insert at point a link to any heading from `org-agenda-files'." (interactive) (let ((buffer-pos (org-id-find (org-id-get-with-outline-path-completion '((nil :maxlevel . 5) (org-agenda-files :maxlevel . 5)))))) (save-excursion (with-current-buffer (get-file-buffer (car buffer-pos)) (org-with-wide-buffer (goto-char (cdr buffer-pos)) (call-interactively 'org-store-link)))) (org-insert-all-links 1 "" " "))) :bind (:map org-mode-map (("C-c i" . me/org-id-insert-link))) :custom (org-id-link-to-org-use-id 'create-if-interactive-and-no-custom-id)) (use-package bookmark :after (org org-id) :init (defun handle-org-bookmark (bookmark) "Handle bookmarks which have Org links." (org-link-open-from-string (alist-get 'org-link bookmark))) (defun bookmark-org-link-to-here () "Bookmark the org link that `org-store-link' would make." (interactive) (when-let ((link (substring-no-properties (org-store-link nil)))) `(,(when (string-match org-link-bracket-re link) (match-string-no-properties 2 link)) (org-link . ,link) (handler . handle-org-bookmark) (position . ,(point)) ,@(when buffer-file-name `((filename . ,(abbreviate-file-name buffer-file-name))))))) (defun use-org-link-for-bookmarks () "Configure bookmarks to current buffer to use Org links." (interactive) (setq-local bookmark-make-record-function #'bookmark-org-link-to-here)) :hook ((org-mode . use-org-link-for-bookmarks))) (use-package ol :after (org-compat org-macs org-fold) :init ;; Allows creating keyboard macros in the text ;; (org-link-set-parameters "kbd" :follow (lambda (kmacro) (kmacro-call-macro nil t nil (kbd kmacro)))) (org-link-set-parameters "goto" :follow (lambda (keyword) (lgrep keyword "*.org" "~/personal-db/notes" nil))) :bind (("C-c l" . org-store-link))) (use-package subr :bind (("C-c C-o" . org-open-at-point-global))) (use-package org-capture :defer t :custom (org-capture-templates '(("t" "Todo" entry (file+headline "~/personal-db/notes/agenda.org" "Todo") "* TODO %?\n%i\n%a\n " :empty-lines 1) ("w" "Work Todo" entry (file+headline "~/personal-db/notes/agenda.org" "Todo") "* TODO %? :piva:\n:PROPERTIES:\n:CATEGORY: Pivå\n:END:\n%i\n%a\n " :empty-lines 1) ("b" "Book" entry (file+olp "~/personal-db/notes/lore.org" "References" "Books") "* UNREAD %?\n:PROPERTIES:\n:AUTHOR:\n:END:\n " :empty-lines 1) ("f" "Formula" entry (file+headline "~/personal-db/notes/lore.org" "Formulas") "* %?\n%U" :empty-lines 1) ("i" "Film" entry (file+olp "~/personal-db/notes/lore.org" "References" "Films") "* %?\n:PROPERTIES:\n:DIRECTOR:\n:STATE: unwatched\n:END:\n " :empty-lines 1) ("g" "Log Coffee" entry (file+datetree "~/personal-db/notes/log.org") "* Brewed pour-over :coffee:cyprus:paramytha:\n:PROPERTIES:\n:WATER: 320g\n:COFFEE: 25g\n:TYPE: [[id:504d708d-b90f-410e-af7f-ac8c6896ee0d][Orange County]]\n:TEMP: 90\n:RATING:\n:PREHEAT: YES\n:TOTALBREWTIME:\n:CLICKS:\n:END:\n%T\n*** Observations " :empty-lines 0 :prepend t) ("r" "Reference" entry (file+headline "~/personal-db/notes/lore.org" "References") "* %?\n%i\n%a\n " :empty-lines 1) ("e" "Event" entry (file+headline "~/personal-db/notes/agenda.org" "Event") "* %?\n%^T" :empty-lines 1) ("a" "Article" entry (file+olp "~/personal-db/notes/lore.org" "References" "Articles") "* %?\n:PROPERTIES:\n:author:\n:END:\n " :empty-lines 1) ("l" "Log" entry (file+datetree "~/personal-db/notes/log.org") "** %? :sweden:bastad:\n%T" :prepend t :empty-lines 0) ("n" "Inbox" entry (file+headline "~/personal-db/notes/inbox.org" "Inbox") "* %?\n%U\n%a\n" :empty-lines 1))) :bind (("C-c c" . org-capture))) (use-package org-agenda :defer t :custom (org-agenda-files '("/home/mccd/personal-db/notes/cal-ios.org" "/home/mccd/dev/sustainably/README.org" "/home/mccd/dev/sustainably/exploration.org" "/home/mccd/system/guix.org" "/home/mccd/personal-db/notes/agenda.org" "/home/mccd/personal-db/notes/log.org" "/home/mccd/personal-db/notes/log.old.org" "/home/mccd/personal-db/notes/inbox.org" "/home/mccd/personal-db/notes/lore.org")) (org-agenda-remove-tags t) (org-agenda-inhibit-startup nil) :config (advice-add 'org-agenda-quit :before (lambda () (interactive) (org-save-all-org-buffers))) :bind (("C-c a" . org-agenda) :map org-agenda-mode-map ("" . org-agenda-tree-to-indirect-buffer))) (use-package autorevert :diminish auto-revert-mode :custom ;;; Auto revert ;; Auto-revert in Emacs is a feature that automatically updates the ;; contents of a buffer to reflect changes made to the underlying file ;; on disk. (revert-without-query (list ".")) ; Do not prompt (auto-revert-stop-on-user-input nil) (auto-revert-verbose t) (auto-save-default t) (auto-save-include-big-deletions t)) (use-package appt :after (org-agenda dbus) :demand t ;; Taken from https://igormelo.org/you_dont_need_org_alert.html ;; Enables notifications for org :custom (appt-message-warning-time 15) (appt-display-interval 5) (appt-display-mode-line nil) (appt-disp-window-function (lambda (remaining new-time msg) (notifications-notify :title (format "Event in %s min" remaining) :body msg :app-name "Agenda" :app-icon nil :timeout 300000 :urgency 'critical))) :config (define-advice appt-activate (:after (&optional _arg) hold-your-horses) "`appt-activate' is too eager, rein it in." (remove-hook 'write-file-functions #'appt-update-list) (when (timerp appt-timer) (timer-set-time appt-timer (current-time) 600))) (define-advice appt-check (:before (&optional _force) from-org-agenda) "Read events from Org agenda if possible." (and (featurep 'org-agenda) (ignore-errors (let ((inhibit-message t)) (org-agenda-to-appt t))))) (appt-activate t)) (use-package magit :ensure t :demand t :bind (("C-x p m" . magit-project-status) :map project-prefix-map ("m" . magit-project-status)) :config (add-to-list 'project-switch-commands '(magit-project-status "magit"))) (use-package skeleton :custom (skeleton-end-hook nil) :init (define-skeleton s/scheme-define-module "Define module" "Module name: " > "(define-module (" str ")" \n > "#:use-module (gnu packages))" \n ) (define-skeleton s/scheme-procedure "Procedure" "Procedure name: " > "(define ("str " " - @ ")" \n > ")" ) (define-skeleton s/scheme-define "G-expression" "Definition name: " > "(define " str \n > - @ ")" ) (define-skeleton s/guix-gexpr "G-expression" nil > "(with-imported-module" \n > "'((guix build utils))" \n > "#~(begin" \n > "(use-modules" \n > "(guix build utils))" \n > - @ "))" ) ;; elisp (define-skeleton s/elisp-use-package "New package" "Package name: " > "(use-package " str \n > @ ")") (define-skeleton s/elisp-new-function "New function" "Function name: " > "(defun " str " () " \n - @ \n -1 ")") (define-skeleton s/lisp-lambda "New lambda" nil > "(lambda ()" \n "(interactive)" \n - @")") ;; Go (define-skeleton s/go-tmpl-translate "Translate" nil > "{{ T .p \"" - @ "\" }}") (define-skeleton s/go-tmpl-insert "Insert template" nil > "{{template \"" - @ "\" .}}") (define-skeleton s/golang-new-function "New function" "Function name: " > "func " str " (" - @ ") {" \n \n -1 "}") (define-skeleton s/golang-if-nil-err "If err != nil" nil > "if err != nil {" \n @ - \n -1 "}") (define-skeleton s/golang-new-struct "New struct" "Struct name: " > "type " str " struct {" \n \n -1 "}") (define-skeleton s/golang-for-loop "For loop" > "for _, v := range " - @ "{" \n \n -1 "}")) (use-package abbrev :after (skeleton go-mode scheme) :diminish abbrev-mode :init (setq-default abbrev-mode t) :config (define-abbrev scheme-mode-abbrev-table "2pr" "" 's/scheme-procedure) (define-abbrev scheme-mode-abbrev-table "2mo" "" 's/scheme-define-module) (define-abbrev scheme-mode-abbrev-table "2de" "" 's/scheme-define) (define-abbrev scheme-mode-abbrev-table "2ge" "" 's/guix-gexpr) (define-abbrev html-mode-abbrev-table "2tr" "" 's/go-tmpl-translate) (define-abbrev html-mode-abbrev-table "2te" "" 's/go-tmpl-insert) (define-abbrev lisp-mode-abbrev-table "2us" "" 's/elisp-use-package) (define-abbrev lisp-mode-abbrev-table "2fu" "" 's/elisp-new-function) (define-abbrev lisp-mode-abbrev-table "2la" "" 's/lisp-lambda) (define-abbrev go-mode-abbrev-table "2err" "" 's/golang-if-nil-err) (define-abbrev go-mode-abbrev-table "2fu" "" 's/golang-new-function) (define-abbrev go-mode-abbrev-table "2st" "" 's/golang-new-struct) (define-abbrev go-mode-abbrev-table "2fo" "" 's/golang-for-loop)) (use-package flyspell :diminish flyspell-mode :bind (:map flyspell-mode-map ("C-." . nil) ("M-" . nil)) :defer t) (use-package face-remap :diminish buffer-face-mode) (use-package org-indent :diminish org-indent-mode) (use-package simple :init (defun sudo-shell-command (command) (interactive "MShell command (root): ") (with-temp-buffer (cd "/sudo::/") (async-shell-command command))) :diminish visual-line-mode) (use-package message :custom (user-mail-address "marc@coquand.email") (message-default-headers "FCC: ~/personal-db/mail/home/out/sent.mbox") (message-signature "Marc") (message-expand-name-standard-ui t) (message-wide-reply-confirm-recipients t) (message-interactive t) (message-fcc-handler-function 'rmail-output) (message-citation-line-function 'message-insert-formatted-citation-line) (message-mail-alias-type 'ecomplete) (message-self-insert-commands nil) (mail-dont-reply-to-names nil) (send-mail-function 'smtpmail-send-it) (message-sendmail-f-is-evil t) (message-sendmail-extra-arguments '("--read-envelope-from")) (message-send-mail-function 'message-send-mail-with-sendmail) (message-sendmail-envelope-from 'header) (message-mail-user-agent t) (mail-user-agent 'message-user-agent) (message-kill-buffer-on-exit t)) (use-package sendmail :defer t :hook (message-setup . flyspell-mode) (message-setup . mail-abbrevs-setup) (message-setup . variable-pitch-mode) :custom (sendmail-program (executable-find "msmtp"))) (use-package rmail :init (defvar selected-rmail-account "home") (defun rmail-reply-t () "Reply only to the sender of the current message. (See rmail-reply.)" (interactive) (rmail-reply t)) (defmacro without-yes-or-no (&rest body) "Override `yes-or-no-p' & `y-or-n-p', not to prompt for input and return t." (declare (indent 1)) `(cl-letf (((symbol-function 'yes-or-no-p) (lambda (&rest _) t)) ((symbol-function 'y-or-n-p) (lambda (&rest _) t))) ,@body)) ;; Copied from the pipe of mu4e (defun rmail-process-file-through-pipe (path pipecmd) "Process file at PATH through a pipe with PIPECMD." (let ((buf (get-buffer-create "*rmail-output"))) (with-current-buffer buf (let ((inhibit-read-only t)) (erase-buffer) (call-process-shell-command pipecmd path t t) (view-mode))) (display-buffer buf))) (defun rmail-pipe-email (cmd) (interactive "sShell command: ") (let ((path (concat "/tmp/rmail-pipe-msg-" (number-to-string (time-convert (current-time) 'integer)) ".mbox"))) (without-yes-or-no (rmail-output-as-seen path nil t) (rmail-process-file-through-pipe path cmd)))) :hook ((rmail-mode . variable-pitch-mode) (rmail-quit . rmail-expunge-and-save)) :bind (("C-c m" . rmail-open) ("C-c M" . rmail-open-no-check) :map rmail-mode-map ("r" . rmail-reply-t) ("R" . rmail-reply) ("|" . rmail-pipe-email)) :config (defun rmail-no-check () (interactive) (rmail-input rmail-file-name)) (defun mail-work () (interactive) (setq rmail-primary-inbox-list (list (concat "imaps://" "marc.coquand%40piva.earth" ":" piva-password "@imap.gmail.com:993")) rmail-default-file "~/personal-db/mail/work/RMAIL" rmail-file-name "~/personal-db/mail/work/RMAIL" user-mail-address "marc.coquand@piva.earth" unrmail-mbox-format 'mboxo user-full-name "Marc Coquand" message-default-headers "FCC: ~/personal-db/mail/work/out/sent.mbox" message-signature "Marc Coquand\nFounder at Pivå")) (defun mail-home () (interactive) (setq rmail-default-file "~/personal-db/mail/home/RMAIL" rmail-primary-inbox-list (list (concat "imaps://" "marcc%40fastmail.fr" ":" fastmail-password "@imap.fastmail.com:993")) rmail-secondary-file-regexp "~/personal-db/mail/home/archives/*.mbox" rmail-output-file-alist '(("\\[PATCH.*\\]" . "patch.mbox") ("[rR]eceipt" . "receipt.mbox") ("[pP]ayment" . "receipt.mbox") ("[oO]rder" . "receipt.mbox") (".*" . "archive.mbox")) rmail-file-name "~/personal-db/mail/home/RMAIL" ;; In case there are errors, you can set '("--set=onerror=delete") rmail-movemail-flags nil user-mail-address "marc@coquand.email" unrmail-mbox-format 'mboxo user-full-name "Marc Coquand" message-default-headers "FCC: ~/personal-db/mail/home/out/sent.mbox" message-signature "Marc")) (defun rmail-open-no-check () "Set account and open rmail. Will not check for new messages." (interactive) (let ((prev selected-rmail-account)) (setq selected-rmail-account (completing-read "Account: " '(("home" 1) ("work" 2)) nil t)) (unless (eq selected-rmail-account prev) ;; First kill all RMAIL buffers to not leave any dangling (kill-matching-buffers "\\.mbox\\|^RMAIL" nil t)) (if (string= selected-rmail-account "home") (mail-home)) (if (string= selected-rmail-account "work") (mail-work))) (rmail-no-check)) (defun rmail-open () "Set account and open rmail. Will check for new messages" (interactive) (let ((prev selected-rmail-account)) (setq selected-rmail-account (completing-read "Account: " '(("home" 1) ("work" 2)) nil t)) (unless (eq selected-rmail-account prev) ;; First kill all RMAIL buffers to not leave any dangling (kill-matching-buffers "\\.mbox\\|^RMAIL" nil t)) (if (string= selected-rmail-account "home") (mail-home)) (if (string= selected-rmail-account "work") (mail-work))) (rmail)) :custom (rmail-remote-password-required nil) (rmail-preserve-inbox nil) (rmail-mime-prefer-html nil) ;; Makes forwarding works as expected (rmail-enable-mime-composing nil) (rmail-displayed-headers "^\\(?:Cc\\|Date\\|From\\|Subject\\|To\\|List-Id\\):")) (use-package mastodon :defer t :hook ((mastodon-mode . variable-pitch-mode)) :custom (mastodon-active-user "marcc") (mastodon-instance-url "https://fosstodon.org")) (use-package eww :custom (browse-url-browser-function 'eww-browse-url) ; Use eww as the default browser (eww-search-prefix "https://lite.duckduckgo.com/lite/?q=") (shr-max-image-proportion 0.3) (shr-width 70)) ;; Can play youtube urls and music. ;; Requires mpv (use-package emms :ensure t :custom (emms-shop-format "on: %s") :init (add-hook 'emms-player-started-hook 'emms-show) :config (emms-all) (emms-default-players)) (put 'secrets-mode 'disabled nil) (custom-set-variables '(custom-safe-themes '(default "~/.emacs.d/BOGO-theme.el" "~/.emacs.d/BOGO-dark-theme.el")))