#+title: Emacs Configuration #+author: jdlugosz963 #+PROPERTY: header-args:emacs-lisp :tangle .config/emacs/init.el * Table of contents :toc: - [[#startup-performance][Startup Performance]] - [[#user-information-setup][User information setup]] - [[#stay-clean][Stay clean]] - [[#package-setup][Package setup]] - [[#update-path][Update Path]] - [[#keboard-setup][Keboard Setup]] - [[#general-setup][General Setup]] - [[#evil-mode-setup][Evil mode setup]] - [[#ui-configuration][UI Configuration]] - [[#basic-ui-setup][Basic UI setup]] - [[#ivy-and-counsel][Ivy and Counsel]] - [[#which-key][Which key]] - [[#modline][Modline]] - [[#hydra][Hydra]] - [[#beacon][Beacon]] - [[#other-keybindings][Other Keybindings]] - [[#org][Org]] - [[#org-mode-setup][Org-mode setup]] - [[#org-evil][Org evil]] - [[#org-make-toc][Org make toc]] - [[#org-fonts][Org fonts]] - [[#org-tempo-setup][Org tempo setup]] - [[#org-babel-setup][Org babel setup]] - [[#org-margin][Org margin]] - [[#org-superstar][Org superstar]] - [[#org-roam][Org roam]] - [[#development][Development]] - [[#languages][Languages]] - [[#dap-mode][Dap mode]] - [[#company][Company]] - [[#projectile][Projectile]] - [[#neotree][Neotree]] - [[#magit][Magit]] - [[#apps][Apps]] - [[#vterm][Vterm]] - [[#dired][Dired]] - [[#emms][EMMS]] - [[#pdf-tools][PDF Tools]] - [[#mu4e][Mu4e]] - [[#runtime-performance][Runtime Performance]] * Startup Performance #+begin_src emacs-lisp (setq gc-cons-threshold (* 50 1000 1000)) (defun jd/display-startup-time () (message "Emacs loaded in %s with %d garbage collections." (format "%.2f seconds" (float-time (time-subtract after-init-time before-init-time))) gcs-done)) (add-hook 'emacs-startup-hook #'jd/display-startup-time) #+end_src * User information setup #+begin_src emacs-lisp (setq user-full-name "Jakub Dlugosz" user-mail-address "jdlugosz963@gmail.com") #+end_src * Stay clean #+begin_src emacs-lisp (setq backup-directory-alist '(("." . "~/.cache/emacs/backup")) backup-by-copying t version-control t delete-old-versions t kept-new-versions 20 kept-old-versions 5) (setq user-emacs-directory "~/.cache/emacs") #+end_src * Package setup #+begin_src emacs-lisp (require 'package) (setq package-archives '(("melpa" . "https://melpa.org/packages/") ("org" . "https://orgmode.org/elpa/") ("gnu-devel" . "https://elpa.gnu.org/devel/") ("elpa" . "https://elpa.gnu.org/packages/"))) (package-initialize) (unless package-archive-contents (package-refresh-contents)) (unless (package-installed-p 'use-package) (package-install 'use-package)) (setq use-package-always-ensure t) (require 'use-package) #+end_src * Update Path #+begin_src emacs-lisp (push ".config/emacs/jd/" load-path) #+end_src * Keboard Setup ** General Setup #+begin_src emacs-lisp (use-package general :config (general-create-definer jd/leader-key-def :keymaps '(normal insert visual emacs) :prefix "SPC" :global-prefix "C-SPC") (general-create-definer jd/ctrl-c-keys :prefix "C-c")) #+end_src ** Evil mode setup #+begin_src emacs-lisp (use-package undo-tree :diminish :config (setq undo-tree-auto-save-history nil) (global-undo-tree-mode 1)) (use-package evil :init (setq evil-want-integration t) (setq evil-want-keybinding nil) (setq evil-want-C-u-scroll t) (setq evil-want-Y-yank-to-eol t) (setq evil-want-C-i-jump nil) (setq evil-undo-system 'undo-tree) (setq evil-kill-on-visual-paste nil) :config (evil-set-initial-state 'exwm-mode 'emacs) (evil-mode 1)) (use-package evil-collection :after evil :config (evil-collection-init)) #+end_src *** Keybinds #+begin_src emacs-lisp (defhydra hydra-resize-window (:timeout 4) ("h" evil-window-decrease-width "<") ("l" evil-window-increase-width ">") ("k" evil-window-decrease-height "^") ("j" evil-window-increase-height "v") ("q" nil "finished" :exit t)) (jd/leader-key-def "w" 'evil-window-map "wr" '(hydra-resize-window/body :which-key "Rezize window") "wd" '(evil-window-delete :which-key "Window delete")) (jd/leader-key-def "ou" '(undo-tree-visualize :which-key "Open")) ;; remove . key (define-key evil-normal-state-map (kbd ".") '()) #+end_src * UI Configuration ** Basic UI setup #+begin_src emacs-lisp (setq inhibit-startup-message t) (setq visible-bell t) (scroll-bar-mode -1) (tool-bar-mode -1) (tooltip-mode -1) (menu-bar-mode -1) (set-fringe-mode 10) (set-face-attribute 'default nil :font "Hack" :height 100) (add-hook 'prog-mode-hook 'menu-bar--display-line-numbers-mode-relative) (global-set-key (kbd "") 'keyboard-escape-quit) (defalias 'yes-or-no-p 'y-or-n-p) (use-package doom-themes :config (load-theme 'doom-badger t)) (use-package diminish) #+end_src ** #+begin_src emacs-lisp (use-package hl-todo :init (setq hl-todo-keyword-faces '(("TODO" . "#FF0000") ("FIXME" . "#FF0000") ("DEBUG" . "#A020F0"))) :config (global-hl-todo-mode 1)) #+end_src ** Ivy and Counsel #+begin_src emacs-lisp (defun jd/switch-buffer () (interactive) (let ((completion-regexp-list '("\\`[^*]" "\\`\\([^T]\\|T\\($\\|[^A]\\|A\\($\\|[^G]\\|G\\($\\|[^S]\\|S.\\)\\)\\)\\).*"))) (call-interactively 'counsel-switch-buffer))) (use-package ivy :diminish :bind (("C-s" . swiper) :map ivy-minibuffer-map ("C-k" . ivy-previous-line) ("C-j" . ivy-next-line) :map ivy-switch-buffer-map ("C-k" . ivy-previous-line)) :config (ivy-mode 1)) (use-package counsel :config (counsel-mode 1)) #+end_src *** Keybinds #+begin_src emacs-lisp (jd/leader-key-def "t" '(:ignore t :which-key "Toogle") "tt" '(counsel-load-theme :which-key "Choose theme")) (jd/leader-key-def "bb" '(jd/switch-buffer :which-key "Buffer switch") "ba" '(counsel-switch-buffer :which-key "Buffer switch") "b" '(:ignore t :which-key "Buffer") "," '(counsel-switch-buffer :which-key "Buffer switch")) #+end_src ** Which key #+begin_src emacs-lisp (use-package which-key :diminish :config (which-key-mode) (setq which-key-idle-delay 0.3)) #+end_src ** Modline #+begin_src emacs-lisp (use-package all-the-icons) (use-package doom-modeline :init (doom-modeline-mode 1) :custom ((doom-modeline-height 15))) #+end_src ** Hydra #+begin_src emacs-lisp (use-package hydra :defer t) (defun jd/text-scale-increase () (interactive) (let ((old-face-attribute (face-attribute 'default :height))) (set-face-attribute 'default nil :height (+ old-face-attribute 10)))) (defun jd/text-scale-decrease () (interactive) (let ((old-face-attribute (face-attribute 'default :height))) (set-face-attribute 'default nil :height (- old-face-attribute 10)))) (defhydra hydra-text-scale-global (:timeout 4) "scale text" ("j" jd/text-scale-increase "in") ("k" jd/text-scale-decrease "out") ("q" nil "finished" :exit t)) (defhydra hydra-text-scale (:timeout 4) "scale text" ("j" text-scale-increase "in") ("k" text-scale-decrease "out") ("q" nil "finished" :exit t)) (jd/leader-key-def "tS" '(hydra-text-scale-global/body :which-key "Scale text") "ts" '(hydra-text-scale/body :which-key "Scale text")) #+end_src ** Beacon #+begin_src emacs-lisp (use-package beacon :config (beacon-mode 1)) #+end_src ** Other Keybindings *** Files #+begin_src emacs-lisp (jd/leader-key-def "f" '(:ignore t :which-key "Files") "fs" '(save-buffer :which-key "File save") "." '(find-file :which-key "Find file") "ff" '(find-file :which-key "Find file")) #+end_src *** Eval #+begin_src emacs-lisp (jd/leader-key-def "e" '(:ignore t :which-key "Eval") "eb" '(eval-buffer :which-key "Eval buffer")) (jd/leader-key-def :keymaps '(visual) "er" '(eval-region :which-key "Eval region")) #+end_src *** Buffers #+begin_src emacs-lisp (defun jd/kill-other-buffers () (interactive) (mapc 'kill-buffer (cdr (buffer-list (current-buffer))))) (jd/leader-key-def "bK" '(jd/kill-other-buffers :which-key "Kill other buffers") "bk" '(kill-current-buffer :which-key "Kill buffer") "bB" '(ibuffer :which-key "Open ibuffer")) #+end_src * Org ** Org-mode setup #+begin_src emacs-lisp (defun jd/org-mode-setup () (org-indent-mode) (variable-pitch-mode 1) (visual-line-mode 1)) (use-package org :pin org :commands (org-capture org-agenda) :hook (org-mode . jd/org-mode-setup) :config (setq org-directory (file-truename "~/Documents/org/")) (setq org-mobile-inbox-for-pull (concat org-directory "flagged.org")) (setq org-mobile-directory "~/Dropbox/Apps/MobileOrg") (setq org-agenda-files '("Tasks.org")) (setq org-ellipsis " ▾") (setq org-agenda-start-with-log-mode t) (setq org-log-done 'time) (setq org-log-into-drawer t) (setq org-return-follows-link t) (setq org-capture-templates `(("t" "Tasks / Projects") ("tt" "Task" entry (file+olp "Tasks.org" "Inbox") "* TODO %?\n %t\n %a\n %i" :empty-lines 1) ("tT" "Task for tomorow" entry (file+olp "Tasks.org" "Inbox") "* TODO %?\n %t\n SCHEDULED: %(org-insert-time-stamp (org-read-date nil t \"+1d\"))\n %a\n %i" :empty-lines 1) ("m" "Metrics Capture") ("mm" "Metrics" table-line (file+headline "Metrics.org" "Metrics") "| %U | %^{Weight} | %^{Waist} | %^{Notes} |" :kill-buffer t))) (setq org-latex-listings 'minted org-latex-packages-alist '(("" "minted")) org-latex-pdf-process '("pdflatex -shell-escape -interaction nonstopmode -output-directory %o %f" "pdflatex -shell-escape -interaction nonstopmode -output-directory %o %f")) (require 'org-tempo)) #+end_src *** Keybinds #+begin_src emacs-lisp (jd/leader-key-def "o" '(:ignore t :which-key "Open/Org") "oc" '(org-capture :which-key "Open org-capture") "oop" '(org-mobile-pull :which-key "Org mobile pull") "ooP" '(org-mobile-push :which-key "Org mobile push") "oa" '(org-agenda :which-key "Open org-agenda")) #+end_src ** Org evil #+begin_src emacs-lisp (use-package evil-org :after org :hook (org-mode . (lambda () evil-org-mode)) :config (require 'evil-org-agenda) (evil-org-agenda-set-keys)) #+end_src ** Org make toc #+begin_src emacs-lisp (use-package toc-org :config (add-hook 'org-mode-hook 'toc-org-mode)) #+end_src ** Org fonts #+begin_src emacs-lisp (defun jd/org-font-setup () ;; Replace list hyphen with dot (font-lock-add-keywords 'org-mode '(("^ *\\([-]\\) " (0 (prog1 () (compose-region (match-beginning 1) (match-end 1) "•-")))))) ;; Set faces for heading levels (dolist (face '((org-level-1 . 1.3) (org-level-2 . 1.2) (org-level-3 . 1.1) (org-level-4 . 1.0) (org-level-5 . 1.1) (org-level-6 . 1.1) (org-level-7 . 1.1) (org-level-8 . 1.1))) (set-face-attribute (car face) nil :font "Monospace" :weight 'Bold :height (cdr face))) ;; Ensure that anything that should be fixed-pitch in Org files appears that way (set-face-attribute 'org-block nil :font "mononoki Nerd Font" :inherit 'fixed-pitch) (set-face-attribute 'org-table nil :inherit 'fixed-pitch) (set-face-attribute 'org-formula nil :inherit 'fixed-pitch) (set-face-attribute 'org-code nil :inherit '(shadow fixed-pitch)) (set-face-attribute 'org-table nil :inherit '(shadow fixed-pitch)) (set-face-attribute 'org-verbatim nil :inherit '(shadow fixed-pitch)) (set-face-attribute 'org-special-keyword nil :inherit '(font-lock-comment-face fixed-pitch)) (set-face-attribute 'org-meta-line nil :inherit '(font-lock-comment-face fixed-pitch)) (set-face-attribute 'org-checkbox nil :inherit 'fixed-pitch) (set-face-attribute 'line-number nil :inherit 'fixed-pitch) (set-face-attribute 'line-number-current-line nil :inherit 'fixed-pitch)) (with-eval-after-load 'org-faces (jd/org-font-setup)) #+end_src ** Org tempo setup #+begin_src emacs-lisp (defun jd/org-tempo-setup () (add-to-list 'org-structure-template-alist '("s" . "src")) (add-to-list 'org-structure-template-alist '("sql" . "src sql")) (add-to-list 'org-structure-template-alist '("sh" . "src sh")) (add-to-list 'org-structure-template-alist '("el" . "src emacs-lisp")) (add-to-list 'org-structure-template-alist '("li" . "src lisp")) (add-to-list 'org-structure-template-alist '("sc" . "src scheme")) (add-to-list 'org-structure-template-alist '("ts" . "src typescript")) (add-to-list 'org-structure-template-alist '("py" . "src python")) (add-to-list 'org-structure-template-alist '("go" . "src go")) (add-to-list 'org-structure-template-alist '("yaml" . "src yaml"))) (with-eval-after-load 'org-tempo (jd/org-tempo-setup)) #+end_src ** Org babel setup *** Load languages #+begin_src emacs-lisp (org-babel-do-load-languages 'org-babel-load-languages '((emacs-lisp . t) (python . t))) #+end_src *** Auto tangle #+begin_src emacs-lisp (defun jd/org-babel-tangle-config () (when (string-equal (file-name-directory (buffer-file-name)) (expand-file-name "~/dotfiles/")) ;; Dynamic scoping to the rescue (let ((org-confirm-babel-evaluate nil)) (org-babel-tangle)))) (add-hook 'org-mode-hook (lambda () (add-hook 'after-save-hook #'jd/org-babel-tangle-config))) #+end_src ** Org margin #+begin_src emacs-lisp (defun jd/org-mode-visual-fill () (setq visual-fill-column-width 100 visual-fill-column-center-text t) (visual-fill-column-mode 1)) (use-package visual-fill-column :hook (org-mode . jd/org-mode-visual-fill)) #+end_src ** Org superstar #+begin_src emacs-lisp (use-package org-superstar :hook (org-mode . org-superstar-mode) :init (setq org-superstar-special-todo-items t) (setq org-superstar-remove-leading-stars t) (setq org-superstar-headline-bullets-list '("◉" "○" "●" "○" "●" "○" "●"))) #+end_src ** Org roam #+begin_src emacs-lisp (defun jd/org-roam-filter-by-tag (tag-name) (lambda (node) (member tag-name (org-roam-node-tags node)))) (defun jd/org-roam-list-notes-by-tag (tag-name) (mapcar #'org-roam-node-file (seq-filter (jd/org-roam-filter-by-tag tag-name) (org-roam-node-list)))) (defun jd/org-roam-refreshagenda-list () (interactive) (setq org-agenda-files (org-roam-list-files))) (use-package org-roam :custom (org-roam-directory (file-truename "~/Documents/org/roam/")) :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)) :bind-keymap ("C-c n d" . org-roam-dailies-map) :config ;; If you're using a vertical completion framework, you might want a more informative completion interface (setq org-roam-node-display-template (concat "${title:*} " (propertize "${tags:10}" 'face 'org-tag))) (org-roam-db-autosync-mode)) #+end_src * Development ** Languages *** Lsp #+begin_src emacs-lisp (use-package lsp-mode :commands (lsp lsp-deferred) :init (setq lsp-headerline-breadcrumb-enable nil) (setq lsp-diagnostics-provider :none) (setq lsp-modeline-diagnostics-enable nil) (setq lsp-keymap-prefix "C-c l") :config (lsp-enable-which-key-integration t)) (use-package lsp-ivy :after lsp) #+end_src *** Python #+begin_src emacs-lisp (defun jd/python-mode-setup () (let ((project-venv-path (concat (projectile-project-root) "venv/"))) (when (projectile--directory-p project-venv-path) (pyvenv-activate project-venv-path) (pyvenv-mode)))) (use-package python-mode :hook (python-mode . lsp-deferred) :hook (python-mode . jd/python-mode-setup) :config (add-to-list 'auto-mode-alist '("\\.py\\'" . python-mode))) (use-package pyvenv :after python-mode) (use-package django-mode :config ;; Remove all django-mode objects from auto-mode-alist (while (rassq 'django-mode auto-mode-alist) (let ((django-mode-object (rassq 'django-mode auto-mode-alist))) (setq auto-mode-alist (delete django-mode-object auto-mode-alist))))) #+end_src *** Type Script #+begin_src emacs-lisp (use-package typescript-mode :config (setq typescript-indent-level 2) (add-to-list 'auto-mode-alist '("\\.tsx\\'" . typescript-mode))) (use-package tide :ensure t :after (typescript-mode company) :hook ((typescript-mode . tide-setup) (typescript-mode . tide-hl-identifier-mode))) (use-package flycheck :ensure t :hook ((after-init . global-flycheck-mode))) #+end_src *** Yaml #+begin_src emacs-lisp (use-package yaml-mode) #+end_src *** Docker #+begin_src emacs-lisp (use-package docker) #+end_src **** Keybinds #+begin_src emacs-lisp (jd/leader-key-def "d" '(:ignore t :which-key "Docker") "dc" '(docker-containers :which-key "Docker containers") "dd" '(docker :which-key "Docker")) #+end_src ** Dap mode #+begin_src emacs-lisp (use-package dap-mode ;; Uncomment the config below if you want all UI panes to be hidden by default! ;; :custom ;; (lsp-enable-dap-auto-configure nil) ;; :config ;; (dap-ui-mode 1) :commands dap-debug :config ;; Set up Node debugging (require 'dap-node) (dap-node-setup) ;; Automatically installs Node debug adapter if needed ;; Bind `C-c l d` to `dap-hydra` for easy access (general-define-key :keymaps 'lsp-mode-map :prefix lsp-keymap-prefix "d" '(dap-hydra t :wk "debugger"))) #+end_src ** Company #+begin_src emacs-lisp (use-package company :after lsp-mode :hook (lsp-mode . company-mode) :bind (:map company-active-map ("" . company-complete-selection)) (:map lsp-mode-map ("" . company-indent-or-complete-common)) :custom (company-minimum-prefix-length 1) (company-idle-delay 0.0) :config (global-company-mode)) (use-package company-box :hook (company-mode . company-box-mode)) #+end_src ** Projectile #+begin_src emacs-lisp (use-package projectile :diminish projectile-mode :init (when (file-directory-p "~/Documents/code") (setq projectile-project-search-path '("~/Documents/code/"))) :custom ((projectile-Completion-system 'ivy)) :config (setq projectile-switch-project-action #'projectile-dired) (projectile-mode)) #+end_src *** Keybinds #+begin_src emacs-lisp (jd/leader-key-def "p" '(projectile-command-map :which-key "Project") "p " '() "SPC" '(projectile-find-file :which-key "Find file in project")) #+end_src ** Neotree #+begin_src emacs-lisp (defun jd/neotree-project-dir () "Open NeoTree using the git root." (interactive) (let ((project-dir (projectile-project-root)) (file-name (buffer-file-name))) (neotree-toggle) (if project-dir (if (neo-global--window-exists-p) (progn (neotree-dir project-dir) (neotree-find file-name))) (message "Could not find git project root.")))) (use-package neotree :init (setq neo-theme 'icons) :config (evil-define-key 'normal neotree-mode-map (kbd "TAB") 'neotree-enter) (evil-define-key 'normal neotree-mode-map (kbd "C-RET") 'neotree-quick-look) (evil-define-key 'normal neotree-mode-map (kbd "q") 'neotree-hide) (evil-define-key 'normal neotree-mode-map (kbd "RET") 'neotree-enter) (evil-define-key 'normal neotree-mode-map (kbd "g") 'neotree-refresh) (evil-define-key 'normal neotree-mode-map (kbd "A") 'neotree-stretch-toggle) (evil-define-key 'normal neotree-mode-map (kbd "H") 'neotree-hidden-file-toggle)) #+end_src *** Keybinds #+begin_src emacs-lisp (jd/leader-key-def "op" '(jd/neotree-project-dir :which-key "Open neotree")) #+end_src ** Magit #+begin_src emacs-lisp (use-package magit :custom (magit-display-buffer-function #'magit-display-buffer-same-window-except-diff-v1)) #+end_src *** Keybinds #+begin_src emacs-lisp (jd/leader-key-def "g" '(:ignore t :which-key "Git") "gg" '(magit-status-here :which-key "Magit status")) #+end_src * Apps ** Vterm #+begin_src emacs-lisp (defun jd/open-new-vterm (&optional jd/vterm-buffer-name) (interactive) (let ((buffer-name (generate-new-buffer-name vterm-buffer-name))) (when jd/vterm-buffer-name (setq buffer-name jd/vterm-buffer-name)) (switch-to-buffer buffer-name) (vterm-mode))) (use-package vterm :config (setq vterm-buffer-name "vterm") (evil-set-initial-state 'vterm-mode 'emacs)) #+end_src *** Keybinds #+begin_src emacs-lisp (jd/leader-key-def "ot" '(jd/open-new-vterm :which-key "Open terminal in current window")) #+end_src ** Dired #+begin_src emacs-lisp (use-package all-the-icons-dired :hook (dired-mode . all-the-icons-dired-mode)) (use-package dired-ranger) (use-package dired-single) (use-package dired :ensure nil :commands (dired dired-jump) :custom ((dired-listing-switches "-agho --group-directories-first")) :config (evil-collection-define-key 'normal 'dired-mode-map "y" 'dired-ranger-copy "p" 'dired-ranger-paste "X" 'dired-ranger-move "h" 'dired-up-directory "l" 'dired-single-buffer) (setq dired-kill-when-opening-new-dired-buffer t) (evil-define-key 'normal dired-mode-map (kbd "q") 'kill-current-buffer)) #+end_src ** EMMS #+begin_src emacs-lisp (use-package emms :config (require 'emms-setup) (emms-all) (emms-standard) (emms-default-players) (emms-mode-line-disable) ; (setq emms-info-functions '(emms-info-tinytag)) (setq emms-browser-covers 'emms-browser-cache-thumbnail-async) (setq emms-lyrics-dir "~/Documents/music/lyrics") (emms-add-directory-tree "~/Documents/music/")) #+end_src ** PDF Tools #+begin_src emacs-lisp (use-package pdf-tools) #+end_src ** Mu4e #+begin_src emacs-lisp ;; Load my mu4e config (require 'jd-mu4e) #+end_src * Runtime Performance #+begin_src emacs-lisp (setq gc-cons-threshold (* 2 1000 1000)) #+end_src