diff options
Diffstat (limited to 'emacs/init.el')
-rw-r--r-- | emacs/init.el | 686 |
1 files changed, 686 insertions, 0 deletions
diff --git a/emacs/init.el b/emacs/init.el new file mode 100644 index 0000000..fa52557 --- /dev/null +++ b/emacs/init.el @@ -0,0 +1,686 @@ +(add-to-list 'load-path (expand-file-name (concat user-emacs-directory "site-lisp"))) + +(setq garbage-collection-messages t) + +(setopt user-mail-address "noa@gaiwan.org") + +;; properly distinguish these chords from their ascii legacy +(define-key input-decode-map [?\C-i] [C-i]) +(define-key input-decode-map [?\C-m] [C-m]) + +(use-package tubthumping-theme + :config + (load-theme 'tubthumping t)) + + +(global-set-key (kbd "<backspace>") 'backward-delete-char-untabify) + +;; I make my caps lock a menu key, so i can open the command palette with it +;; https://alexschroeder.ch/wiki/2020-07-16_Emacs_everything + +(define-key context-menu-mode-map (kbd "<menu>") nil) +(global-unset-key (kbd "M-x")) +(global-set-key (kbd "<menu>") 'execute-extended-command) + +(setopt + cursor-type 'bar + minibuffer-depth-indicate-mode t) + +(setopt + browse-url-browser-function 'eww-browse-url + url-cookie-trusted-urls '() + url-cookie-untrusted-urls '(".*")) + + +;;; candidate completion +(use-package vertico + :ensure t + :custom + (vertico-mode t) + (vertico-count 12) + (vertico-cycle t) + (read-buffer-completion-ignore-case t) + (read-file-name-completion-ignore-case t) + (completion-ignore-case t) +) + +(use-package orderless + :ensure t + :custom + (completion-styles '(orderless basic)) + (completion-category-defaults nil) + (completion-category-overrides '((file (styles partial-completion))))) + +(setopt confirm-nonexistent-file-or-buffer 'after-completion) + +(use-package nano-modeline + :ensure t + :hook ( + (prog-mode . nano-modeline-prog-mode) + (text-mode . nano-modeline-text-mode) + (org-mode . nano-modeline-org-mode) + (pdf-view-mode . nano-modeline-pdf-mode) + (mu4e-headers-mode . nano-modeline-mu4e-headers-mode) + (mu4e-view-mode . nano-modeline-mu4e-message-mode) + (term-mode . nano-modeline-term-mode) + (xwidget-webkit-mode . nano-modeline-xwidget-mode) + (messages-buffer-mode . nano-modeline-message-mode) + (org-capture-mode . nano-modeline-org-capture-mode) + (org-agenda-mode . nano-modeline-org-agenda-mode) + ) + :config + (setq-default mode-line-format nil) + ) + +(use-package beacon + :ensure t + :delight + :custom (beacon-mode t)) + +(use-package nov + :ensure t + :mode ("\\.epub\\'" . nov-mode)) + +(use-package shr + :custom + (shr-max-width nil)) + +;; also check out jinx https://github.com/minad/jinx +(use-package spell-fu + :ensure t + :hook (text-mode . spell-fu-mode)) + +(use-package minimap + :ensure t + :custom + (minimap-width-fraction 0.1) + (minimap-minimum-width 10) + (minimap-window-location 'right) + (minimap-update-delay 0.15)) + +;; consult-buffer replaces the buffer menu. as well as listing buffers, it lists bookmarks and recent files. +(use-package consult + :ensure t + :bind ( + ([remap switch-to-buffer] . consult-buffer) + ([remap yank-pop] . consult-yank-pop) + ([remap goto-line] . consult-goto-line)) + :custom + (completion-in-region-function 'consult-completion-in-region) + ) + +(use-package marginalia + :ensure t + :custom + (marginalia-mode t) + (marginalia-max-relative-age most-positive-fixnum) + (marginalia-align 'right)) + + +(global-set-key (kbd "<C-i>") 'completion-at-point) + +;; Add extensions +(use-package cape + :ensure t + :bind + ("C-c p f" . cape-file) + ("C-c p :" . cape-emoji) + :init + (add-hook 'completion-at-point-functions #'cape-file) + (add-hook 'completion-at-point-functions #'cape-dabbrev) + (add-hook 'completion-at-point-functions #'cape-elisp-block) + ;;(add-hook 'completion-at-point-functions #'cape-history) + ;;(add-hook 'completion-at-point-functions #'cape-keyword) + ;;(add-hook 'completion-at-point-functions #'cape-tex) + ;;(add-hook 'completion-at-point-functions #'cape-sgml) + ;;(add-hook 'completion-at-point-functions #'cape-rfc1345) + ;;(add-hook 'completion-at-point-functions #'cape-abbrev) + ;;(add-hook 'completion-at-point-functions #'cape-dict) + ;;(add-hook 'completion-at-point-functions #'cape-elisp-symbol) + ;;(add-hook 'completion-at-point-functions #'cape-line) +) + +(use-package org-roam + :ensure t + :custom + (org-roam-directory (expand-file-name "~/data/notes/")) + (org-roam-completion-everywhere t) + :bind ( + ("C-c n l" . org-roam-buffer-toggle) + ("C-c n f" . org-roam-node-find) + ("C-c n g" . org-roam-graph) + ("C-c n i" . org-roam-node-insert) + ("C-c n c" . org-roam-capture) + ;; Dailies + ("C-c n j" . org-roam-dailies-capture-today) + ) + :custom + ;; If you're using a vertical completion framework, you might want a more informative completion interface + (org-roam-node-display-template (concat "${title:*} " (propertize "${tags:10}" 'face 'org-tag))) + (org-roam-db-autosync-mode t) + :config + (require 'org-roam-protocol) +) + +(use-package org + :custom + (org-startup-indented t) + (org-pretty-entities t) + (org-use-sub-superscripts "{}") + (org-hide-emphasis-markers t) + (org-startup-with-inline-images t) + (org-image-actual-width '(300)) + (org-auto-align-tags nil) + (org-tags-column 0) + (org-catch-invisible-edits 'show-and-error) + (org-special-ctrl-a/e t) + (org-insert-heading-respect-content t) + (org-ellipsis "…") + + (org-agenda-tags-column t) + (org-agenda-block-separator ?─) + (org-agenda-time-grid + '((daily today require-timed) + (800 1000 1200 1400 1600 1800 2000) + " ┄┄┄┄┄ " "┄┄┄┄┄┄┄┄┄┄┄┄┄┄┄")) + (org-agenda-current-time-string + "◀── now ─────────────────────────────────────────────────") + + :custom-face + ;;(org-ellipsis ((t (:inherit default :box nil)))) +) + +(use-package org-appear + :ensure t + :after org + :hook + (org-mode . org-appear-mode)) + +(use-package org-modern + :ensure t + :after org + :hook + (org-mode . org-modern-mode) + (org-agenda-finalize . org-modern-agenda) + :custom + (org-modern-keyword nil) + (org-modern-checkbox nil) + (org-modern-table nil)) + +(use-package olivetti + :ensure t + :hook ( + (text-mode . olivetti-mode) + (eww-after-render . olivetti-mode) + (nov-post-html-render . olivetti-mode) + (mu4e-view-mode . olivetti-mode) + ) +) + +(use-package delight + :ensure t) + +(use-package eldoc + :delight + :custom + (global-eldoc-mode t)) + +(use-package rainbow-mode + :ensure t + :delight + :hook (prog-mode . rainbow-mode)) + +(use-package hyperspace + :disabled + :ensure t + :bind ("<C-m>" . hyperspace)) + +(use-package hyperbole + :disabled + :ensure t) + +;; use a bar cursor and blink it and don't stop blinking it. i don't know how i feel about this yet to be honest, but it helps me know which window is active so for now i'm keeping it +(setopt + cursor-type 'bar + blink-cursor-mode -1 + blink-cursor-interval 0.7) + +(setopt + dired-recursive-deletes 'always + dired-recursive-copies 'always + dired-clean-up-buffers-too nil + delete-by-moving-to-trash t + dired-dwim-target t + dired-listing-switches "-alv") + +;;; indentation: tabs and whitespace settings + +;; my rules for inserting tabs are that the tab key should insert tabs. i personally prefer tabs. spaces don't work for me because then people go trigger happy with alignment, which doesn't really work with proportional fonts. + +;; todo: elastic tabstops +;; https://nickgravgaard.com/elastic-tabstops/ +;; https://github.com/tenbillionwords/spaceship-mode + +;; always insert \t for tab. +(setopt + electric-indent-mode nil + backward-delete-char-untabify-method nil + tab-width 8) + +;; do a naive duplicate whitespace on return, vi/nano style. +(define-key global-map (kbd "TAB") 'self-insert-command) +(defun naive-return-and-indent () + "insert a newline and copy the indentation of the previous line" + (interactive) + (open-line 1) + (let* ((start (progn (beginning-of-line) (point))) + (indent (progn (back-to-indentation) (point))) + (end (progn (end-of-line) (point))) + (whitespace (buffer-substring start indent))) + (delete-trailing-whitespace start end) + (beginning-of-line 2) + (insert whitespace))) +(define-key global-map (kbd "RET") 'naive-return-and-indent) + +;;; interface + +;; we want to make sure that various bits of the interface are hidden. but this isn't an "all gui chrome is useless" rampage. i personally think the scrollbar is useful, i like the visual indication it gives of how far i am through a file. +(setopt + menu-bar-mode nil + tool-bar-mode nil + tooltip-mode nil ;; show tooltips in the echo area + echo-keystrokes 0.1) ;; show chord progress (almost) immediately + +(use-package paren + :custom + (show-paren-mode t) + + ) + +;; populate and enable the context menu +;; (setopt context-menu-functions '( +;; context-menu-ffap +;; occur-context-menu +;; context-menu-region +;; context-menu-undo +;; goto-address-context-menu) +;; context-menu-mode t) +(use-package mouse + :custom + (context-menu-mode t)) + + + +(use-package tab-bar + :disabled + :custom + (tab-bar-mode t) + (tab-bar-format '(tab-bar-format-align-right tab-bar-format-global tab-bar-format-menu-bar)) + + ) + + +(use-package font-lock + :custom + (global-font-lock-mode t) + (font-lock-maximum-decoration nil)) + +(setopt + display-time-mode t + display-battery-mode t + inhibit-startup-screen t + mouse-drag-and-drop-region nil + mouse-yank-at-point t + delete-selection-mode nil ;; deleting should be an explicit action + ) + +(global-set-key (kbd "C-t") 'tab-new) + +;; shift click to select region with the mouse. This annoyingly rings the bell for an error +(global-unset-key (kbd "S-<down-mouse-1>")) +(global-set-key (kbd "S-<down-mouse-1>") 'mouse-save-then-kill) + +;;; packages +(setopt package-archives '( + ("gnu" . "https://elpa.gnu.org/packages/") + ("nongnu" . "https://elpa.nongnu.org/nongnu/") + ("melpa-stable" . "https://stable.melpa.org/packages/") + ("melpa" . "https://melpa.org/packages/"))) + +;;; saving + +;; backups are pointless in long emacs sessions imo, but autosaves are useful +(setopt make-backup-files nil + backup-by-copying t + create-lockfiles nil + auto-save-mode 1 + auto-save-interval 20 ;; every twenty keystrokes + auto-save-timeout 5 ;; every 5 seconds + auto-save-default t + auto-save-no-message t + version-control t + auto-save-visited-mode t) +(add-hook 'focus-out-hook (lambda () (interactive) (save-some-buffers t))) +(add-hook 'mouse-leave-buffer-hook (lambda () (interactive) (save-some-buffers t))) + +;;; scrolling +(setopt + mouse-wheel-scroll-amount '(1 ((shift) . 1)) + mouse-wheel-progressive-speed nil + mouse-wheel-follow-mouse 't + scroll-step 1 + pixel-scroll-precision-mode t + pixel-scroll-precision-use-momentum t) + + +;;; sentences +(setopt sentence-end-double-space nil) + +;;; spellcheck +(setopt + flyspell-mode t + ispell-program-name "aspell" + ispell-dictionary "en_GB" + ispell-extra-args '("--sug-mode=ultra") + require-final-newline t + window-min-height 1 + window-combination-resize t + window-resize-pixelwise t + frame-resize-pixelwise t) + +;;; history +(setopt + history-length 250 + kill-ring-max 25) + +(use-package savehist + :custom + (savehist-file "~/.config/emacs/savehist") + (savehist-additional-variables '( + kill-ring + command-history + set-variable-value-history + custom-variable-history + query-replace-history + read-expression-history + minibuffer-history + read-char-history + face-name-history + bookmark-history + file-name-history)) + (savehist-mode t)) + +(use-package frame + :custom + (window-divider-mode t) + (window-divider-default-right-width 1) + (window-divider-default-bottom-width 1) + (window-divider-default-places t) + ) + +(use-package emacs + :init + ;; Add prompt indicator to `completing-read-multiple'. + ;; We display [CRM<separator>], e.g., [CRM,] if the separator is a comma. + (defun crm-indicator (args) + (cons (format "[CRM%s] %s" + (replace-regexp-in-string + "\\`\\[.*?]\\*\\|\\[.*?]\\*\\'" "" + crm-separator) + (car args)) + (cdr args))) + (advice-add #'completing-read-multiple :filter-args #'crm-indicator) + + ;; Do not allow the cursor in the minibuffer prompt + (setq minibuffer-prompt-properties + '(read-only t cursor-intangible t face minibuffer-prompt)) + (add-hook 'minibuffer-setup-hook #'cursor-intangible-mode) + + :custom + ;; Support opening new minibuffers from inside existing minibuffers. + (enable-recursive-minibuffers t) + (debug-on-error t) + + ;; Hide commands in M-x which do not work in the current mode. + (read-extended-command-predicate 'command-completion-default-include-p)) + +(setopt + recentf-max-menu-items 25 + recentf-save-file "~/.config/emacs/recentf" + recentf-mode 1 + bookmark-default-file "~/.config/emacs/bookmarks") + + + +;;; miscellaneous +(setopt + save-place-mode 1) + +(use-package goto-addr + :custom + (global-goto-address-mode t) + (goto-address-mail-face 'link) + (goto-address-mail-mouse-face 'link) + (goto-address-url-face 'link) + (goto-address-url-mouse-face 'link) +) + +(setenv "PAGER" "cat") +(setenv "TERM" "dumb") +(setenv "GPG_AGENT_INFO" nil) +(defalias 'yes-or-no-p 'y-or-n-p) +(setq disabled-command-function nil) +(setopt custom-file (make-temp-file "custom")) +(setq inhibit-startup-echo-area-message "noa") ;; #userfreedom + +(use-package simple + :delight visual-line-mode + :custom + (global-visual-line-mode t)) + +(setopt + kill-whole-line t + uniquify-after-kill-buffer-p t + uniquify-buffer-name-style 'forward + uniquify-ignore-buffers-re "^\\*" + uniquify-separator "/") + +(setopt + save-interprogram-paste-before-kill t + apropos-do-all t + mouse-yank-at-point t + require-final-newline t + visible-bell t + load-prefer-newer t + frame-inhibit-implied-resize t + ediff-window-setup-function 'ediff-setup-windows-plain) + +(load "server") +(unless (server-running-p) (server-start)) + +(setopt help-at-pt-display-when-idle t) + +;;; search +;; search without regard for unicode characters +(setopt search-default-mode 'char-fold-to-regexp + replace-char-fold t + search-highlight t + isearch-lazy-highlight t) +;; when exiting isearch after a match, put the cursor at the beginning of the search rather than the end +(setq isearch-mode-end-hook '(goto-beginning-of-search)) +(defun goto-beginning-of-search () + "positions the point at the beginning of the isearch match" + (when (and (not isearch-mode-end-hook-quit) + isearch-forward) + (goto-char isearch-other-end))) +;; wrap searches +(setopt isearch-wrap-pause 'no-ding) +;; jump straight to previous match when change search direction +(setopt isearch-repeat-on-direction-change t) + +(use-package ctrlf + :ensure t + :bind ( + ([remap isearch-forward] . ctrlf-forward-default) + ([remap isearch-backward] . ctrlf-backward-default) + ([remap isearch-forward-regexp] . ctrlf-forward-alternate) + ([remap isearch-backward-regexp] . ctrlf-backward-alternate) + ([remap isearch-forward-symbol] . ctrlf-forward-symbol) + ([remap isearch-forward-symbol-at-point] . ctrlf-forward-symbol-at-point) + ) + :custom + (ctrlf-go-to-end-of-match nil "It makes more sense to go to the start of the match, because i start searching where i want to be.") +) + +(global-set-key (kbd "M-o") 'other-window) + +(global-set-key (kbd "C-x k") 'kill-this-buffer) + +;; undo C-/ +;; redo C-S-/ +(setopt undo-no-redo t) + +;; mu4e +(use-package mu4e + :custom + (mu4e-headers-skip-duplicates t) + (mu4e-view-show-images t) + (mu4e-view-show-addresses t) + (mu4e-compose-format-flowed nil) + (mu4e-change-filenames-when-moving t) + (mu4e-use-fancy-chars nil) + (mu4e-confirm-quit nil) + (mu4e-headers-leave-behavior 'apply) + (mu4e-headers-precise-alignment t) + (mu4e-search-threads nil) + (mu4e-hide-index-messages t) + (mu4e-get-mail-command "mbsync -c ~/.config/mbsyncrc fastmail") + (mu4e-maildir "~/data/mail") + (mu4e-drafts-folder "/Drafts") + (mu4e-sent-folder "/Sent") + (mu4e-refile-folder "/Archive") + (mu4e-trash-folder "/Trash") + (mu4e-bookmarks '( + (:name "Waiting room" :query "\"maildir:/Waiting room\"" :key ?w) + (:name "Inbox" :query "maildir:/Inbox" :key ?i) + (:name "Feeds" :query "maildir:/Feeds" :key ?f) + (:name "Paper trail" :query "\"maildir:/Paper trail\"" :key ?p))) + + :bind ( + :map mu4e-headers-mode-map + ("d" . my-move-to-trash) + :map mu4e-view-mode-map + ("d" . my-move-to-trash) + ) + :init + (fset 'my-move-to-trash "mTrash") ;; function to move mails to trash + :custom-face + ) + +(use-package button + :custom-face + ;;(button ((t (:underline t :foreground unspecified)))) + ) + +(use-package faces + :custom-face + ;; (link ((t (:underline t :foreground unspecified)))) + ;; (fringe ((t (:background unspecified)))) + ;; (mode-line ((t (:background unspecified :box 2)))) + ;; (mode-line-active ((t (:inverse-video t)))) + ;; (mode-line-inactive ((t (:background unspecified :box 2 :weight unspecified)))) + ;; (help-key-binding ((t (:inherit default :background unspecified :foreground unspecified :box 1)))) + ) + +(global-set-key (kbd "M-z") 'zap-up-to-char) + +(use-package markdown-mode + :ensure t + :mode ("\\.md\\'" . markdown-mode) + ) +(use-package valign + :ensure t + :after markdown-mode + :hook (markdown-mode . valign-mode)) + + +;;; TOUCHSCREEN +;; this should be obsolete in emacs 30 +;; Copyright 2024-present Naheel Azawy. All rights reserved. +(defvar touchscreen-last-time) +(defvar touchscreen-last-pos-pixel) +(defvar touchscreen-last-dist 0) +(defvar touchscreen-begin-char) + +(defun touchscreen-time () + "Time in seconds." + (time-convert (current-time) 'integer)) + +(defun touchscreen-handle-touch-begin (input) + "Handle touch begining at input INPUT." + (interactive "e") + (let* ((event (nth 1 input)) + (pos-pixel (nth 3 event)) + (pos-char (nth 6 event)) + (win (nth 1 event))) + ;; (message (format "%s" input)) + (if (not (equal (selected-window) win)) + ;; switch window + (select-window win)) + ;; set globals + (setq touchscreen-last-time (touchscreen-time)) + (setq touchscreen-last-pos-pixel pos-pixel) + (setq touchscreen-begin-char pos-char) + )) + +(defun touchscreen-handle-touch-update (input) + "Handle touch update at input INPUT." + (interactive "e") + (let* ((event (nth 0 (nth 1 input))) + (pos-pixel (nth 3 event)) + (pos-char (nth 6 event)) + (diff-time (- (touchscreen-time) touchscreen-last-time)) + (diff-pixel (- (cdr touchscreen-last-pos-pixel) (cdr pos-pixel))) + (diff-char (abs (- touchscreen-begin-char pos-char)))) + + (if (= (length (nth 1 input)) 2) + ;; pinch zoom + (let* ((event2 (nth 1 (nth 1 input))) + (pos-pixel2 (nth 3 event2)) + (dist (sqrt (+ (expt (- (car pos-pixel2) (car pos-pixel)) 2) + (expt (- (cdr pos-pixel2) (cdr pos-pixel)) 2)))) + (dist-diff (- dist touchscreen-last-dist))) + (setq touchscreen-last-dist dist) + (if (> dist-diff 0) + (text-scale-increase 0.1) + (if (< dist-diff 0) + (text-scale-decrease 0.1))) + ) + + (if (> diff-time 1) + ;; TODO: set marker on long press + (goto-char pos-char)) + (if (> diff-char 1) + ;; scroll + (progn + (move-to-window-line nil) + (if (> diff-pixel 0) + (pixel-scroll-pixel-up diff-pixel) + (if (< diff-pixel 0) + (pixel-scroll-pixel-down (* -1 diff-pixel)))) + (setq touchscreen-last-time (touchscreen-time)) + (setq touchscreen-last-pos-pixel pos-pixel)) + )))) + +(defun touchscreen-handle-touch-end (input) + "Handle touch end at input INPUT." + (interactive "e") + (let* ((event (nth 1 input)) + (pos-char (nth 6 event))) + (if (= touchscreen-begin-char pos-char) + ;; move cursor + (goto-char pos-char)))) + +(global-set-key [touchscreen-begin] #'touchscreen-handle-touch-begin) +(global-set-key [touchscreen-update] #'touchscreen-handle-touch-update) +(global-set-key [touchscreen-end] #'touchscreen-handle-touch-end) |