init.el 106 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836
  1. ;;; init.el --- Configuration entry point -*- lexical-binding: t -*-
  2. ;;; Commentary:
  3. ;;; Code:
  4. (require 'cl-lib)
  5. (require 'xdg)
  6. ;; Some other config files
  7. (cl-eval-when (compile load eval)
  8. (add-to-list 'load-path (expand-file-name "elisp" user-emacs-directory))
  9. (add-to-list 'load-path (expand-file-name "third-party" user-emacs-directory)))
  10. ;; Set package dir to follow no-littering conventions
  11. (setq package-user-dir (expand-file-name "var/elpa"
  12. user-emacs-directory))
  13. ;; Use melpa
  14. (require 'package)
  15. (add-to-list 'package-archives '("melpa" . "https://melpa.org/packages/") t)
  16. (package-initialize)
  17. ;; Ensure use-package is installed
  18. (unless (package-installed-p 'use-package)
  19. (package-refresh-contents)
  20. (package-install 'use-package))
  21. ;; use-package
  22. (eval-when-compile
  23. (require 'use-package)
  24. (setq use-package-always-ensure t
  25. package-user-dir (expand-file-name "var/elpa"
  26. user-emacs-directory)))
  27. ;; no-littering
  28. (use-package no-littering
  29. :autoload (no-littering-theme-backups
  30. no-littering-expand-etc-file-name)
  31. :init
  32. (no-littering-theme-backups)
  33. (setq custom-file (no-littering-expand-etc-file-name "custom.el")))
  34. ;; load things saved with custom
  35. (load custom-file t t)
  36. ;; diminish
  37. (use-package diminish
  38. :config
  39. (diminish 'visual-line-mode)
  40. (diminish 'abbrev-mode))
  41. ;; Private config loading
  42. (require 'private nil t)
  43. (defun my/get-private (key)
  44. "Get the private config variable KEY from the private configuration file."
  45. (alist-get key my/private-config))
  46. ;; basic stuff
  47. (use-package emacs
  48. :hook (;;(emacs-lisp-mode . my/-emacs-lisp-mode-setup-evil-lookup)
  49. ;;(prog-mode . electric-pair-local-mode)
  50. ((text-mode tex-mode prog-mode) . auto-fill-mode)
  51. ((text-mode tex-mode prog-mode) . my/-enable-show-trailing-whitespace))
  52. :init
  53. (with-eval-after-load 'find-func
  54. (when (file-directory-p "~/src/emacs/src/")
  55. (setq find-function-C-source-directory "~/src/emacs/src/")))
  56. (defun my/-enable-show-trailing-whitespace ()
  57. (setq-local show-trailing-whitespace t))
  58. ;; (defun my/-emacs-lisp-mode-setup-evil-lookup ()
  59. ;; (setq-local evil-lookup-func
  60. ;; #'my/describe-symbol-at-point))
  61. (defun my/describe-symbol-at-point ()
  62. "Calls `describe-symbol' on the return value of `symbol-at-point'."
  63. (interactive)
  64. (let ((form (symbol-at-point)))
  65. (if (consp form)
  66. (describe-symbol (cadr form))
  67. (describe-symbol form))))
  68. ;; Increase responsiveness
  69. (setq gc-cons-threshold 80000000
  70. read-process-output-max (* 1024 1024)) ;; 1mb
  71. (global-so-long-mode 1)
  72. ;; Terminal mouse support
  73. (xterm-mouse-mode 1)
  74. ;; Make cursor more visible
  75. (global-hl-line-mode 1)
  76. (blink-cursor-mode -1)
  77. ;; Enable all disabled stuff
  78. (setq disabled-command-function nil)
  79. ;; Stop some annoying stuff
  80. (setq extended-command-suggest-shorter nil
  81. suggest-key-bindings nil)
  82. ;; Better scrolling
  83. (setq mouse-scroll-delay 0
  84. scroll-conservatively 10
  85. scroll-margin 2
  86. scroll-preserve-screen-position t)
  87. ;; Make show paren instant
  88. (setq show-paren-delay 0)
  89. (show-paren-mode 1)
  90. ;; Display line numbers
  91. (global-display-line-numbers-mode 1)
  92. ;; Allow the frame to be any size
  93. (setq frame-resize-pixelwise t)
  94. ;; Don't use a gtk file picker
  95. (setq use-file-dialog nil)
  96. ;; Make yes-or-no-p less verbose (and not use windows)
  97. (setq use-dialog-box nil
  98. use-short-answers t)
  99. ;; Disable startup screen
  100. (setq inhibit-startup-screen t
  101. server-client-instructions nil)
  102. ;; show column numbers
  103. (column-number-mode 1)
  104. ;; Disable the menu and tool bars
  105. (menu-bar-mode -1)
  106. (tool-bar-mode -1)
  107. ;; No scroll bars
  108. (scroll-bar-mode -1)
  109. ;; Visual line mode
  110. (global-visual-line-mode 1)
  111. ;; Make some commands easier to enter multiple times
  112. (repeat-mode 1)
  113. ;; Easier buffer navigation
  114. (keymap-global-set "C-c <" #'previous-buffer)
  115. (keymap-global-set "C-c >" #'next-buffer)
  116. (keymap-global-set "C-c k" #'previous-buffer)
  117. (keymap-global-set "C-c j" #'next-buffer)
  118. ;; Seems useful...
  119. (keymap-global-set "C-c u" #'browse-url)
  120. ;; Set fonts
  121. (add-to-list 'default-frame-alist '(font . "FiraCode Nerd Font Mono-12"))
  122. (add-hook 'server-after-make-frame-hook
  123. (lambda ()
  124. (set-fontset-font t 'japanese-jisx0208 "IPAGothic")))
  125. ;; Enable color in compilation buffers
  126. (add-hook 'compilation-filter-hook 'ansi-color-compilation-filter)
  127. ;; Some settings for programming
  128. (setq-default indent-tabs-mode nil
  129. tab-width 4
  130. fill-column 80
  131. comment-multi-line t
  132. comment-empty-lines 'eol)
  133. (add-to-list 'auto-mode-alist '("\\.[cC][nN][fF]\\'" . conf-mode))
  134. (keymap-set emacs-lisp-mode-map "C-c C-r" #'eval-region)
  135. (defun my/-fix-emacs-lisp-mode-system-files ()
  136. (when (string-prefix-p lisp-directory buffer-file-name)
  137. ;; system Emacs files use tab characters and look weird without this.
  138. (setq-local tab-width 8)))
  139. (add-hook 'emacs-lisp-mode-hook #'my/-fix-emacs-lisp-mode-system-files)
  140. ;; Tree sitter download locations
  141. (setq treesit-language-source-alist
  142. '((c "https://github.com/tree-sitter/tree-sitter-c")
  143. (cpp "https://github.com/tree-sitter/tree-sitter-cpp")
  144. (java "https://github.com/tree-sitter/tree-sitter-java")
  145. ;; (glsl "https://github.com/tree-sitter-grammars/tree-sitter-glsl")
  146. (python "https://github.com/tree-sitter/tree-sitter-python")
  147. (rust "https://github.com/tree-sitter/tree-sitter-rust")
  148. (json "https://github.com/tree-sitter/tree-sitter-json")
  149. (yaml "https://github.com/ikatyang/tree-sitter-yaml")
  150. (css "https://github.com/tree-sitter/tree-sitter-css")
  151. (go "https://github.com/tree-sitter/tree-sitter-go")
  152. (gomod "https://github.com/camdencheek/tree-sitter-go-mod")
  153. (javascript "https://github.com/tree-sitter/tree-sitter-javascript")
  154. (typescript "https://github.com/tree-sitter/tree-sitter-typescript"
  155. nil "typescript/src")
  156. (tsx "https://github.com/tree-sitter/tree-sitter-typescript"
  157. nil "tsx/src")
  158. (bash "https://github.com/tree-sitter/tree-sitter-bash")
  159. (cmake "https://github.com/uyha/tree-sitter-cmake")
  160. (blueprint "https://github.com/huanie/tree-sitter-blueprint")
  161. (kdl "https://github.com/tree-sitter-grammars/tree-sitter-kdl")))
  162. ;; Tree sitter major mode conversions
  163. (dolist (ent '((c-mode . c-ts-mode)
  164. (c++-mode . c++-ts-mode)
  165. (c-or-c++-mode . c-or-c++-ts-mode)
  166. (python-mode . python-ts-mode)
  167. (java-mode . java-ts-mode)
  168. (rust-mode . rust-ts-mode)
  169. (json-mode . json-ts-mode)
  170. (yaml-mode . yaml-ts-mode)
  171. (css-mode . css-ts-mode)
  172. (javascript-mode . js-ts-mode)
  173. (cmake-mode . cmake-ts-mode)))
  174. (add-to-list 'major-mode-remap-alist ent))
  175. (defun my/treesit-compile-all (force)
  176. "Compile all the modules defined in `treesit-language-source-alist'.
  177. If FORCE, recompile all modules, even ones that are already compiled.
  178. Interactively, force the recompile if called with a prefix."
  179. (interactive "P")
  180. (let ((did-build nil))
  181. (dolist (lang treesit-language-source-alist)
  182. (when (or force (not (treesit-language-available-p (car lang))))
  183. (treesit-install-language-grammar (car lang))
  184. (setq did-build t)))
  185. (unless did-build
  186. (message "All defined parsers installed!")))))
  187. (use-package auth-source
  188. :ensure nil
  189. :custom
  190. (auth-sources '("~/.authinfo.gpg")))
  191. (use-package tramp
  192. :ensure nil
  193. :config
  194. (connection-local-set-profile-variables
  195. 'remote-direct-async-process
  196. '((tramp-direct-async-process . t)))
  197. (connection-local-set-profiles
  198. '(:protocol "ssh")
  199. 'remote-direct-async-process))
  200. (use-package midnight
  201. :ensure nil
  202. :config
  203. (add-to-list 'clean-buffer-list-kill-never-buffer-names
  204. "*mu4e-main*")
  205. (add-to-list 'clean-buffer-list-kill-never-buffer-names
  206. "*Async-native-compile-log*")
  207. (add-to-list 'clean-buffer-list-kill-never-buffer-names
  208. "*dashboard*")
  209. (add-to-list 'clean-buffer-list-kill-never-buffer-names
  210. "*elfeed-search*")
  211. (midnight-mode 1))
  212. (defvar my/kill-some-buffers-exclude-names
  213. '("*mu4e-main*" "*Async-native-compile-log*" "*dashboard*" "*elfeed-search*"
  214. "*Messages*" "*scratch*")
  215. "List of literal buffer names that `my/kill-some-buffers' should not kill.")
  216. (defun my/kill-some-buffers-excluded-buffer-p (buffer)
  217. "Return non-nil if BUFFER should be excluded from `my/kill-some-buffers'."
  218. (cl-find (buffer-name buffer) my/kill-some-buffers-exclude-names
  219. :test 'equal))
  220. (defun my/buffer-visible-p (buffer)
  221. "Return non-nil if BUFFER is visible.
  222. BUFFER can be a string or a buffer."
  223. (cond
  224. ((stringp buffer)
  225. (not (string-prefix-p " " buffer)))
  226. ((bufferp buffer)
  227. (and (stringp (buffer-name buffer))
  228. (my/buffer-visible-p (buffer-name buffer))))
  229. (t
  230. (signal 'wrong-type-argument `((or bufferp stringp) ,buffer)))))
  231. (defvar my/kill-some-buffers-default-pred 'my/buffer-visible-p
  232. "Default predicate for `my/kill-some-buffers'.")
  233. (defun my/kill-some-buffers-prompt-for (buffer)
  234. "Generate a prompt for BUFFER."
  235. (let* ((process (get-buffer-process buffer))
  236. (process-p (and (process-live-p process)
  237. (not (process-query-on-exit-flag process))))
  238. (modified-p (and (buffer-file-name buffer)
  239. (buffer-modified-p buffer))))
  240. (format "Buffer \"%s\" %s. Kill? "
  241. (buffer-name buffer)
  242. (cond
  243. ((and process-p modified-p)
  244. "HAS BEEN EDITED AND HAS A LIVE PROCESS")
  245. (modified-p
  246. "HAS BEEN EDITED")
  247. (process-p
  248. "HAS A LIVE PROCESS")
  249. (t "is unmodified")))))
  250. (cl-defun my/kill-some-buffers (&optional auto-unmod pred)
  251. "Improved version of `kill-some-buffers'.
  252. Ask the user weather to kill each visible buffer whose name is not in
  253. `my/kill-some-buffers-exclude-names'.
  254. When AUTO-UNMOD is non-nil, as it is with a prefix argument, automatically kill
  255. unmodified buffers, and then ask about the rest.
  256. When PRED is non-nil, it is a function that will be run in each buffer (not just
  257. visible ones). If it returns t, that buffer will be considered for killing. If
  258. PRED is nil, the value of `my/kill-some-buffers-default-pred' is used."
  259. (interactive "P")
  260. ;; we already ask, no need to do it again
  261. (let ((kill-buffer-query-functions nil)
  262. (all-action (when auto-unmod 'unmod))
  263. (had-valid-buffer)
  264. (ask-again-buffers)
  265. (to-kill))
  266. (cl-flet ((ask-about (buffer allow-unmod)
  267. (unless all-action
  268. (read-answer
  269. (my/kill-some-buffers-prompt-for buffer)
  270. `(("yes" ?y "save and kill this buffer")
  271. ("no" ?n "skip this buffer")
  272. ("all" ?! "save and kill all remaining buffers")
  273. ("nosave" ?l "kill this buffer without saving")
  274. ,@(when allow-unmod
  275. '(("unmod" ?a
  276. "kill unmodified buffers, ask about the rest")))
  277. ("quit" ?q "exit")))))
  278. (act-on (ans buffer allow-unmod)
  279. (when (equal ans "all")
  280. (setq all-action 'all))
  281. (when (and allow-unmod
  282. (equal ans "unmod"))
  283. (setq all-action 'unmod))
  284. (cond
  285. ((and (eq all-action 'unmod)
  286. (buffer-file-name buffer)
  287. (buffer-modified-p buffer))
  288. (push buffer ask-again-buffers))
  289. ((or (eq all-action 'all)
  290. (eq all-action 'unmod)
  291. (equal ans "yes"))
  292. (when (buffer-file-name buffer)
  293. (with-current-buffer buffer
  294. (save-buffer)))
  295. (push buffer to-kill))
  296. ((equal ans "nosave")
  297. (with-current-buffer buffer
  298. (set-buffer-modified-p nil))
  299. (push buffer to-kill))
  300. ;; Skip buffer
  301. ;; ((equal ans "no"))
  302. ((equal ans "quit")
  303. (cl-return-from my/kill-some-buffers)))))
  304. (dolist (buffer (buffer-list))
  305. (when (and (not (my/kill-some-buffers-excluded-buffer-p buffer))
  306. (funcall (or pred my/kill-some-buffers-default-pred) buffer))
  307. (setq had-valid-buffer t)
  308. (act-on (ask-about buffer t) buffer t)))
  309. (unless had-valid-buffer
  310. (message "Nothing to do..."))
  311. (setq all-action nil)
  312. (dolist (buffer ask-again-buffers)
  313. (act-on (ask-about buffer nil) buffer nil))
  314. ;; Do this last so that tty frames don't auto-close half way through
  315. (mapc 'kill-buffer to-kill))))
  316. (keymap-global-set "C-x K" 'my/kill-some-buffers)
  317. (use-package tab-bar
  318. :ensure nil
  319. :init
  320. (setq tab-bar-show 1
  321. tab-bar-tab-hints t
  322. icon-preference '(symbol text image emoji))
  323. (tab-bar-mode 1))
  324. ;; jinx (better flyspell)
  325. (use-package jinx
  326. :hook (emacs-startup . global-jinx-mode)
  327. :config
  328. (evil-define-key 'normal 'global
  329. "z=" #'jinx-correct)
  330. (defun my/jinx-visit-dictionary (language &optional other-window)
  331. "Visit the dictionary file for LANGUAGE in another window.
  332. With OTHER-WINDOW, visit the file in another window. Interactively, use the
  333. current buffer's language, prompting if there is more than one. OTHER-WINDOW is
  334. t with a prefix argument."
  335. (interactive (list
  336. (let ((langs (split-string jinx-languages " ")))
  337. (if (length= langs 1)
  338. (car langs)
  339. (completing-read "Language: " langs nil t)))
  340. current-prefix-arg))
  341. (let* ((config-dir (expand-file-name "enchant" (xdg-config-home)))
  342. (dict-path (expand-file-name (concat language ".dic") config-dir)))
  343. (if other-window
  344. (find-file-other-window dict-path)
  345. (find-file dict-path)))))
  346. ;; recentf
  347. (use-package recentf
  348. :init
  349. (setq recentf-exclude `("^/tmp/.*"
  350. "^~/.mail/[^/]/Drafts/.*"
  351. ,(format "^%svar/dape-breakpoints" user-emacs-directory)
  352. ,(format "^%svar/elpa/.*" user-emacs-directory)
  353. ,(format "^%svar/elfeed/.*" user-emacs-directory)
  354. ,(format "^%svar/gnus/.*" user-emacs-directory)
  355. ,(format "^%svar/ellama-sessions/.*" user-emacs-directory)
  356. ,(format "^%setc/gnus/.*" user-emacs-directory)
  357. ,(format "^%svar/bookmark-default.el" user-emacs-directory)))
  358. :bind ("C-c r" . recentf)
  359. :config
  360. (recentf-mode 1))
  361. ;; bookmarks
  362. (use-package bookmark
  363. :ensure nil
  364. :bind ("C-c B" . my/bookmark-find-file)
  365. :config
  366. (defun my/bookmark-find-file (&optional name)
  367. "Run `find-file' in or on bookmark NAME.
  368. If NAME points to a directory, run `find-file' with `default-directory' in that
  369. directory. Otherwise, run `find-file' on that file."
  370. (interactive (list (bookmark-completing-read
  371. "Find file in" bookmark-current-bookmark)))
  372. (unless name
  373. (error "No bookmark specified"))
  374. (bookmark-maybe-historicize-string name)
  375. (when-let ((file (bookmark-get-filename name)))
  376. (if (file-directory-p file)
  377. (let ((default-directory (file-name-as-directory file)))
  378. (call-interactively 'find-file))
  379. (find-file file)))))
  380. ;; kitty keyboard protocol
  381. (use-package kkp
  382. :defer nil
  383. :config
  384. (global-kkp-mode 1)
  385. ;; (defun my/-kkp-after-terminal-setup ()
  386. ;; ;; Make tab and backtab work properly
  387. ;; (define-key input-decode-map [(control ?i)] [tab])
  388. ;; (define-key input-decode-map [(control ?I)] [backtab])
  389. ;; (define-key input-decode-map [(control ?m)] [return]))
  390. ;; (defun my/-kkp-after-terminal-teardown (term)
  391. ;; (with-selected-frame (car (frames-on-display-list term))
  392. ;; (define-key input-decode-map [(control ?i)] nil t)
  393. ;; (define-key input-decode-map [(control ?I)] nil t)
  394. ;; (define-key input-decode-map [(control ?m)] nil t)))
  395. ;; (advice-add 'kkp--terminal-setup :after 'my/-kkp-after-terminal-setup)
  396. ;; (advice-add 'kkp--terminal-teardown :after 'my/-kkp-after-terminal-teardown)
  397. (defun my/quoted-insert (arg)
  398. "Insert the next character using read-key, not read-char."
  399. (interactive "*p")
  400. ;; Source: https://github.com/benjaminor/kkp/issues/11
  401. (let ((char (read-key)))
  402. ;; Ensure char is treated as a character code for insertion
  403. (unless (characterp char)
  404. (user-error "%s is not a valid character"
  405. (key-description (vector char))))
  406. (when (numberp char)
  407. (while (> arg 0)
  408. (insert-and-inherit char)
  409. (setq arg (1- arg))))))
  410. (keymap-global-set "C-q" #'my/quoted-insert)
  411. (defun my/-kkp-fix-map-y-or-n-p (oldfun &rest args)
  412. "Fix `map-y-or-n-p' when used in a terminal with kkp enabled."
  413. (let ((status (kkp--terminal-has-active-kkp-p)))
  414. (condition-case err
  415. (progn
  416. (when status (kkp-disable-in-terminal))
  417. (apply oldfun args))
  418. (quit
  419. ;; We won't die in this case, so just re-enable kkp
  420. (when (and status (not (kkp--terminal-has-active-kkp-p)))
  421. (kkp-enable-in-terminal))
  422. (signal 'quit nil))
  423. (t
  424. (when (and status (not (kkp--terminal-has-active-kkp-p)))
  425. ;; this does async stuff that will make kitty send characters after
  426. ;; Emacs exits. We prevent that by not re-enabling if this frame (or
  427. ;; Emacs) is about to die
  428. (let ((will-die))
  429. (mapbacktrace
  430. (lambda (_evald func _args _flags)
  431. (when (or (eq func 'save-buffers-kill-emacs)
  432. (eq func 'server-save-buffers-kill-terminal))
  433. (setq will-die t))))
  434. (unless will-die
  435. (kkp-enable-in-terminal))))
  436. (when err
  437. (signal (car err) (cdr err)))))))
  438. (advice-add #'map-y-or-n-p :around
  439. #'my/-kkp-fix-map-y-or-n-p))
  440. ;; mozc
  441. (require 'mozc nil t)
  442. (setq default-input-method "japanese-mozc")
  443. ;; migemo
  444. (use-package migemo
  445. :config
  446. (setq migemo-dictionary "/usr/share/migemo/utf-8/migemo-dict"
  447. migemo-coding-system 'utf-8-unix
  448. migemo-regex-dictionary nil
  449. migemo-user-dictionary nil
  450. migemo-isearch-enable-p t)
  451. (keymap-global-set "C-c w" 'isearch-forward)
  452. (keymap-global-set "C-c W" 'isearch-backward)
  453. (migemo-init))
  454. ;; undo-tree
  455. (use-package undo-tree
  456. :defer nil
  457. :hook (undo-tree-visualizer-mode . my/-undo-tree-visualizer-mode-setup)
  458. :config
  459. (defun my/-undo-tree-visualizer-mode-setup ()
  460. (visual-line-mode -1)
  461. (setq truncate-lines t))
  462. (global-undo-tree-mode))
  463. ;; evil
  464. (use-package evil
  465. :init
  466. (setq evil-want-integration t
  467. evil-want-C-d-scroll nil
  468. evil-want-keybinding nil
  469. evil-undo-system 'undo-tree
  470. evil-search-module 'isearch
  471. evil-respect-visual-line-mode t)
  472. :config
  473. (evil-mode 1)
  474. (evil-define-key '(normal visual motion) proced-mode-map
  475. "u" #'proced-unmark)
  476. (evil-define-key '(normal visual motion) dired-mode-map
  477. "u" #'dired-unmark)
  478. (evil-define-key '(normal visual motion) profiler-report-mode-map
  479. (kbd "TAB") #'profiler-report-toggle-entry)
  480. (eldoc-add-command 'evil-insert
  481. 'evil-append
  482. 'evil-insert-line
  483. 'evil-append-line))
  484. (use-package evil-collection
  485. :after evil
  486. :diminish evil-collection-unimpaired-mode
  487. :config
  488. (evil-collection-init))
  489. (use-package evil-surround
  490. :after evil
  491. :config
  492. (evil-define-key 'operator evil-surround-mode-map
  493. "z" #'evil-surround-edit
  494. "Z" #'evil-Surround-edit)
  495. (evil-define-key 'visual evil-surround-mode-map
  496. "gz" #'evil-surround-region
  497. "gZ" #'evil-Surround-region)
  498. (global-evil-surround-mode 1))
  499. (use-package evil-terminal-cursor-changer
  500. :after evil
  501. :config
  502. (evil-terminal-cursor-changer-activate))
  503. (use-package evil-numbers
  504. :after evil
  505. :bind (("C-c =" . evil-numbers/inc-at-pt)
  506. ("C-c +" . evil-numbers/inc-at-pt)
  507. ("C-c -" . evil-numbers/dec-at-pt)
  508. ("C-c C-=" . evil-numbers/inc-at-pt-incremental)
  509. ("C-c C-+" . evil-numbers/inc-at-pt-incremental)
  510. ("C-c C--" . evil-numbers/dec-at-pt-incremental)))
  511. (use-package evil-cleverparens
  512. :hook ((prog-mode . my/-enable-evil-cleverparens)
  513. (evil-cleverparens-mode . paredit-mode))
  514. :bind (:map paredit-mode-map
  515. ("C-<return>" . paredit-RET)
  516. ("C-RET" . paredit-RET)
  517. :map evil-cleverparens-mode-map
  518. ("C-c o" . evil-cp-open-below-form))
  519. :custom
  520. (evil-cleverparens-use-s-and-S nil)
  521. (evil-cleverparens-complete-parens-in-yanked-region t)
  522. :config
  523. (eldoc-add-command 'paredit-RET
  524. 'paredit-open-round
  525. 'paredit-open-angled
  526. 'paredit-open-bracket
  527. 'paredit-open-angled
  528. 'paredit-open-parenthesis
  529. 'delete-indentation
  530. 'evil-cp-insert
  531. 'evil-cp-append
  532. 'evil-cp-insert-at-beginning-of-form
  533. 'evil-cp-insert-at-end-of-form)
  534. (keymap-unset evil-cleverparens-mode-map "<normal-state> M-o" t)
  535. (defun my/-enable-evil-cleverparens ()
  536. (if (member major-mode '(lisp-mode emacs-lisp-mode
  537. lisp-interaction-mode))
  538. (evil-cleverparens-mode 1)
  539. (electric-pair-local-mode 1)))
  540. (cl-defun my/range-inside-thing-p (thing beg end &optional no-edge)
  541. "Return non-nil if BEG and END fall inside the bounds of THING.
  542. With NO-EDGE, return nil if beg or end fall on the edge of the range."
  543. (save-excursion
  544. ;; this fixes that fact that `thing-at-point-bounds-of-string-at-point'
  545. ;; errors if called at the end of the buffer
  546. (condition-case nil
  547. (let ((sb (progn (goto-char beg) (bounds-of-thing-at-point thing)))
  548. (eb (progn (goto-char end) (bounds-of-thing-at-point thing))))
  549. (and sb eb (equal sb eb)
  550. (or (not no-edge)
  551. (and (/= beg (car sb))
  552. (< beg (cdr sb))
  553. (/= end (car sb))
  554. (< end (cdr sb))))))
  555. ;; if the error happens, we aren't in a string
  556. (wrong-type-argument nil))))
  557. (defun my/-evil-cp-region-ok-p-no-string (oldfun beg end)
  558. (or
  559. (and (sp-point-in-comment beg)
  560. (sp-point-in-comment end))
  561. (and (sp-point-in-string beg)
  562. (sp-point-in-string end)
  563. (my/range-inside-thing-p 'string beg end t))
  564. (funcall oldfun beg end)))
  565. (defun my/column-num-at-pos (pos)
  566. "Return the column number at POS."
  567. (save-excursion
  568. (goto-char pos)
  569. (current-column)))
  570. (defun my/-evil-cp-block-ok-p-no-string (oldfun beg end)
  571. (when (> beg end) (cl-rotatef beg end))
  572. (or
  573. (save-excursion
  574. (goto-char beg)
  575. (let ((start-off (current-column))
  576. (end-off (my/column-num-at-pos end)))
  577. (cl-block nil
  578. (dotimes (_ (count-lines beg end) t)
  579. (let ((bol (pos-bol)))
  580. (unless (sp-region-ok-p (+ bol start-off)
  581. (+ bol end-off))
  582. (cl-return))
  583. (forward-line))))))
  584. (funcall oldfun beg end)))
  585. (advice-add 'sp-region-ok-p :around 'my/-evil-cp-region-ok-p-no-string)
  586. (advice-add 'evil-cp--balanced-block-p :around 'my/-evil-cp-block-ok-p-no-string))
  587. ;; be (hopefully) safer
  588. (require 'trusted-files)
  589. (keymap-global-set "C-c t" 'trusted-files-map)
  590. (trusted-files-modeline-mode)
  591. ;; better lisp editing
  592. (use-package adjust-parens
  593. :hook (prog-mode . adjust-parens-mode)
  594. :config
  595. (defun my/lisp-indent-adjust-parens ()
  596. "Like `lisp-indent-adjust-parens', but got to first char on line first.
  597. Also, this works even if the region is active (it indents every line in the
  598. region)."
  599. (interactive)
  600. (save-mark-and-excursion
  601. (let ((end (mark t))
  602. (line-count 1)
  603. (indent-cols))
  604. (when (and (region-active-p) end)
  605. (setq mark-active nil
  606. line-count (count-lines (point) end))
  607. (when (> (point) end)
  608. (let ((start (point)))
  609. (goto-char end)
  610. (setq end start))))
  611. ;; find the indentation column of each line
  612. (save-excursion
  613. (dotimes (_ line-count)
  614. (back-to-indentation)
  615. (push (- (point) (pos-bol)) indent-cols)
  616. (forward-line))
  617. (cl-callf nreverse indent-cols))
  618. (cl-loop repeat line-count
  619. for indent-col in indent-cols
  620. for bol = (pos-bol)
  621. do (back-to-indentation)
  622. ;; skip this line if the indentation has changed
  623. when (= (- (point) bol) indent-col) do
  624. (lisp-indent-adjust-parens)
  625. ;; if the indent failed, stop
  626. (when (= (- (point) bol) indent-col)
  627. (cl-return))
  628. do (forward-line)))))
  629. (defun my/lisp-dedent-adjust-parens ()
  630. "Like `lisp-dedent-adjust-parens', but got to first char on line first.
  631. Also, this works even if the region is active (it just jumps to the first line
  632. in the region and indents once)."
  633. (interactive)
  634. (save-mark-and-excursion
  635. (let ((end (mark t)))
  636. (when (and (region-active-p) end)
  637. (setq mark-active nil)
  638. (when (> (point) end)
  639. (goto-char end))))
  640. (back-to-indentation)
  641. (lisp-dedent-adjust-parens)))
  642. (eldoc-add-command 'my/lisp-indent-adjust-parens
  643. 'my/lisp-dedent-adjust-parens
  644. 'lisp-indent-adjust-parens
  645. 'lisp-dedent-adjust-parens)
  646. (evil-define-key '(normal visual) adjust-parens-mode-map
  647. (kbd "<tab>") #'my/lisp-indent-adjust-parens
  648. (kbd "<backtab>") #'my/lisp-dedent-adjust-parens
  649. (kbd "C-c C-i") #'my/lisp-indent-adjust-parens
  650. (kbd "C-c S-<tab>") #'my/lisp-dedent-adjust-parens))
  651. ;; for when the files are just too large
  652. (use-package vlf
  653. :demand t
  654. :config
  655. (require 'vlf-setup))
  656. ;; allow copy from terminal
  657. (use-package xclip
  658. :config
  659. (setq xclip-method 'wl-copy
  660. xclip-program (symbol-name xclip-method))
  661. (xclip-mode 1)
  662. (defun my/-xclip-detect-wl-paste-error (oldfun type)
  663. (if (eq xclip-method 'wl-copy)
  664. ;; Direct from `xclip-get-selection'
  665. (when (and (getenv "WAYLAND_DISPLAY")
  666. (memq type '(clipboard CLIPBOARD primary PRIMARY)))
  667. (let* ((exit-code 0)
  668. (output
  669. (with-output-to-string
  670. (setq exit-code
  671. (apply #'call-process (replace-regexp-in-string
  672. "\\(.*\\)copy" "\\1paste"
  673. xclip-program 'fixedcase)
  674. nil standard-output nil
  675. "-n" (if (memq type '(primary PRIMARY))
  676. '("-p")))))))
  677. (if (zerop exit-code)
  678. output
  679. "")))
  680. (funcall oldfun type)))
  681. (advice-add 'xclip-get-selection :around 'my/-xclip-detect-wl-paste-error))
  682. ;; Set the WAYLAND_DISPLAY environment variable
  683. (require 'xdg)
  684. (defun my/detect-wayland-display ()
  685. "Try to set the WAYLAND_DISPLAY environment variable.
  686. This attempts to detect a running Wayland session and set the WAYLAND_DISPLAY
  687. environment variable accordingly."
  688. (let ((found '(nil . nil)))
  689. (dolist (entry (directory-files-and-attributes
  690. (xdg-runtime-dir) nil nil 'nosort)
  691. (cdr found))
  692. (cl-destructuring-bind (name . attrs) entry
  693. (when-let (((string-match (rx bos "wayland-" (group (+ (any "0-9"))) eos)
  694. name))
  695. (id (string-to-number (match-string 1 name)))
  696. ((or (not (car found)) (< id (car found))))
  697. ;; socket
  698. ((string-prefix-p "s" (file-attribute-modes attrs))))
  699. (setq found (cons id name)))))))
  700. (unless (getenv "WAYLAND_DISPLAY")
  701. (setenv "WAYLAND_DISPLAY" (my/detect-wayland-display)))
  702. ;; which-key
  703. (use-package which-key
  704. :diminish which-key-mode
  705. :config
  706. (which-key-mode 1))
  707. ;; avy
  708. (use-package avy
  709. :bind (("C-c C-j" . avy-resume)
  710. ("M-s s" . evil-avy-goto-char-2)
  711. ("M-s S" . evil-avy-goto-line))
  712. :init
  713. (define-minor-mode my/evil-avy-mode
  714. "A minor mode for binding avy commands to s and S in evil's normal and
  715. visual states."
  716. :keymap (make-sparse-keymap))
  717. (evil-define-key '(normal visual operator motion) my/evil-avy-mode-map
  718. "s" #'evil-avy-goto-char-2
  719. "S" #'evil-avy-goto-line)
  720. (define-globalized-minor-mode my/evil-avy-global-mode my/evil-avy-mode
  721. (lambda () (my/evil-avy-mode 1))
  722. :predicate '((not magit-mode dired-mode
  723. proced-mode mu4e-main-mode
  724. mu4e-view-mode mu4e-headers-mode
  725. ibuffer-mode calc-mode calc-trail-mode
  726. gnus-group-mode) t))
  727. (my/evil-avy-global-mode 1)
  728. :config
  729. (avy-setup-default))
  730. ;; better `replace-regexp'
  731. (use-package visual-regexp
  732. :bind (("C-c q" . vr/replace)
  733. ("C-M-%" . vr/query-replace)))
  734. ;; better `align-regexp'
  735. (use-package ialign
  736. :defer t
  737. :custom
  738. (ialign-initial-repeat t))
  739. ;; ace-window
  740. (use-package ace-window
  741. :diminish ace-window-mode
  742. :bind ("M-o" . ace-window)
  743. :init
  744. (setq aw-scope 'frame
  745. aw-minibuffer-flag t))
  746. ;; savehist
  747. (use-package savehist
  748. :config
  749. (savehist-mode 1))
  750. ;; vertico
  751. (use-package vertico
  752. :bind (:map vertico-map
  753. ("C-RET" . vertico-exit-input)
  754. ("C-<return>" . vertico-exit-input)
  755. ("C-S-k" . kill-line)
  756. ("C-k" . vertico-previous)
  757. ("C-j" . vertico-next)
  758. ("RET" . vertico-directory-enter)
  759. ("DEL" . vertico-directory-delete-char)
  760. ("M-DEL" . vertico-directory-delete-word))
  761. :hook (minibuffer-setup . cursor-intangible-mode)
  762. :init
  763. (defun my/crm-indicator (args)
  764. (cons (format "[CRM%s] %s"
  765. (replace-regexp-in-string
  766. "\\`\\[.*?]\\*\\|\\[.*?]\\*\\'" ""
  767. crm-separator)
  768. (car args))
  769. (cdr args)))
  770. (advice-add #'completing-read-multiple :filter-args #'my/crm-indicator)
  771. (setq vertico-cycle t
  772. enable-recursive-minibuffers t
  773. ;; read-extended-command-predicate #'command-completion-default-include-p
  774. read-extended-command-predicate nil
  775. minibuffer-prompt-properties '(read-only t ;; noindent 3
  776. cursor-intangible t
  777. face minibuffer-prompt))
  778. (vertico-mode 1)
  779. ;; for jinx
  780. (require 'vertico-multiform)
  781. (add-to-list 'vertico-multiform-categories
  782. '(jinx grid (vertico-grid-annotate . 20)))
  783. (vertico-multiform-mode 1))
  784. ;; orderless
  785. (use-package orderless
  786. :autoload orderless-define-completion-style
  787. :hook (text-mode . my/-setup-text-mode-completion-styles)
  788. :init
  789. (defun my/-setup-text-mode-completion-styles ()
  790. (setq-local completion-styles '(basic)))
  791. (orderless-define-completion-style my/orderless-with-initialism
  792. (orderless-matching-styles '(orderless-initialism
  793. orderless-regexp)))
  794. (setq orderless-matching-styles '(orderless-regexp)
  795. completion-styles '(orderless basic)
  796. completion-category-defaults nil
  797. completion-category-overrides '((file
  798. (styles basic partial-completion))
  799. (command
  800. (styles my/orderless-with-initialism basic)))))
  801. ;; marginalia
  802. (use-package marginalia
  803. :bind (:map minibuffer-local-map
  804. ("M-a" . marginalia-cycle))
  805. :init
  806. (marginalia-mode 1))
  807. ;; embark
  808. (use-package embark
  809. :bind (("C-," . embark-act)
  810. ("C-;" . embark-dwim)
  811. :map help-map
  812. ("B" . embark-bindings)
  813. :map embark-symbol-map
  814. ("h" . helpful-symbol)
  815. :map embark-become-file+buffer-map
  816. ("b" . consult-buffer)
  817. ("B" . switch-to-buffer))
  818. :init
  819. (setq embark-quit-after-action nil
  820. embark-indicators '(embark-minimal-indicator
  821. embark-isearch-highlight-indicator
  822. embark-highlight-indicator))
  823. :config
  824. (defvar-keymap my/embark-string-map
  825. :doc "Keymap for Embark string actions."
  826. :parent embark-expression-map
  827. "R" 'repunctuate-sentences)
  828. (defun my/embark-target-string ()
  829. "Target the string at point."
  830. (if-let (((not (eobp))) ; prevent next line from causing errors
  831. (bounds (bounds-of-thing-at-point 'string)))
  832. (append (list 'string (buffer-substring-no-properties (car bounds)
  833. (cdr bounds)))
  834. bounds)))
  835. (add-to-list 'embark-around-action-hooks
  836. '(repunctuate-sentences embark--mark-target))
  837. (add-to-list 'embark-keymap-alist
  838. '(string my/embark-string-map))
  839. (add-to-list 'embark-target-finders 'my/embark-target-string)
  840. (add-to-list 'display-buffer-alist
  841. '("\\`\\*Embark Collect \\(Live\\|Completions\\)\\*"
  842. nil
  843. (window-parameters (mode-line-format . none))))
  844. (evil-define-key '(normal motion) org-mode-map
  845. (kbd "C-,") #'embark-act))
  846. ;; consult
  847. (use-package consult
  848. :bind (("C-s" . consult-line)
  849. ("C-x b" . consult-buffer)
  850. ("C-S-s" . consult-ripgrep)
  851. ("C-x C-S-f" . consult-fd)
  852. ("C-x c k" . consult-keep-lines)
  853. ("C-x c f" . consult-focus-lines)
  854. ("C-x c r" . consult-recent-file)
  855. ("C-x c b" . consult-bookmark)
  856. ("C-x c d" . consult-fd)
  857. ("C-c p" . consult-fd)
  858. ("C-x c g" . consult-ripgrep)
  859. ("C-c P" . consult-ripgrep)
  860. ("C-x c y" . consult-yank-from-kill-ring)
  861. ("M-g i" . consult-imenu)
  862. ("M-g I" . consult-imenu-multi)
  863. ("M-g r" . consult-imenu-multi)
  864. :map help-map
  865. ("TAB". consult-info)
  866. ("RET" . consult-man))
  867. :hook (minibuffer-setup . my/consult-setup-minibuffer-completion)
  868. :init
  869. (defun my/consult-setup-minibuffer-completion ()
  870. (setq-local completion-in-region-function #'consult-completion-in-region))
  871. (evil-declare-motion #'consult-line))
  872. (use-package consult-eglot
  873. :commands consult-eglot-symbols)
  874. ;; wgrep
  875. (use-package wgrep)
  876. ;; integration for embark and consult
  877. (use-package embark-consult
  878. :hook (embark-collect-mode . consult-preview-at-point-mode))
  879. ;; corfu (autocomplete)
  880. (use-package corfu
  881. :bind (("M-<tab>" . completion-at-point)
  882. :map corfu-map
  883. ("C-j" . corfu-next)
  884. ("C-k" . corfu-previous)
  885. ("M-SPC" . corfu-insert-separator)
  886. ("M-m" . my/corfu-move-to-minibuffer))
  887. :init
  888. (defun my/corfu-move-to-minibuffer ()
  889. (interactive)
  890. (when completion-in-region--data
  891. (let ((completion-extra-properties corfu--extra)
  892. (completion-cycle-threshold completion-cycling))
  893. (apply #'consult-completion-in-region completion-in-region--data))))
  894. (setq corfu-cycle t
  895. corfu-auto t
  896. corfu-on-exact-match nil
  897. corfu-popupinfo-delay '(1.0 . 0.5)
  898. completion-cycle-threshold nil
  899. global-corfu-minibuffer
  900. ;; only enable corfu in the minibuffer in graphical frames
  901. (lambda ()
  902. (and (display-graphic-p)
  903. (not (eq (current-local-map)
  904. read-passwd-map)))))
  905. (global-corfu-mode 1)
  906. (corfu-popupinfo-mode 1)
  907. :config
  908. (add-to-list 'corfu-continue-commands #'my/corfu-move-to-minibuffer)
  909. (defun my/help-buffer-exists-p ()
  910. "Return if the buffer that `help-buffer' would, or nil if it doesn't exist."
  911. (or (and help-xref-following (derived-mode-p 'help-mode))
  912. (get-buffer "*Help*")))
  913. (defun my/-corfu-popupinfo-close-help-buffer (oldfun &rest args)
  914. (if (derived-mode-p 'emacs-lisp-mode)
  915. (let ((help-buf (my/help-buffer-exists-p)))
  916. (prog1
  917. (apply oldfun args)
  918. (when-let (((not help-buf))
  919. (buf (help-buffer)))
  920. ;; Ensure that, even if `help-buffer' returns nil in the future, we
  921. ;; don't kill the current buffer
  922. (kill-buffer buf))))
  923. (apply oldfun args)))
  924. (advice-add 'corfu-popupinfo--get-documentation :around
  925. 'my/-corfu-popupinfo-close-help-buffer))
  926. (use-package corfu-terminal
  927. :init
  928. (corfu-terminal-mode 1)
  929. :config
  930. (require 'corfu-terminal-popupinfo)
  931. (corfu-terminal-popupinfo-mode 1))
  932. (use-package dabbrev
  933. :ensure nil
  934. :config
  935. (add-to-list 'dabbrev-ignored-buffer-regexps "\\` ")
  936. (add-to-list 'dabbrev-ignored-buffer-modes 'doc-view-mode)
  937. (add-to-list 'dabbrev-ignored-buffer-modes 'pdf-view-mode)
  938. (add-to-list 'dabbrev-ignored-buffer-modes 'tags-table-mode))
  939. ;; cape (a bunch of capfs!)
  940. (use-package cape
  941. :bind (([remap dabbrev-expand] . cape-dabbrev)
  942. ("C-c P" . cape-line)
  943. ("C-c f" . cape-file))
  944. :hook (text-mode . my/-cape-setup-text-mode)
  945. :init
  946. (defun my/-cape-setup-text-mode ()
  947. ;; Only run this if we are not in `TeX-mode'
  948. (unless (bound-and-true-p TeX-mode-p)
  949. (setq-local completion-at-point-functions
  950. (append completion-at-point-functions (list 'cape-dict
  951. 'cape-dabbrev))
  952. corfu-auto nil))))
  953. ;; xref
  954. (use-package xref
  955. :init
  956. (evil-define-key '(normal motion) 'global
  957. "gr" #'xref-find-references)
  958. (setq xref-show-xrefs-function #'consult-xref
  959. xref-show-definitions-function #'consult-xref))
  960. ;; popup.el
  961. (use-package popup)
  962. ;; posframe
  963. (use-package posframe
  964. :init
  965. (defun my/posframe-tip (name msg)
  966. "Like `popup-tip', but with a posframe.
  967. NAME should be the buffer name to pass to `posframe-show'. MSG is the message to
  968. display."
  969. (unwind-protect
  970. (progn
  971. (posframe-show name
  972. :string msg
  973. :position (point)
  974. :max-width 80
  975. :border-width 2
  976. :border-color "white")
  977. (clear-this-command-keys)
  978. (push (read-event) unread-command-events)
  979. (posframe-hide name))
  980. (posframe-hide name))))
  981. (defun my/floating-tooltip (name msg)
  982. "If `display-graphic-p', call `my/posframe-tip', otherwise `popup-tip'.
  983. MSG is the message to show in the popup. NAME is the name of the buffer to pass
  984. to `posframe-show' if the display is graphical."
  985. (if (display-graphic-p)
  986. (my/posframe-tip name msg)
  987. (popup-tip msg)))
  988. ;; flymake
  989. (use-package flymake
  990. :config
  991. (require 'consult-flymake))
  992. ;; flycheck
  993. (use-package flycheck
  994. :hook ((sh-mode emacs-lisp-mode) . trusted-files-flycheck-mode-if-safe)
  995. :custom
  996. (flycheck-indication-mode 'left-margin)
  997. :init
  998. (setq flycheck-display-errors-function nil))
  999. (use-package consult-flycheck)
  1000. (defun my/sly-notes-at-point (&optional pos buffer)
  1001. "Return the sly notes at POS in BUFFER.
  1002. If BUFFER is nil, the current buffer is used."
  1003. (with-current-buffer (or buffer (current-buffer))
  1004. (unless pos
  1005. (setq pos (point)))
  1006. (cl-loop for overlay in (overlays-at pos)
  1007. for note = (overlay-get overlay 'sly-note)
  1008. when note
  1009. collect note)))
  1010. (defun my/diagnostic-at-point ()
  1011. "Show the diagnostics under point."
  1012. (interactive)
  1013. (let ((message))
  1014. (when-let (((bound-and-true-p flymake-mode))
  1015. (diag (get-char-property (point) 'flymake-diagnostic)))
  1016. (cl-callf nconc message (string-split (flymake--diag-text diag) "\n" t)))
  1017. (when (bound-and-true-p flycheck-mode)
  1018. (cl-callf nconc message
  1019. (mapcar 'flycheck-error-message (flycheck-overlay-errors-at (point)))))
  1020. ;; sly (lazy-loaded)
  1021. (when (featurep 'sly)
  1022. (cl-callf nconc message (mapcar (lambda (note)
  1023. (plist-get note :message))
  1024. (my/sly-notes-at-point))))
  1025. ;; jinx
  1026. (when-let (((bound-and-true-p jinx-mode))
  1027. (jinx-msg (jinx--get-overlays (point) (1+ (point)))))
  1028. (push "misspelled word" message))
  1029. (when message
  1030. (my/floating-tooltip " *my-diagnostic-posframe*"
  1031. (mapconcat (lambda (msg)
  1032. (concat "•" msg))
  1033. message "\n")))))
  1034. (defconst my/consult-flymake-flycheck-narrow
  1035. '((?e . "Error")
  1036. (?w . "Warning")
  1037. (?i . "Info")
  1038. (?n . "Info")))
  1039. (defun my/-consult-replace-flymake-error-level (candidates)
  1040. "Return CANDIDATES with the flymake error level note replaced with info."
  1041. (cl-loop for cand in candidates
  1042. collect
  1043. (cl-loop
  1044. with start = nil
  1045. for i below (length cand)
  1046. for props = (text-properties-at i cand)
  1047. for face = (plist-get props 'face)
  1048. when (eq face 'compilation-info) do
  1049. (setq start (or start i))
  1050. else when start do
  1051. (setf (substring cand start i)
  1052. (propertize (string-pad "info" (- i start))
  1053. 'face (flycheck-error-level-error-list-face
  1054. 'info)))
  1055. (cl-return cand)
  1056. finally return cand)))
  1057. (defun my/consult-flymake-flycheck-candidates (&optional project)
  1058. "Return combined candidate list for flymake and flycheck.
  1059. With PROJECT, return the candiadeets for that project."
  1060. (let ((had-errors))
  1061. (prog1
  1062. (seq-uniq
  1063. (append
  1064. (when-let (((bound-and-true-p flymake-mode))
  1065. (diags (if project (flymake--project-diagnostics
  1066. project)
  1067. (flymake-diagnostics))))
  1068. (setq had-errors t)
  1069. (my/-consult-replace-flymake-error-level
  1070. (consult-flymake--candidates diags)))
  1071. (when (boundp 'flycheck-mode)
  1072. (if project
  1073. (cl-loop for buf in (project-buffers project)
  1074. append
  1075. (with-current-buffer buf
  1076. (when (and flycheck-mode flycheck-current-errors)
  1077. (setq had-errors t)
  1078. (consult-flycheck--candidates))))
  1079. (when (and flycheck-mode flycheck-current-errors)
  1080. (setq had-errors t)
  1081. (consult-flycheck--candidates))))))
  1082. (unless had-errors
  1083. (user-error "No errors (Flymake: %s | Flycheck: %s)"
  1084. (cond
  1085. ((not (bound-and-true-p flymake-mode))
  1086. "not running")
  1087. ((seq-difference (flymake-running-backends)
  1088. (flymake-reporting-backends))
  1089. "running")
  1090. (t "finished"))
  1091. (if (boundp 'flycheck-last-status-change)
  1092. flycheck-last-status-change
  1093. "not running"))))))
  1094. (defun my/consult-flymake-flycheck (&optional project)
  1095. "Jump to flymake or flycheck error.
  1096. With PROJECT, give diagnostics for all buffers in the current project."
  1097. (interactive "P")
  1098. (consult--read
  1099. (consult--with-increased-gc
  1100. (my/consult-flymake-flycheck-candidates
  1101. (and project (project-current))))
  1102. :prompt "Error: "
  1103. :category 'flymake-flycheck-error
  1104. :history t
  1105. :require-match t
  1106. :sort nil
  1107. :narrow (consult--type-narrow my/consult-flymake-flycheck-narrow)
  1108. :group (consult--type-group my/consult-flymake-flycheck-narrow)
  1109. :lookup #'consult--lookup-candidate
  1110. :state (consult--jump-state)))
  1111. (with-eval-after-load 'flymake
  1112. (keymap-set flymake-mode-map "C-c e" 'my/diagnostic-at-point)
  1113. (keymap-set flymake-mode-map "C-c E" 'my/consult-flymake-flycheck))
  1114. (with-eval-after-load 'flycheck
  1115. (keymap-set flycheck-mode-map "C-c e" 'my/diagnostic-at-point)
  1116. (keymap-set flycheck-mode-map "C-c E" 'my/consult-flymake-flycheck))
  1117. (with-eval-after-load 'jinx
  1118. (keymap-set jinx-mode-map "C-c e" 'my/diagnostic-at-point))
  1119. ;; eldoc
  1120. (use-package eldoc
  1121. :diminish eldoc-mode
  1122. :init
  1123. (setq-default eldoc-echo-area-use-multiline-p nil))
  1124. ;; eglot
  1125. (use-package eglot
  1126. :demand t
  1127. :pin gnu ;; try to force Elpa version to fix warnings
  1128. :hook ((eglot-managed-mode . my/-eglot-setup))
  1129. :init
  1130. ;; (defun my/eglot-in-text-mode-only ()
  1131. ;; (when (eq major-mode 'text-mode)
  1132. ;; (trusted-files-eglot-ensure-if-safe)))
  1133. (defvar my/-eglot-documentation-buffer nil
  1134. "Buffer for showing documentation for `my/eglot-documentation-at-point'.")
  1135. (defun my/eglot-documentation-at-point ()
  1136. "Show documentation for a symbol at point."
  1137. (interactive)
  1138. (if-let (server (eglot-current-server))
  1139. (progn
  1140. (if-let* (((not (buffer-live-p my/-eglot-documentation-buffer)))
  1141. (name (generate-new-buffer-name "*eglot documentation*")))
  1142. (setq my/-eglot-documentation-buffer (generate-new-buffer name)))
  1143. (eglot-hover-eldoc-function
  1144. (lambda (info _ _)
  1145. (if-let (((not (seq-empty-p info)))
  1146. (buff (current-buffer)))
  1147. (with-current-buffer my/-eglot-documentation-buffer
  1148. (read-only-mode -1)
  1149. (erase-buffer)
  1150. (insert info)
  1151. (goto-char (point-min))
  1152. (special-mode)
  1153. (read-only-mode 1)
  1154. (when (not (get-buffer-window my/-eglot-documentation-buffer nil))
  1155. (switch-to-buffer-other-window my/-eglot-documentation-buffer t)
  1156. (switch-to-buffer-other-window buff t)))))))))
  1157. (defun my/-eglot-cleanup-doc-buffer (_server &optional _interactive _timeout
  1158. preserve-buffers)
  1159. (when (and (not preserve-buffers)
  1160. (buffer-live-p my/-eglot-documentation-buffer)
  1161. (cl-every (lambda (buffer)
  1162. (with-current-buffer buffer
  1163. (let ((server (eglot-current-server)))
  1164. (or (not (eglot-lsp-server-p server))
  1165. (eglot--shutdown-requested server)))))
  1166. (buffer-list)))
  1167. (kill-buffer my/-eglot-documentation-buffer)))
  1168. (advice-add 'eglot-shutdown :after 'my/-eglot-cleanup-doc-buffer)
  1169. (defun my/-eglot-setup ()
  1170. "Setup eldoc variables for `eglot-managed-mode-hook'."
  1171. (setq-local evil-lookup-func #'my/eglot-documentation-at-point)
  1172. (evil-define-key '(normal motion) 'local
  1173. "K" #'evil-lookup
  1174. "gR" #'eglot-rename
  1175. "gA" #'eglot-code-actions
  1176. "gs" #'consult-eglot-symbols)
  1177. (eglot-inlay-hints-mode -1))
  1178. (setq eglot-autoshutdown t
  1179. eglot-ignored-server-capabilities '(:documentOnTypeFormattingProvider))
  1180. :config
  1181. (add-to-list 'eglot-server-programs
  1182. (cons '(c-mode c-ts-mode c++-mode c++-ts-mode objc-mode)
  1183. '("clangd" "--all-scopes-completion" "--background-index"
  1184. "--clang-tidy" "--completion-style=detailed"
  1185. "--header-insertion=never" "--pch-storage=memory"
  1186. "--function-arg-placeholders"
  1187. "--compile-commands-dir=build"))))
  1188. ;; LTeX (languagetool)
  1189. (require 'ltex-eglot)
  1190. ;; apheleia (code formatter)
  1191. (use-package apheleia
  1192. :defer nil
  1193. :bind ("C-c o" . apheleia-format-buffer)
  1194. :init
  1195. (add-to-list 'auto-mode-alist `(,(rx "/.clang-format" eos) . yaml-ts-mode))
  1196. :config
  1197. (with-eval-after-load 'apheleia
  1198. (setf (alist-get 'java-mode apheleia-mode-alist) 'clang-format
  1199. (alist-get 'java-ts-mode apheleia-mode-alist) 'clang-format))
  1200. (apheleia-global-mode +1))
  1201. ;; gud
  1202. (use-package gud
  1203. :demand t
  1204. :ensure nil
  1205. :after (project evil)
  1206. :bind (:map project-prefix-map
  1207. ("U" . my/project-gdb))
  1208. :config
  1209. (setq gdb-debuginfod-enable-setting t)
  1210. (defvar my/project-gdb-command nil
  1211. "Command to use in `my/project-gdb'.")
  1212. (put 'my/project-gdb-command 'safe-local-variable (lambda (val)
  1213. (stringp val)))
  1214. (defun my/project-gdb (project command-line)
  1215. "Run gdb in the project root"
  1216. (interactive (let* ((project (project-current t))
  1217. (default-directory (project-root project)))
  1218. (list project (gud-query-cmdline 'gdb))))
  1219. (let ((default-directory (project-root project)))
  1220. (gdb command-line)))
  1221. (evil-set-initial-state 'gdb-locals-mode 'motion)
  1222. (evil-collection-inhibit-insert-state 'gdb-locals-mode-map)
  1223. (evil-define-key '(normal motion visual) gdb-locals-mode-map
  1224. (kbd "TAB") (keymap-lookup gdb-locals-mode-map "TAB")
  1225. (kbd "RET") #'gdb-edit-locals-value
  1226. (kbd "<mouse-1>") #'gdb-edit-locals-value
  1227. "q" #'kill-current-buffer)
  1228. (evil-set-initial-state 'gdb-registers-mode 'motion)
  1229. (evil-collection-inhibit-insert-state 'gdb-registers-mode-map)
  1230. (evil-define-key '(normal motion visual) gdb-registers-mode-map
  1231. (kbd "TAB") (keymap-lookup gdb-registers-mode-map "TAB")
  1232. (kbd "RET") #'gdb-edit-register-value
  1233. (kbd "<mouse-1>") #'gdb-edit-register-value
  1234. "q" #'kill-current-buffer
  1235. (kbd "C-c f") #'gdb-registers-toggle-filter
  1236. (kbd "C-c F") (lambda ()
  1237. "Customize the filter for the registers buffer."
  1238. (interactive)
  1239. (customize-option-other-window
  1240. 'gdb-registers-filter-pattern-list)))
  1241. (evil-set-initial-state 'gdb-frames-mode 'motion)
  1242. (evil-collection-inhibit-insert-state 'gdb-frames-mode-map)
  1243. (evil-define-key '(normal motion visual) gdb-frames-mode-map
  1244. "q" #'kill-current-buffer
  1245. (kbd "RET") #'gdb-select-frame)
  1246. (evil-set-initial-state 'gdb-breakpoints-mode 'motion)
  1247. (evil-collection-inhibit-insert-state 'gdb-breakpoints-mode-map)
  1248. (evil-define-key '(normal motion visual) gdb-breakpoints-mode-map
  1249. (kbd "TAB") (keymap-lookup gdb-breakpoints-mode-map "TAB")
  1250. "q" #'gdb-delete-frame-or-window
  1251. "D" #'gdb-delete-breakpoint
  1252. (kbd "RET") #'gdb-goto-breakpoint
  1253. (kbd "<mouse-1>") #'gdb-goto-breakpoint
  1254. (kbd "SPC") #'gdb-toggle-breakpoint)
  1255. (evil-set-initial-state 'gdb-threads-mode 'motion)
  1256. (evil-collection-inhibit-insert-state 'gdb-threads-mode-map)
  1257. (evil-define-key '(normal motion visual) gdb-threads-mode-map
  1258. (kbd "TAB") (keymap-lookup gdb-threads-mode-map "TAB")
  1259. "q" #'gdb-delete-frame-or-window
  1260. "D" #'gdb-frame-disassembly-for-thread
  1261. (kbd "C-c f") #'gdb-display-stack-for-thread
  1262. (kbd "C-c i") #'gdb-interrupt-thread
  1263. (kbd "C-c l") #'gdb-display-locals-for-thread
  1264. (kbd "C-c r") #'gdb-display-registers-for-thread
  1265. (kbd "C-c c") #'gdb-continue-thread
  1266. (kbd "C-c d") #'gdb-display-disassembly-for-thread
  1267. (kbd "C-c s") #'gdb-step-thread
  1268. (kbd "C-c F") #'gdb-frame-stack-for-thread
  1269. (kbd "C-c L") #'gdb-frame-locals-for-thread
  1270. (kbd "C-c R") #'gdb-frame-registers-for-thread
  1271. (kbd "RET") #'gdb-select-thread
  1272. (kbd "<mouse-2>") #'gdb-select-thread))
  1273. ;; dape
  1274. (use-package dape
  1275. :hook ((after-init . dape-breakpoint-load)
  1276. (kill-emacs . dape-breakpoint-save)
  1277. (dape-start . save-some-buffers)
  1278. (dape-display-source . pulse-momentary-highlight-one-line))
  1279. :bind (:map dape-info-parent-mode-map
  1280. ("<tab>" . dape--info-buffer-tab))
  1281. :init
  1282. (setopt dape-default-breakpoints-file (no-littering-expand-var-file-name
  1283. "dape-breakpoints"))
  1284. :config
  1285. (setopt dape-buffer-window-arrangement 'right)
  1286. (dape-breakpoint-global-mode 1))
  1287. ;; dumb-jump
  1288. (use-package dumb-jump
  1289. :init
  1290. (add-hook 'xref-backend-functions #'dumb-jump-xref-activate))
  1291. ;; yasnippet
  1292. (use-package yasnippet
  1293. :demand t
  1294. :bind ("C-c s" . yas-expand)
  1295. :config
  1296. (yas-global-mode 1))
  1297. ;; project.el
  1298. (use-package project
  1299. :bind (([remap project-compile] . my/project-compile-or-default)
  1300. :map project-prefix-map
  1301. ("s" . my/project-eshell)
  1302. ("u" . my/project-run))
  1303. :init
  1304. (defvar eshell-buffer-name)
  1305. (defun my/project-eshell (prompt &optional arg)
  1306. "Switch to or create an eshell buffer in the current projects root."
  1307. (interactive (list t current-prefix-arg))
  1308. (if-let ((proj (project-current prompt))
  1309. (default-directory (project-root proj))
  1310. (eshell-buffer-name
  1311. (concat "*eshell for project " default-directory "*")))
  1312. (eshell arg)))
  1313. (defun my/project-eshell-or-default (&optional arg)
  1314. "Open an eshell for the current project, otherwise, open a normal eshell."
  1315. (interactive "P")
  1316. (unless (my/project-eshell nil arg)
  1317. (eshell arg)))
  1318. (defun my/project-compile-or-default ()
  1319. "If in a project, run `project-compile', otherwise run `compile'."
  1320. (interactive)
  1321. (if (project-current)
  1322. (call-interactively 'project-compile)
  1323. (call-interactively 'compile)))
  1324. (defvar my/project-run-command nil
  1325. "Command to run with `my/project-run'.")
  1326. (put 'my/project-run-command 'safe-local-variable (lambda (val)
  1327. (stringp val)))
  1328. (defvar my/project-run-dir nil
  1329. "Directory to run project in with `my/project-run'.")
  1330. (put 'my/project-run-dir 'safe-local-variable (lambda (val)
  1331. (stringp val)))
  1332. (defvar my/-project-run-history '()
  1333. "Commands previously run with `my/project-run'")
  1334. (defvar my/project-root-marker ".project-root"
  1335. "Marker file to look for in non-vc backed projects.")
  1336. (defun my/project-get-root-dir ()
  1337. "Get the root dir for the current project"
  1338. (let* ((proj (project-current nil))
  1339. (default-directory (if proj
  1340. (project-root proj)
  1341. default-directory)))
  1342. (if my/project-run-dir
  1343. (expand-file-name my/project-run-dir)
  1344. default-directory)))
  1345. (defvar my/compile-use-comint t
  1346. "Weather or not to use comint by default for compile buffers.")
  1347. (defun my/-compile-use-comint-by-default (args)
  1348. (if my/compile-use-comint
  1349. (list (car args) t)
  1350. args))
  1351. (advice-add 'compile :filter-args 'my/-compile-use-comint-by-default)
  1352. (defun my/project-run (command comint)
  1353. "Like `project-compile', but for running a project.
  1354. COMMAND and COMINT are like `compile'."
  1355. (interactive
  1356. (list
  1357. (let ((default-directory (my/project-get-root-dir)))
  1358. (read-shell-command "Run Command: "
  1359. (or (car my/-project-run-history)
  1360. my/project-run-command)
  1361. (if (and my/project-run-command
  1362. (equal my/project-run-command
  1363. (car-safe my/-project-run-history)))
  1364. '(my/-project-run-history . 1)
  1365. 'my/-project-run-history)))
  1366. (consp current-prefix-arg)))
  1367. (let* ((default-directory (my/project-get-root-dir))
  1368. (compilation-buffer-name-function (lambda (_)
  1369. (progn "*run project*")))
  1370. (compilation-directory default-directory)
  1371. (compile-history nil)
  1372. (compile-command nil))
  1373. (compile command comint)
  1374. (when (not my/project-run-command)
  1375. (setq my/project-run-command command))))
  1376. :config
  1377. (defun my/project-try-dotfile (dir)
  1378. (if-let (root (locate-dominating-file dir my/project-root-marker))
  1379. (list 'vc nil root)))
  1380. (add-hook 'project-find-functions #'my/project-try-dotfile))
  1381. ;; comint
  1382. (use-package comint
  1383. :ensure nil
  1384. :after evil
  1385. :config
  1386. (evil-set-initial-state 'comint-mode 'normal))
  1387. ;; nxml
  1388. (use-package nxml-mode
  1389. :ensure nil
  1390. :hook (nxml-mode . my/-nxml-setup)
  1391. :init
  1392. (defun my/-nxml-setup ()
  1393. "Setup `nxml-mode'."
  1394. (sgml-electric-tag-pair-mode 1)
  1395. (setq-local completion-at-point-functions
  1396. '(rng-completion-at-point cape-file)))
  1397. (add-to-list 'auto-mode-alist
  1398. `(,(concat
  1399. (regexp-opt '("gschema" "gresource" "ui")) "\\'")
  1400. . nxml-mode)))
  1401. ;; devdocs
  1402. (use-package devdocs
  1403. :bind (("C-h D" . devdocs-lookup)))
  1404. ;; Bibtex (built in)
  1405. (require 'bibtex)
  1406. ;; Better URL highlighting and matching
  1407. (dolist (field '("howpublished" "url"))
  1408. (add-to-list 'bibtex-generate-url-list
  1409. `((,field . ,(rx "http" (? "s") "://" (+ (not "}"))))
  1410. "%s"
  1411. (,field
  1412. ,(rx (? (or "\\url{" "{")) (group "http" (? "s") "://"
  1413. (+ (not "}")))
  1414. (? "}"))
  1415. ,(lambda (field)
  1416. (match-string 1 field))))))
  1417. (defun my/bibtex-in-entry-p (&optional exclude-braces)
  1418. "Return t is point is inside a BibTeX entry.
  1419. When EXCLUDE-BRACES is non-nil, don't count the first and last brace of the
  1420. entry as in the entry. That is, if the point is on the first { or last } of the
  1421. entry, return nil."
  1422. (save-excursion
  1423. (when (and exclude-braces (eq ?\} (char-after)))
  1424. (forward-char))
  1425. ;; go to top level and check if the character at point is {
  1426. (let ((start-pos (point))
  1427. (last-valid (point)))
  1428. (condition-case nil
  1429. (while t
  1430. (backward-up-list 1 t t)
  1431. (setq last-valid (point)))
  1432. (error
  1433. (and
  1434. (eq ?\{ (char-after last-valid))
  1435. (or (not exclude-braces)
  1436. (not (= start-pos last-valid)))))))))
  1437. (defvar my/bibtex-indent-width 4
  1438. "Width to indent for `my/bibtex-calculate-indentation'.")
  1439. (defun my/bibtex-calculate-indentation ()
  1440. "Calculate the column to indent to on the current line."
  1441. (save-excursion
  1442. (back-to-indentation)
  1443. (if (my/bibtex-in-entry-p t)
  1444. my/bibtex-indent-width
  1445. 0)))
  1446. (defun my/bibtex-empty-line-p ()
  1447. "Return t if the current line is only blank characters."
  1448. (save-excursion
  1449. (beginning-of-line)
  1450. (looking-at (rx (* blank) eol))))
  1451. (defun my/bibtex-indent-line ()
  1452. "Indent the current line."
  1453. (interactive)
  1454. (save-excursion
  1455. (beginning-of-line)
  1456. (when (looking-at (rx (+ blank)))
  1457. (delete-region (point) (match-end 0)))
  1458. (indent-to (my/bibtex-calculate-indentation)))
  1459. (when (looking-at (rx (+ blank) eol))
  1460. (end-of-line)))
  1461. (defun my/bibtex-indent-or-find-text ()
  1462. "Either indent the current line or jump to the current fields text.
  1463. If the current line is only whitespace call `my/bibtex-calculate-indentation',
  1464. otherwise, call `bibtex-find-text'."
  1465. (interactive)
  1466. (if (my/bibtex-empty-line-p)
  1467. (my/bibtex-indent-line)
  1468. (bibtex-find-text)))
  1469. (defun my/bibtex-indent-or-find-text-and-insert ()
  1470. "Like `my/bibtex-indent-or-find-text', but enter insert mode after."
  1471. (interactive)
  1472. (my/bibtex-indent-or-find-text)
  1473. (if (my/bibtex-empty-line-p)
  1474. (evil-append 1)
  1475. (evil-insert 1)))
  1476. (defun my/-bibtex-setup-indent ()
  1477. "Set up `bibtex-mode' indentation stuff."
  1478. (setq-local indent-line-function 'my/bibtex-indent-line
  1479. electric-indent-chars '(?\n ?\{ ?\} ?,)))
  1480. (defun my/-bibtex-fix-fill-prefix ()
  1481. "`bivtex-mode' has a bad habbit of messing up `fill-prefix'."
  1482. (when (eq major-mode 'bibtex-mode)
  1483. (setq-local fill-prefix nil)))
  1484. (advice-add 'bibtex-mode :after 'my/-bibtex-fix-fill-prefix)
  1485. (add-hook 'bibtex-mode-hook 'my/-bibtex-setup-indent)
  1486. (keymap-set bibtex-mode-map "RET" 'newline-and-indent)
  1487. (keymap-set bibtex-mode-map "TAB" 'my/bibtex-indent-or-find-text)
  1488. (evil-define-key 'normal bibtex-mode-map
  1489. (kbd "TAB") 'my/bibtex-indent-or-find-text-and-insert)
  1490. ;; Latex help (from elisp file)
  1491. (require 'latex-help)
  1492. ;; AUCTeX
  1493. (use-package auctex
  1494. :hook ((LaTeX-mode . turn-on-reftex)
  1495. (LaTeX-mode . LaTeX-math-mode)
  1496. (LaTeX-mode . my/-setup-LaTeX-mode)
  1497. (LaTeX-mode . trusted-files-flycheck-mode-if-safe))
  1498. :bind (:map TeX-mode-map
  1499. ("C-c ?" . latex-help))
  1500. :init
  1501. (add-to-list 'major-mode-remap-alist '(plain-tex-mode . plain-TeX-mode))
  1502. (add-to-list 'major-mode-remap-alist '(latex-mode . LaTeX-mode))
  1503. (add-to-list 'major-mode-remap-alist '(ams-tex-mode . AmSTeX-mode))
  1504. (add-to-list 'major-mode-remap-alist '(context-mode . ConTeXt-mode))
  1505. (add-to-list 'major-mode-remap-alist '(texinfo-mode . Texinfo-mode))
  1506. (add-to-list 'major-mode-remap-alist '(doctex-mode . docTeX-mode))
  1507. (add-to-list 'auto-mode-alist '("/\\.latexmkrc\\'" . perl-mode))
  1508. (add-to-list 'auto-mode-alist '("\\.[tT]e[xX]\\'" . LaTeX-mode))
  1509. :config
  1510. (defun my/-auctex-texdoc-setup-env (oldfun &rest args)
  1511. (let ((process-environment process-environment)
  1512. (emacs-cmd (concat "emacsclient" (and (not (display-graphic-p)) " -nw"))))
  1513. (setenv "PDFVIEWER_texdoc" "evince")
  1514. (setenv "MDVIEWER_texdoc" emacs-cmd)
  1515. (setenv "PAGER_texdoc" emacs-cmd)
  1516. (apply oldfun args)))
  1517. (advice-add 'TeX-documentation-texdoc :around 'my/-auctex-texdoc-setup-env)
  1518. (defun my/-setup-LaTeX-mode ()
  1519. (setq evil-lookup-func 'latex-help-at-point))
  1520. (setq TeX-auto-save t
  1521. TeX-parse-self t
  1522. reftex-plug-into-AUCTeX t)
  1523. (evil-define-operator my/evil-LaTeX-fill (beg end)
  1524. "Like `evil-fill', but using auctex."
  1525. ;; The code here came straight from `evil-fill'
  1526. :move-point nil
  1527. :type line
  1528. (save-excursion
  1529. (ignore-errors (LaTeX-fill-region beg end))))
  1530. (evil-define-operator my/evil-LaTeX-fill-and-move (beg end)
  1531. "Like `evil-fill-and-move', but using auctex."
  1532. ;; The code here came straight from `evil-fill-and-move'
  1533. :move-point nil
  1534. :type line
  1535. (let ((marker (make-marker)))
  1536. (move-marker marker (1- end))
  1537. (ignore-errors
  1538. (LaTeX-fill-region beg end)
  1539. (goto-char marker)
  1540. (evil-first-non-blank))))
  1541. (evil-define-key 'normal TeX-mode-map
  1542. "gq" 'my/evil-LaTeX-fill-and-move
  1543. "gw" 'my/evil-LaTeX-fill)
  1544. (setq-default TeX-master nil)
  1545. (require 'tex)
  1546. (TeX-global-PDF-mode 1))
  1547. ;; blueprint
  1548. (use-package blueprint-ts-mode
  1549. :hook (blueprint-ts-mode . trusted-files-eglot-ensure-if-safe)
  1550. :after eglot)
  1551. ;; python-ts-mode
  1552. (use-package python-ts-mode
  1553. :ensure nil
  1554. :hook (python-ts-mode . trusted-files-eglot-ensure-if-safe))
  1555. ;; java-ts-mode
  1556. (use-package java-ts-mode
  1557. :hook ((java-ts-mode . trusted-files-eglot-ensure-if-safe)
  1558. (java-ts-mode . my/-setup-java-ts-mode))
  1559. :config
  1560. (defun my/-setup-java-ts-mode ()
  1561. (let ((rules (car treesit-simple-indent-rules)))
  1562. (setcdr rules
  1563. (cons '((and (parent-is "array_initializer")
  1564. (node-is "array_initializer"))
  1565. parent-bol java-ts-mode-indent-offset)
  1566. (nthcdr 1 rules))))))
  1567. ;; c-ts-mode
  1568. (use-package c-ts-mode
  1569. :after evil
  1570. :hook ((c-ts-mode c++-ts-mode) . trusted-files-eglot-ensure-if-safe)
  1571. :init
  1572. (setq-default c-ts-mode-indent-offset 4)
  1573. :config
  1574. (evil-define-key 'normal 'c-ts-mode-map
  1575. "go" #'ff-find-other-file
  1576. "gO" #'ff-find-other-file-other-window)
  1577. (evil-define-key 'normal 'c++-ts-mode-map
  1578. "go" #'ff-find-other-file
  1579. "gO" #'ff-find-other-file-other-window)
  1580. (evil-define-key 'normal 'objc-mode-map
  1581. "go" #'ff-find-other-file
  1582. "gO" #'ff-find-other-file-other-window))
  1583. ;; GLSL
  1584. (use-package glsl-mode)
  1585. ;; php-mode
  1586. (use-package php-mode
  1587. :hook (php-mode . trusted-files-eglot-ensure-if-safe))
  1588. ;; web-mode
  1589. (use-package web-mode
  1590. :hook (web-mode . trusted-files-eglot-ensure-if-safe)
  1591. :init
  1592. (add-to-list 'eglot-server-programs
  1593. '(web-mode . ("vscode-html-language-server" "--stdio"))))
  1594. ;; JavaScript
  1595. (use-package js
  1596. :ensure nil
  1597. :hook (js-ts-mode . trusted-files-eglot-ensure-if-safe))
  1598. ;; TypeScript
  1599. (use-package typescript-ts-mode
  1600. :ensure nil
  1601. :hook (typescript-ts-mode . trusted-files-eglot-ensure-if-safe)
  1602. :init
  1603. (add-to-list 'auto-mode-alist `(,(rx ".ts" eos) . typescript-ts-mode)))
  1604. ;; Polymode
  1605. (use-package polymode
  1606. :config
  1607. (define-hostmode my/poly-web-hostmode
  1608. :mode 'web-mode)
  1609. (define-innermode my/poly-php-innermode
  1610. :mode 'php-mode
  1611. :head-matcher (regexp-quote "<?php")
  1612. :tail-matcher (regexp-quote "?>")
  1613. :head-mode 'body
  1614. :tail-mode 'body)
  1615. (define-polymode my/poly-web-mode
  1616. :hostmode 'my/poly-web-hostmode
  1617. :innermodes '(my/poly-php-innermode))
  1618. (add-to-list 'auto-mode-alist '("\\.php\\|\\.phtml\\'" . my/poly-web-mode)))
  1619. ;; shell-mode
  1620. (use-package sh-script
  1621. :ensure nil
  1622. :hook (sh-mode . my/-setup-sh-mode)
  1623. :init
  1624. (defun my/-setup-sh-mode ()
  1625. (add-to-list 'completion-at-point-functions #'cape-file)))
  1626. ;; go mode
  1627. (use-package go-mode
  1628. :defer nil
  1629. :hook (go-mode . trusted-files-eglot-ensure-if-safe))
  1630. (use-package go-ts-mode
  1631. :ensure nil
  1632. :hook (go-ts-mode . trusted-files-eglot-ensure-if-safe))
  1633. ;; rust
  1634. (use-package rust-mode)
  1635. (use-package rust-ts-mode
  1636. :ensure nil
  1637. :hook (rust-ts-mode . trusted-files-eglot-ensure-if-safe))
  1638. ;; zig
  1639. (use-package zig-mode
  1640. :hook (zig-mode . trusted-files-eglot-ensure-if-safe))
  1641. ;; lua
  1642. (use-package lua-mode
  1643. :hook (lua-mode . trusted-files-eglot-ensure-if-safe))
  1644. ;; markdown
  1645. (use-package markdown-mode
  1646. :hook (markdown-mode . auto-fill-mode))
  1647. ;; groovy
  1648. (use-package groovy-mode)
  1649. ;; cmake
  1650. (require 'cmake-mode)
  1651. (require 'cmake-ts-mode)
  1652. (with-eval-after-load 'cmake-mode
  1653. (setq cmake-ts-mode-indent-offset tab-width))
  1654. ;; kdl
  1655. (require 'kdl-ts-mode)
  1656. ;; json
  1657. (use-package json-ts-mode
  1658. :hook (json-ts-mode . trusted-files-eglot-ensure-if-safe))
  1659. (use-package json-mode)
  1660. ;; csv
  1661. (use-package csv-mode)
  1662. ;; firejail
  1663. (require 'firejail-mode)
  1664. ;; yaml
  1665. (use-package yaml-ts-mode
  1666. :hook ((yaml-ts-mode . trusted-files-eglot-ensure-if-safe)
  1667. (yaml-ts-mode . my/-setup-yaml-ts-mode))
  1668. :init
  1669. (defun my/-setup-yaml-ts-mode ()
  1670. (setq indent-line-function #'yaml-indent-line)))
  1671. (use-package yaml-mode)
  1672. ;; yuck (config language for eww)
  1673. (use-package yuck-mode)
  1674. ;; Some Elisp indentation stuff
  1675. ;; Source: https://github.com/magit/emacsql
  1676. ;; emacsql.el line 394
  1677. (defun my/lisp-inside-plist-p ()
  1678. "Return t if point is inside a plist."
  1679. (save-excursion
  1680. (let ((start (point)))
  1681. (beginning-of-defun)
  1682. (when-let ((sexp (nth 1 (parse-partial-sexp (point) start))))
  1683. (goto-char sexp)
  1684. (looking-at (rx "(" (* (syntax whitespace)) ":"))))))
  1685. (defun my/-calculate-indent-fix-plists (oldfun &rest args)
  1686. "This function is meant to advise `calculate-lisp-indent'.
  1687. It calls OLDFUN with ARGS in such an environment as to prevent the default
  1688. indentation of plists."
  1689. (if (and (eq major-mode 'emacs-lisp-mode)
  1690. (save-excursion
  1691. (beginning-of-line)
  1692. (my/lisp-inside-plist-p)))
  1693. (let ((lisp-indent-offset 1))
  1694. (apply oldfun args))
  1695. (apply oldfun args)))
  1696. (advice-add 'calculate-lisp-indent :around
  1697. 'my/-calculate-indent-fix-plists)
  1698. (defvar my/max-lisp-noindent-comment-search-lines 30
  1699. "Max lines to search for the noindent comment.")
  1700. (defun my/-calculate-lisp-indent-noindent-comment (oldfun &rest args)
  1701. "This function is meant to advise `calculate-lisp-indent'.
  1702. It calls OLDFUN with ARGS, unless the line ends with the comment
  1703. ; noindent [LINES]
  1704. In this case, it just returns the current amount of indentation. LINES is the
  1705. number of lines that this comment affects. This is limited by
  1706. `my/max-lisp-noindent-comment-search-lines'.
  1707. This only works if its on the first or second form in a block. I think this is
  1708. because the indentation code only checks those and then assumes the same
  1709. indentation for every following line in the same block. This is probably OK as
  1710. I can't imagine too many instances where you need to randomly change the indent
  1711. midway through a block, and in those cases you can just stick this on the first
  1712. line in the block and manually deal with indentation."
  1713. (if (and (save-excursion
  1714. (end-of-line)
  1715. (re-search-backward
  1716. (rx (+ ";") (syntax whitespace) "noindent"
  1717. (? (syntax whitespace) (group (+ num)))
  1718. line-end)
  1719. (pos-bol (- my/max-lisp-noindent-comment-search-lines))
  1720. t))
  1721. (save-excursion
  1722. ;; if we are on a blank line, move forward a line
  1723. (when (zerop (length (buffer-substring-no-properties
  1724. (pos-bol) (pos-eol))))
  1725. (beginning-of-line 2))
  1726. (<= (count-lines (match-beginning 0) (pos-eol))
  1727. (if-let ((match (match-string 1)))
  1728. (string-to-number match)
  1729. 1))))
  1730. (save-excursion
  1731. (beginning-of-line)
  1732. (looking-at (rx (* blank)))
  1733. (length (match-string 0)))
  1734. (apply oldfun args)))
  1735. (advice-add 'calculate-lisp-indent :around
  1736. 'my/-calculate-lisp-indent-noindent-comment)
  1737. ;; sly
  1738. (use-package sly
  1739. ;; :hook (lisp-mode . my/-lisp-mode-autoconnect-sly)
  1740. :bind (:map sly-mode-map
  1741. ("C-c e" . my/diagnostic-at-point))
  1742. :autoload sly-connected-p
  1743. :init
  1744. (defun my/-lisp-mode-autoconnect-sly ()
  1745. (unless (sly-connected-p)
  1746. (trusted-files-sly-if-safe)))
  1747. (setq inferior-lisp-program "/usr/bin/sbcl")
  1748. (defun my/-sly-fix-special-buffers ()
  1749. (when (string-match-p (rx bos "*" (* any) "*" eos) (buffer-name))
  1750. (setq-local show-trailing-whitespace nil)))
  1751. (add-hook 'lisp-mode-hook 'my/-sly-fix-special-buffers)
  1752. :config
  1753. (evil-define-key 'insert sly-mrepl-mode-map
  1754. (kbd ",") 'self-insert-command)
  1755. (evil-define-key nil sly-mrepl-mode-map
  1756. (kbd "C-c ,") 'sly-mrepl-shortcut)
  1757. (sly-symbol-completion-mode -1)
  1758. (setq common-lisp-hyperspec-root
  1759. (concat "file://" (expand-file-name "~/src/clhs/HyperSpec/")))
  1760. (defun my/-hyperspec-loopup-in-eww (oldfun &rest r)
  1761. (let ((browse-url-browser-function #'eww-browse-url))
  1762. (apply oldfun r)))
  1763. (advice-add 'hyperspec-lookup :around #'my/-hyperspec-loopup-in-eww))
  1764. ;; pdf-tools
  1765. (use-package pdf-tools
  1766. :hook (pdf-view-mode . my/setup-pdf-view-mode)
  1767. :init
  1768. (setq pdf-misc-print-program-executable "lp")
  1769. (defun my/setup-pdf-view-mode ()
  1770. (display-line-numbers-mode -1)
  1771. (evil-define-key '(motion normal visual) 'local
  1772. (kbd "C-s") #'isearch-forward
  1773. (kbd "C-r") #'isearch-backward)
  1774. (setq-local cursor-type nil))
  1775. (pdf-tools-install))
  1776. ;; doc view
  1777. (use-package doc-view
  1778. :ensure nil
  1779. :hook (doc-view-mode . my/-setup-doc-view-mode)
  1780. :init
  1781. (defun my/-setup-doc-view-mode ()
  1782. (display-line-numbers-mode -1)
  1783. (evil-define-key '(motion normal visual) 'local
  1784. (kbd "C-s") #'isearch-forward
  1785. (kbd "C-r") #'isearch-backward)))
  1786. ;; calc
  1787. (use-package calc
  1788. :ensure nil
  1789. :bind (("C-c m" . quick-calc)
  1790. :map calc-mode-map
  1791. ("M-<tab>" . calc-roll-up)
  1792. ("M-TAB" . calc-roll-up))
  1793. :hook ((calc-mode calc-trail-mode) . my/setup-calc-calc-trail-mode)
  1794. :init
  1795. (defun my/setup-calc-calc-trail-mode ()
  1796. (setq-local doom-modeline-percent-position '()
  1797. truncate-partial-width-windows nil)
  1798. (visual-line-mode -1)
  1799. (display-line-numbers-mode -1)
  1800. (toggle-truncate-lines 1))
  1801. :config
  1802. (evil-define-key '(normal visual motion) calc-edit-mode-map
  1803. (kbd "RET") 'calc-edit-return
  1804. (kbd "<return>") 'calc-edit-return)
  1805. (defun my/-calc-float-mode-string ()
  1806. (cl-destructuring-bind (mode prec) calc-float-format
  1807. (concat
  1808. (upcase-initials (symbol-name mode))
  1809. (unless (zerop prec)
  1810. (concat ": " (number-to-string prec))))))
  1811. (doom-modeline-def-segment calc
  1812. "Display calculator icons and info."
  1813. (concat
  1814. (doom-modeline-spc)
  1815. (when-let ((icon (doom-modeline-icon 'faicon "nf-fa-calculator" "🖩" "")))
  1816. (concat
  1817. (doom-modeline-display-icon icon)
  1818. (doom-modeline-vspc)))
  1819. (doom-modeline--buffer-simple-name)
  1820. (when (eq major-mode 'calc-mode)
  1821. (concat
  1822. (doom-modeline-spc)
  1823. (number-to-string calc-internal-prec)
  1824. (doom-modeline-spc)
  1825. (upcase-initials (symbol-name calc-angle-mode))
  1826. (doom-modeline-spc)
  1827. (my/-calc-float-mode-string)
  1828. (when calc-prefer-frac
  1829. (concat
  1830. (doom-modeline-spc)
  1831. "Frac"))
  1832. (cond
  1833. (calc-algebraic-mode
  1834. (concat
  1835. (doom-modeline-spc)
  1836. "Alg"))
  1837. (calc-incomplete-algebraic-mode
  1838. (concat
  1839. (doom-modeline-spc)
  1840. "IAlg"))))))))
  1841. ;; sage (for when calc is not enough)
  1842. (use-package sage-shell-mode
  1843. :demand
  1844. :bind ("C-c g" . my/run-sage)
  1845. :hook (sage-shell-mode . my/-setup-sage-shell-mode)
  1846. :init
  1847. (defun my/-setup-sage-shell-mode ()
  1848. (setq-local comint-dynamic-complete-functions
  1849. '(comint-c-a-p-replace-by-expanded-history)))
  1850. :config
  1851. (defun my/run-sage (p)
  1852. "Like `sage-shell:run-sage', but does not ask anything without a prefix
  1853. argument."
  1854. (interactive "P")
  1855. (let ((sage-shell:ask-command-options p))
  1856. (funcall-interactively #'sage-shell:run-sage
  1857. (sage-shell:read-command)))))
  1858. ;; fricas (because I like calculators)
  1859. (add-to-list 'load-path "/usr/lib/fricas/emacs")
  1860. (use-package fricas
  1861. :ensure nil
  1862. :custom
  1863. (fricas-run-command "fricas -nosman")
  1864. :init
  1865. ;; Fix `fricas-mode' messing up `completion-at-point-functions'
  1866. (advice-add #'fricas-mode :around
  1867. #'(lambda (oldfun &rest r)
  1868. (let ((temp-capfs))
  1869. (let ((completion-at-point-functions '(t)))
  1870. (apply oldfun r)
  1871. (setq temp-capfs completion-at-point-functions))
  1872. (setq-local completion-at-point-functions temp-capfs)))
  1873. '((name . "my/-fricas-fix-capfs")))
  1874. :config
  1875. (face-spec-set 'fricas-type-time '((t (:foreground unspecified
  1876. :background unspecified
  1877. :inherit font-lock-type-face))))
  1878. (face-spec-set 'fricas-message '((t (:foreground unspecified
  1879. :background unspecified
  1880. :inherit error))))
  1881. (face-spec-set 'fricas-undefined '((t (:foreground unspecified
  1882. :background unspecified
  1883. :inherit nerd-icons-lblue))))
  1884. (face-spec-set 'fricas-algebra '((t (:foreground unspecified
  1885. :background unspecified
  1886. :weight bold
  1887. :inherit fricas-prompt))))
  1888. (face-spec-set 'fricas-TeX '((t (:foreground "black"
  1889. :background "white"
  1890. :inherit fricas-prompt)))))
  1891. ;; gnuplot (mostly for org-plot)
  1892. (use-package gnuplot)
  1893. ;; eat
  1894. (use-package eat
  1895. :bind (("C-c V" . my/project-eat-or-default)
  1896. :map eat-mode-map
  1897. ("M-o" . ace-window)
  1898. :map eat-semi-char-mode-map
  1899. ("M-o" . ace-window)
  1900. :map eat-eshell-emacs-mode-map
  1901. ("M-o" . ace-window)
  1902. :map eat-eshell-semi-char-mode-map
  1903. ("M-o" . ace-window))
  1904. :config
  1905. (defvar my/project-eat-hash-table (make-hash-table :test 'equal)
  1906. "Hash table that maps project root dirs to eat buffers.")
  1907. (defun my/-eat-shell-for-cwd ()
  1908. "Return a good shell for CWD, or nil if the default shell should be used."
  1909. (when (file-remote-p default-directory)
  1910. "/bin/sh"))
  1911. (defun my/project-eat (prompt &optional arg)
  1912. "Switch to or create a eat buffer in the current projects root."
  1913. (interactive (list t current-prefix-arg))
  1914. (if-let ((proj (project-current prompt))
  1915. (default-directory (project-root proj)))
  1916. (if-let ((eat-buff (gethash default-directory
  1917. my/project-eat-hash-table))
  1918. ((buffer-live-p eat-buff)))
  1919. (switch-to-buffer eat-buff)
  1920. (let ((eat-buffer-name (concat "*eat for project " default-directory
  1921. "*"))
  1922. (eat-term-name (if (file-remote-p default-directory)
  1923. "xterm-256color"
  1924. eat-term-name)))
  1925. (puthash default-directory
  1926. (eat (my/-eat-shell-for-cwd) arg)
  1927. my/project-eat-hash-table)))))
  1928. (defun my/project-eat-or-default (&optional arg)
  1929. "Open an eat for the current project, otherwise, open a normal eat."
  1930. (interactive "P")
  1931. (unless (my/project-eat nil)
  1932. (if-let ((eat-buff (gethash nil my/project-eat-hash-table))
  1933. ((buffer-live-p eat-buff)))
  1934. (switch-to-buffer eat-buff)
  1935. (puthash nil (let ((eat-term-name (if (file-remote-p default-directory)
  1936. "xterm-256color"
  1937. eat-term-name)))
  1938. (eat (my/-eat-shell-for-cwd) arg))
  1939. my/project-eat-hash-table)))))
  1940. ;; eshell stuff
  1941. (use-package eshell
  1942. :ensure nil
  1943. :defer nil
  1944. :hook ((eshell-load . eat-eshell-visual-command-mode)
  1945. (eshell-load . eat-eshell-mode)
  1946. (eshell-mode . my/-eshell-mode-setup)
  1947. (eshell-directory-change . my/-eshell-maybe-setup-remote-aliases))
  1948. :bind (:map eshell-mode-map
  1949. ("TAB" . completion-at-point)
  1950. ("<tab>" . completion-at-point))
  1951. :init
  1952. (defun my/-eshell-filter-alias-list ()
  1953. (cl-remove-if-not (lambda (elt)
  1954. (or (string-match-p
  1955. (rx bos
  1956. (or "clear" "find-file"
  1957. "ls" "la" "git"
  1958. (and "eshell/" (+ (not " "))))
  1959. (or " " eos))
  1960. (cl-second elt))))
  1961. eshell-command-aliases-list))
  1962. (defun my/-eshell-maybe-setup-remote-aliases ()
  1963. (if (file-remote-p default-directory)
  1964. (setq-local eshell-command-aliases-list (my/-eshell-filter-alias-list))
  1965. (kill-local-variable 'eshell-command-aliases-list)))
  1966. (defun my/-eshell-mode-setup ()
  1967. "Setup function run from `eshell-mode-hook'"
  1968. (setq-local corfu-auto nil)
  1969. (my/-eshell-maybe-setup-remote-aliases))
  1970. (setq-default eshell-command-aliases-list
  1971. '(("clear" "clear t")
  1972. ("e" "find-file $1")
  1973. ("n" "find-file $1")
  1974. ("emacs" "find-file $1")
  1975. ("nvim" "find-file $1")
  1976. ("ls" "eza --git -F $*")
  1977. ("la" "ls -a $*")
  1978. ("l" "ls -l $*")
  1979. ("ll" "la -l $*")
  1980. ("gt" "git status $*")
  1981. ("gp" "git push $*")
  1982. ("gu" "git pull $*")
  1983. ("gf" "git fetch $*")
  1984. ("ga" "git add $*")
  1985. ("gcm" "git commit -m ${string-join $* \" \"}")
  1986. ("ldg" "ledger -f \"$HOME/docs/finance/finances.ledger\" $*")
  1987. ("tp" "trash-put $*")
  1988. ("trr" "trash-restore $*")
  1989. ("tre" "trash-empty $*")
  1990. ("tre" "trash-empty $*")
  1991. ("trm" "trash-rm $*")
  1992. ("rm" "echo 'rm: I''m unsafe! Don''t use me.'; false")
  1993. ("\\rm" "eshell/rm")))
  1994. (defvar my/eshell-bm-auto-ls t
  1995. "Weather or not to run ls after `eshell/bm'")
  1996. (defun eshell/bm (&optional name)
  1997. "Change to directory of bookmark NAME.
  1998. If no name is given, list all bookmarks instead."
  1999. (if name
  2000. (progn
  2001. (eshell/cd (bookmark-get-filename name))
  2002. (when my/eshell-bm-auto-ls
  2003. (eshell/ls)))
  2004. (eshell-print (string-join (bookmark-all-names) " ")))))
  2005. (use-package esh-help
  2006. :hook (eshell-mode . my/-setup-eshell-help-func)
  2007. :init
  2008. (defun my/-setup-eshell-help-func ()
  2009. (eldoc-mode 1)
  2010. (setq-local evil-lookup-func #'esh-help-run-help))
  2011. (setup-esh-help-eldoc))
  2012. (use-package eshell-syntax-highlighting
  2013. :init
  2014. (eshell-syntax-highlighting-global-mode 1))
  2015. (use-package eshell-starship
  2016. :ensure nil
  2017. :demand t
  2018. :hook (eshell-prompt-mode . eshell-starship-prompt-mode)
  2019. :config
  2020. (eshell-starship-setup-evil-keybindings)
  2021. (set-face-attribute 'eshell-starship-icon-face nil
  2022. :family "FiraCode Nerd Font"))
  2023. (defun my/open-shell-dwim (&optional arg)
  2024. "Open either an Eshell or eat terminal based on `default-directory'.
  2025. If `default-directory' is remote, call `my/project-eat-or-default'. Otherwise,
  2026. call `my/project-eshell-or-default'. ARG is the same as for either of the above
  2027. functions (only eshell uses it at the time of writing)."
  2028. (interactive "P")
  2029. (if (file-remote-p default-directory)
  2030. (my/project-eat-or-default)
  2031. (my/project-eshell-or-default arg)))
  2032. (keymap-global-set "C-c v" #'my/open-shell-dwim)
  2033. ;; proced
  2034. (use-package proced
  2035. :bind ("C-x j" . proced)
  2036. :init
  2037. (evil-define-key '(motion visual normal) proced-mode-map
  2038. "u" 'proced-unmark)
  2039. (setq proced-auto-update-flag t
  2040. proced-auto-update-interval 1)
  2041. (defun my/-setup-proced-mode ()
  2042. (visual-line-mode -1)
  2043. (setq-local truncate-lines t))
  2044. (add-hook 'proced-mode-hook 'my/-setup-proced-mode))
  2045. ;; dired
  2046. (use-package dired
  2047. :ensure nil
  2048. :custom
  2049. (dired-listing-switches
  2050. "-l --almost-all --human-readable --group-directories-first --no-group")
  2051. (dired-hide-details-hide-symlink-targets nil)
  2052. :init
  2053. (setq-default dired-kill-when-opening-new-dired-buffer t)
  2054. (setq delete-by-moving-to-trash t
  2055. dired-recursive-copies 'always
  2056. dired-recursive-deletes 'always
  2057. dired-dwim-target t
  2058. dired-create-destination-dirs 'ask
  2059. dired-create-destination-dirs-on-trailing-dirsep t
  2060. dired-isearch-filenames 'dwim
  2061. dired-do-revert-buffer (lambda (dir)
  2062. (not (file-remote-p dir)))
  2063. dired-clean-up-buffers-too t
  2064. dired-clean-confirm-killing-deleted-buffers t)
  2065. (evil-define-key '(normal visual motion) dired-mode-map
  2066. "u" #'dired-unmark
  2067. "U" #'dired-unmark-all-marks))
  2068. ;; dirvish
  2069. (use-package dirvish
  2070. :defer nil
  2071. :bind (("C-c b" . dirvish-quick-access)
  2072. :map dirvish-mode-map
  2073. ("<mouse-1>" . dirvish-subtree-toggle-or-open)
  2074. ("<mouse-2>" . dired-mouse-find-file-other-window)
  2075. ("<mouse-3>" . dired-mouse-find-file))
  2076. :hook (((dirvish-directory-view-mode dired-mode dirvish-mode) .
  2077. my/-setup-dirvish-lines)
  2078. ;; ((dirvish-directory-view-mode dired-mode dirvish-mode) .
  2079. ;; auto-revert-mode)
  2080. ((dirvish-mode dired-mode) . my/-setup-dirvish-mouse))
  2081. :custom
  2082. (dirvish-subtree-always-show-state t)
  2083. (dirvish-reuse-session t)
  2084. (dirvish-quick-access-function 'dired)
  2085. :init
  2086. (defun my/-setup-dirvish-lines ()
  2087. (setq-local truncate-lines t))
  2088. (defun my/-setup-dirvish-mouse ()
  2089. (setq-local mouse-1-click-follows-link nil
  2090. mouse-1-click-in-non-selected-windows nil))
  2091. (defvar my/-dirvish-base-quick-access-entries
  2092. `(("h" "~/" "Home")
  2093. ("d" "~/downloads/" "Downloads")
  2094. ("e" ,user-emacs-directory "Emacs user directory")
  2095. ("z" "~/.config/zsh/" "Zsh user directory")
  2096. ("o" "~/docs/" "Documents")
  2097. ("w." "~/workspace/" "Workspace")))
  2098. (defun my/-dirvish-build-quick-access-entries (bookmarks)
  2099. ;; NOTE called from a variable watcher for `bookmark-alist' and so must
  2100. ;; never set that variable
  2101. (let (out)
  2102. (dolist (bme bookmarks
  2103. (append my/-dirvish-base-quick-access-entries
  2104. (sort out
  2105. (lambda (elt1 elt2)
  2106. (string< (car elt1) (car elt2))))))
  2107. (let ((name (car bme)))
  2108. (let-alist (cdr bme)
  2109. (when (and (file-directory-p .filename)
  2110. (string-match (rx bos (group (any "a-z" "A-Z")) "ws" eos)
  2111. name))
  2112. (setf (alist-get (concat "w" (match-string 1 name)) out
  2113. nil nil 'equal)
  2114. (list .filename
  2115. (concat
  2116. (capitalize
  2117. (car (last (string-split .filename "/" t))))
  2118. " Workspace")))))))))
  2119. :config
  2120. (require 'dirvish-extras)
  2121. (defun my/-dirvish-bookmark-alist-watcher (_sym newval oper where)
  2122. (when (and (not where) (memq oper '(set makunbound defvaralias)))
  2123. (setopt dirvish-quick-access-entries
  2124. (my/-dirvish-build-quick-access-entries newval))))
  2125. (add-variable-watcher 'bookmark-alist #'my/-dirvish-bookmark-alist-watcher)
  2126. (defvar-local my/-dirvish-uid-name-cache nil
  2127. "Cons of path and a hash table mapping user ids to their names.")
  2128. (dirvish-define-attribute file-owner-mode
  2129. "The file's owner and mode."
  2130. :index 2
  2131. :when (and (dirvish-prop :root) dired-hide-details-mode
  2132. (> win-width 60))
  2133. (let ((root (dirvish-prop :root))
  2134. (uid (file-attribute-user-id f-attrs)))
  2135. (unless (or (dirvish-prop :remote) (stringp uid))
  2136. (unless (and (equal root (car my/-dirvish-uid-name-cache))
  2137. (hash-table-p (cdr my/-dirvish-uid-name-cache)))
  2138. (setq my/-dirvish-uid-name-cache
  2139. (cons root (make-hash-table :test 'equal))))
  2140. (if-let ((name (gethash uid (cdr my/-dirvish-uid-name-cache))))
  2141. (setq uid name)
  2142. (let* ((new-attrs (file-attributes f-name 'string))
  2143. (new-name (file-attribute-user-id new-attrs)))
  2144. (puthash uid new-name
  2145. (cdr my/-dirvish-uid-name-cache))
  2146. (setq uid new-name))))
  2147. (cons 'right (propertize (format " %s %s" uid (file-attribute-modes f-attrs))
  2148. 'face (or hl-face 'dirvish-file-time)))))
  2149. (let ((cur-val dirvish-ui-setup-items))
  2150. (cl-pushnew '("o" file-owner-mode "File owner and mode")
  2151. cur-val :test 'equal)
  2152. (setopt dirvish-ui-setup-items cur-val))
  2153. (add-to-list 'dirvish-libraries '(dirvish file-owner-mode))
  2154. (setopt dirvish-attributes
  2155. '(vc-state subtree-state nerd-icons file-size file-owner-mode))
  2156. (evil-define-key 'normal dirvish-mode-map
  2157. (kbd "q") #'dirvish-quit
  2158. (kbd "a") #'dirvish-quick-access
  2159. (kbd "f") #'dirvish-file-info-menu
  2160. (kbd "y") #'dirvish-yank-menu
  2161. (kbd "N") #'dirvish-narrow
  2162. (kbd "^") #'dirvish-history-last
  2163. (kbd "h") #'dirvish-history-jump
  2164. (kbd "s") #'dirvish-quicksort
  2165. (kbd "o") #'dirvish-quicksort
  2166. (kbd "v") #'dirvish-vc-menu
  2167. (kbd "TAB") #'dirvish-subtree-toggle
  2168. (kbd "M-f") #'dirvish-history-go-forward
  2169. (kbd "M-b") #'dirvish-history-go-backward
  2170. (kbd "M-l") #'dirvish-history-go-forward
  2171. (kbd "M-h") #'dirvish-history-go-backward
  2172. (kbd "M-l") #'dirvish-ls-switches-menu
  2173. (kbd "M-m") #'dirvish-mark-menu
  2174. (kbd "M-t") #'dirvish-layout-toggle
  2175. (kbd "M-s") #'dirvish-setup-menu
  2176. (kbd "M-e") #'dirvish-emerge-menu
  2177. (kbd "M-j") #'dirvish-fd-jump
  2178. (kbd "/") #'dirvish-fd
  2179. (kbd "?") #'dirvish-dispatch)
  2180. (dirvish-override-dired-mode)
  2181. (dirvish-define-preview eza (file)
  2182. "Use `eza' to generate directory preview."
  2183. :require ("eza")
  2184. (when (file-directory-p file)
  2185. `(shell . ("eza" "-la" "--color=always" "--icons"
  2186. "--group-directories-first" ,file))))
  2187. (add-to-list 'dirvish-preview-dispatchers 'eza))
  2188. ;; ibuffer
  2189. (use-package ibuffer
  2190. :bind ("C-x C-b" . ibuffer))
  2191. ;; magit
  2192. (use-package magit
  2193. :init
  2194. (defvar-keymap my/magit-personal-prefix-map
  2195. :doc "Keymap for some useful Magit commands."
  2196. "s" #'magit-stage
  2197. "d" #'magit-diff-dwim
  2198. "D" #'magit-diff
  2199. "b" #'magit-blame)
  2200. (keymap-global-set "C-c i" my/magit-personal-prefix-map)
  2201. (evil-define-key '(normal visual motion) magit-mode-map
  2202. "s" #'magit-stage-file
  2203. "S" #'magit-stage-modified))
  2204. (use-package forge
  2205. :config
  2206. (add-to-list 'forge-alist '("git.zander.im" "git.zander.im/api/v1"
  2207. "git.zander.im" forge-gitea-repository)))
  2208. ;; org-mode
  2209. (use-package org
  2210. :pin gnu
  2211. :bind (("C-c c" . org-capture)
  2212. ("C-c a" . org-agenda)
  2213. ("C-c l" . org-store-link)
  2214. :map org-mode-map
  2215. ("C-c t" . org-table-create))
  2216. :hook (org-mode . org-table-header-line-mode)
  2217. :init
  2218. (font-lock-add-keywords 'org-mode
  2219. `((,(rx bol (* " ") (group "-") " ")
  2220. (0 (prog1 nil
  2221. (compose-region (match-beginning 1)
  2222. (match-end 1) "•"))))))
  2223. (setq org-directory "~/org"
  2224. org-agenda-files '("~/org/")
  2225. org-log-into-drawer t
  2226. org-log-done 'time
  2227. org-log-redeadline 'time
  2228. org-log-reschedule 'time
  2229. org-preview-latex-default-process 'dvisvgm
  2230. org-highlight-latex-and-related '(native entities)
  2231. org-startup-with-inline-images t
  2232. org-adapt-indentation t
  2233. org-hide-leading-stars t
  2234. org-html-with-latex 'dvisvgm
  2235. org-preview-latex-process-alist
  2236. '((dvisvgm
  2237. :image-input-type "dvi"
  2238. :image-output-type "svg"
  2239. :image-size-adjust (1.7 . 1.5)
  2240. :latex-compiler ("pdflatex -interaction nonstopmode -output-format=dvi -output-directory=%o %f")
  2241. :image-converter ("dvisvgm %o%b.dvi --no-fonts --exact-bbox --scale=%S --output=%O"))))
  2242. (defun my/-org-allow-in-derived-mode (oldfun &rest r)
  2243. "Allow OLDFUN to run, even if `major-mode' is only derived from `org-mode'.
  2244. R is rest of the arguments to OLDFUN."
  2245. (let ((major-mode (if (derived-mode-p 'org-mode)
  2246. 'org-mode
  2247. major-mode)))
  2248. (apply oldfun r)))
  2249. (advice-add 'org-element-at-point :around 'my/-org-allow-in-derived-mode)
  2250. (advice-add 'org-table-header-line-mode :around 'my/-org-allow-in-derived-mode))
  2251. (use-package evil-org
  2252. :after org
  2253. :hook (org-mode . evil-org-mode)
  2254. :init
  2255. (require 'evil-org-agenda)
  2256. (evil-org-agenda-set-keys))
  2257. ;; ledger
  2258. (use-package ledger-mode)
  2259. (use-package flycheck-ledger
  2260. :hook (ledger-mode . trusted-files-flycheck-mode-if-safe))
  2261. ;; khard contacts
  2262. (require 'khard)
  2263. ;; This is also in khard (see above), it's just also here so that if I remove
  2264. ;; that file ever, other things will not break.
  2265. (defun my/message-in-header-p (name &optional testfn)
  2266. "If in field NAME, return the start of the header, otherwise, return nil.
  2267. The name is compared with the field name using TESTFN (defaults to `equal')."
  2268. (save-excursion
  2269. (when (and (message-point-in-header-p)
  2270. (message-beginning-of-header t))
  2271. (beginning-of-line)
  2272. (when (and (looking-at (rx bol (group (+? any)) ":" (? " ")))
  2273. (funcall (or testfn 'equal) (match-string 1) name))
  2274. (match-end 0)))))
  2275. ;; mu4e
  2276. (use-package mu4e
  2277. :ensure nil
  2278. :defer nil
  2279. :hook ((mu4e-index-updated . my/-mu4e-enable-index-messages)
  2280. (mu4e-main-mode . my/-mu4e-setup-main-mode)
  2281. (mu4e-view-mode . my/-mu4e-setup-view-mode)
  2282. (mu4e-compose-mode . my/-mu4e-setup-compose-mode))
  2283. :bind (("C-x C-m" . mu4e)
  2284. :map message-mode-map
  2285. ("C-c k" . khard-insert-email-contact)
  2286. :map mu4e-headers-mode-map
  2287. ([remap mu4e-headers-mark-for-trash] .
  2288. my/mu4e-headers-mark-for-trash)
  2289. :map mu4e-view-mode-map
  2290. ([remap mu4e-view-mark-for-trash] .
  2291. my/mu4e-view-mark-for-trash))
  2292. :init
  2293. (require 'mu4e)
  2294. (evil-define-key '(normal motion) mu4e-main-mode-map "q" #'bury-buffer)
  2295. (evil-define-key '(normal motion) mu4e-view-mode-map "gy" #'mu4e-view-save-url)
  2296. (defun my/-mu4e-setup-view-mode ()
  2297. (setq-local global-hl-line-mode nil))
  2298. (defun my/-mu4e-setup-main-mode ()
  2299. (setq-local default-directory "~/"))
  2300. (defun my/-mu4e-enable-index-messages ()
  2301. (setq mu4e-hide-index-messages nil))
  2302. (defun my/mu4e-update-mail-and-index-silent ()
  2303. "Run `mu4e-update-mail-and-index' without any messages in the background."
  2304. (setq mu4e-hide-index-messages t)
  2305. (mu4e-update-mail-and-index t))
  2306. (defun my/mu4e-headers-mark-for-trash ()
  2307. "Move the message a point to the trash without marking it was deleted
  2308. (trashed)."
  2309. (interactive)
  2310. (when (mu4e-thread-message-folded-p)
  2311. (mu4e-warn "Cannot mark folded messages"))
  2312. (mu4e-mark-at-point 'move mu4e-trash-folder)
  2313. (when mu4e-headers-advance-after-mark
  2314. (mu4e-headers-next)))
  2315. (defun my/mu4e-view-mark-for-trash ()
  2316. "Like `my/mu4e-headers-mark-for-trash', but for `mu4e-view-mode'."
  2317. (interactive)
  2318. (mu4e--view-in-headers-context
  2319. (my/mu4e-headers-mark-for-trash)))
  2320. (defun my/-mu4e-enable-autocomplete-in-header ()
  2321. ;; corfu auto must be t (not the integer returned by
  2322. ;; `my/message-in-header-p'
  2323. (setq-local corfu-auto (and (not (window-minibuffer-p))
  2324. (my/message-in-header-p "To")
  2325. t)))
  2326. (defun my/-mu4e-setup-compose-mode ()
  2327. (add-hook 'post-command-hook 'my/-mu4e-enable-autocomplete-in-header
  2328. nil t)
  2329. (add-to-list
  2330. (make-local-variable 'completion-at-point-functions)
  2331. (cape-capf-super #'mu4e-complete-contact #'khard-message-mode-capf)))
  2332. (defun my/-mu4e-fix-cycle-threshold ()
  2333. (setq-local completion-cycle-threshold nil))
  2334. (advice-add 'mu4e--compose-setup-completion :after
  2335. 'my/-mu4e-fix-cycle-threshold)
  2336. (defvar my/mu4e-interesting-mail-query
  2337. (concat "flag:unread AND NOT flag:trashed AND NOT "
  2338. "maildir:/protonmail/Trash AND NOT maildir:/protonmail/Spam")
  2339. "Flag for mail which will appear as \"unread\" and will be notified.")
  2340. (setq message-kill-buffer-on-exit t
  2341. message-confirm-send t
  2342. message-send-mail-function 'sendmail-send-it
  2343. mu4e-change-filenames-when-moving t
  2344. mu4e-context-policy 'pick-first
  2345. mu4e-attachment-dir "~/downloads/"
  2346. mu4e-last-update-buffer " *mu4e-last-update*"
  2347. mu4e-index-update-error-warning nil
  2348. mu4e-get-mail-command "mbsync protonmail"
  2349. mu4e-completing-read-function #'completing-read-default
  2350. mu4e-compose-context-policy 'ask-if-none
  2351. mu4e-contexts
  2352. (list (make-mu4e-context
  2353. :name "Personal"
  2354. :match-func (lambda (msg)
  2355. (when msg
  2356. (string-match-p "^/protonmail/"
  2357. (mu4e-message-field msg
  2358. :maildir))))
  2359. :vars `((user-mail-address . ,(my/get-private 'mu4e-email))
  2360. (user-full-name . ,(my/get-private 'mu4e-name))
  2361. (message-signature . nil)
  2362. (mu4e-refile-folder . "/protonmail/Archive")
  2363. (mu4e-sent-folder . "/protonmail/Sent")
  2364. (mu4e-drafts-folder . "/protonmail/Drafts")
  2365. (mu4e-trash-folder . "/protonmail/Trash")
  2366. (mu4e-bookmarks
  2367. . ((:name "Inbox"
  2368. :query "maildir:/protonmail/Inbox"
  2369. :key ?i)
  2370. (:name "Unread"
  2371. :query ,my/mu4e-interesting-mail-query
  2372. :key ?u))))))))
  2373. (use-package mu4e-alert
  2374. :after mu4e
  2375. :hook (after-init . mu4e-alert-enable-notifications)
  2376. :init
  2377. (setq mu4e-alert-set-window-urgency nil
  2378. mu4e-alert-interesting-mail-query my/mu4e-interesting-mail-query)
  2379. :config
  2380. (mu4e-alert-set-default-style 'libnotify))
  2381. (mu4e t)
  2382. (mu4e-context-switch nil "Personal")
  2383. ;; mu4e compose HTML messages
  2384. (use-package org-mime)
  2385. (require 'org-mu4e-compose)
  2386. (setq ;; mail-user-agent 'org-mu4e-user-agent
  2387. org-mime-org-html-with-latex-default 'dvisvgm
  2388. org-mime-export-options '(:with-latex dvisvgm :with-footnotes t))
  2389. ;; (evil-define-key '(normal visual) org-mu4e-compose-mode-map
  2390. ;; "G" #'mu4e-compose-goto-bottom
  2391. ;; "gg" #'mu4e-compose-goto-top)
  2392. ;; (evil-define-key 'normal org-mu4e-compose-mode-map
  2393. ;; "ZZ" #'message-send-and-exit
  2394. ;; "ZD" #'message-dont-send
  2395. ;; "ZQ" #'message-kill-buffer
  2396. ;; "ZF" #'mml-attach-file)
  2397. ;; (evil-define-key 'normal mu4e-view-mode-map
  2398. ;; "R" 'org-mu4e-compose-reply
  2399. ;; "cr" 'org-mu4e-compose-reply)
  2400. ;; (evil-define-key 'normal mu4e-headers-mode-map
  2401. ;; "R" 'org-mu4e-compose-reply
  2402. ;; "cr" 'org-mu4e-compose-reply)
  2403. ;; (defun my/-setup-org-mu4e-compose-mode ()
  2404. ;; "Setup up stuff in `org-mu4e-compose' buffers."
  2405. ;; (setq-local ltex-eglot-variable-save-method 'file)
  2406. ;; ;; this should come last so it can pick up the above
  2407. ;; ;; (trusted-files-eglot-ensure-if-safe)
  2408. ;; )
  2409. ;; (add-hook 'org-mu4e-compose-mode-hook #'my/-setup-org-mu4e-compose-mode)
  2410. ;; elfeed
  2411. (use-package elfeed
  2412. :bind (("C-c d" . elfeed))
  2413. :custom
  2414. (elfeed-feeds
  2415. '(("https://archlinux.org/feeds/news/" linux arch)
  2416. ("https://9to5linux.com/feed/atom" linux news)))
  2417. :config
  2418. (setq elfeed-log-buffer-name " *elfeed-log*")
  2419. (evil-define-key '(normal motion) elfeed-search-mode-map
  2420. "r" #'elfeed-search-fetch)
  2421. (elfeed-db-load))
  2422. ;; helpful
  2423. (use-package helpful
  2424. :hook ((emacs-lisp-mode . my/-helpful-setup-emacs-lisp-mode)
  2425. (helpful-mode . my/-setup-helpful-mode))
  2426. :bind (:map help-map
  2427. ("f" . helpful-callable)
  2428. ("v" . helpful-variable)
  2429. ("k" . helpful-key)
  2430. ("o" . helpful-symbol)
  2431. ("x" . helpful-command)
  2432. ("F" . helpful-function)
  2433. :map helpful-mode-map
  2434. ("<mouse-8>" . my/helpful-history-back)
  2435. ("<mouse-9>" . my/helpful-history-forward)
  2436. ("<normal-state><" . my/helpful-history-back)
  2437. ("<normal-state>>" . my/helpful-history-forward))
  2438. :init
  2439. (defun my/-helpful-setup-emacs-lisp-mode ()
  2440. (setq-local evil-lookup-func #'helpful-at-point))
  2441. (defun my/-setup-helpful-mode ()
  2442. (setq-local evil-lookup-func #'helpful-at-point
  2443. tab-width 8))
  2444. (defvar my/helpful-symbol-history-size 50
  2445. "Max size of `my/helpful-symbol-history'.")
  2446. (defvar my/helpful-symbol-history '()
  2447. "History of helpful symbols.")
  2448. (defvar my/-helpful-inhibit-history nil
  2449. "If non-nil, don't add symbols to `my/helpful-symbol-history'.")
  2450. (defvar my/-helpful-last-entry nil
  2451. "Last entry looked up with helpful.")
  2452. (defun my/helpful-history-back (count)
  2453. "Go back COUNT symbols in `my/helpful-symbol-history'. If called
  2454. interactively, COUNT defaults to 1."
  2455. (interactive "p")
  2456. (my/helpful-history-forward (- count)))
  2457. (defun my/helpful-history-forward (count)
  2458. "Move COUNT symbols in `my/helpful-symbol-history'. If COUNT is negative,
  2459. move back. If COUNT is larger than the history, go to the newest entry. Go to
  2460. the oldest entry if -COUNT is larger than the history."
  2461. (interactive "p")
  2462. (when helpful--sym
  2463. (let* ((hist-len (length my/helpful-symbol-history))
  2464. (current-pos (seq-position my/helpful-symbol-history
  2465. (cons helpful--sym
  2466. helpful--callable-p)
  2467. 'equal))
  2468. (new-pos (- current-pos count)))
  2469. (cond
  2470. ;; if already at the newest element, signal an error
  2471. ((and (> count 0) (= current-pos 0))
  2472. (message "%s" "No newer symbol!"))
  2473. ;; if already at the oldest element, signal an error
  2474. ((and (< count 0) (= (1+ current-pos) hist-len))
  2475. (message "%s" "No older symbol!"))
  2476. (t
  2477. (let ((my/-helpful-inhibit-history t)
  2478. (entry (cond
  2479. ((<= new-pos 0)
  2480. (seq-first my/helpful-symbol-history))
  2481. ((>= new-pos hist-len)
  2482. (car (last my/helpful-symbol-history)))
  2483. (t
  2484. (nth new-pos my/helpful-symbol-history)))))
  2485. (if (cdr entry)
  2486. (helpful-callable (car entry))
  2487. (helpful-variable (car entry)))))))))
  2488. (defun my/-helpful-switch-buffer-function (helpful-buf)
  2489. "Like `pop-to-buffer', but kill previous helpful buffers and save the new
  2490. buffers `helpful--sym' to `my/helpful-symbol-history'."
  2491. (cl-loop with window = nil
  2492. for buf in (buffer-list)
  2493. when (and
  2494. (not (eq buf helpful-buf))
  2495. (eq (buffer-local-value 'major-mode buf) 'helpful-mode))
  2496. do
  2497. (when-let (cur-window (get-buffer-window buf nil))
  2498. (setq window cur-window))
  2499. (kill-buffer buf)
  2500. finally
  2501. (let ((entry (cons (buffer-local-value 'helpful--sym helpful-buf)
  2502. (buffer-local-value 'helpful--callable-p
  2503. helpful-buf))))
  2504. (unless my/-helpful-inhibit-history
  2505. (when-let (from-current-hist
  2506. (member my/-helpful-last-entry
  2507. my/helpful-symbol-history))
  2508. (setq my/helpful-symbol-history from-current-hist))
  2509. (cl-pushnew entry my/helpful-symbol-history :test 'equal)
  2510. (setq my/helpful-symbol-history
  2511. (seq-take my/helpful-symbol-history
  2512. my/helpful-symbol-history-size)))
  2513. (setq my/-helpful-last-entry entry))
  2514. (if window
  2515. (window--display-buffer helpful-buf window 'reuse)
  2516. (pop-to-buffer helpful-buf))))
  2517. (setq helpful-switch-buffer-function 'my/-helpful-switch-buffer-function
  2518. helpful-max-buffers 2))
  2519. (defun my/greyify-color (color percent &optional frame)
  2520. "Make COLOR closer to black by PERCENT on FRAME.
  2521. Color can be any color which can be passed to `color-values'."
  2522. (cl-destructuring-bind (&optional r g b)
  2523. (color-name-to-rgb color frame)
  2524. (when (and r g b)
  2525. (let ((scale (- 1.0 (/ percent 100.0))))
  2526. (color-rgb-to-hex (* r scale)
  2527. (* g scale)
  2528. (* b scale))))))
  2529. ;; rainbow-delimiters
  2530. (use-package rainbow-delimiters
  2531. :hook (prog-mode . rainbow-delimiters-mode)
  2532. :config
  2533. ;; generate dark version of the rainbow delimiters faces
  2534. (defun my/-rainbow-delimiters-recalc-dark-faces (&optional frame)
  2535. (unless frame (setq frame (selected-frame)))
  2536. (dotimes (i 9)
  2537. (when-let ((old-face (intern-soft
  2538. (format "rainbow-delimiters-depth-%d-face"
  2539. (1+ i))))
  2540. (new-face
  2541. (intern
  2542. (format "my/rainbow-delimiters-depth-%d-dark-face"
  2543. (1+ i))))
  2544. (old-color (face-attribute old-face :foreground frame))
  2545. (new-color (my/greyify-color old-color 50 frame)))
  2546. (set-face-attribute new-face frame :foreground new-color))))
  2547. (add-hook 'after-make-frame-functions
  2548. #'my/-rainbow-delimiters-recalc-dark-faces)
  2549. (add-hook 'server-after-make-frame-hook
  2550. #'my/-rainbow-delimiters-recalc-dark-faces)
  2551. (defun my/rainbow-delimiters-parinfer-pick-face (depth match loc)
  2552. "Version of `rainbow-delimiters-default-pick-face' that colors closing
  2553. parenthesis darker than opening ones. This function defers to
  2554. `rainbow-delimiters-default-pick-face' and just changes the output if it returns
  2555. one of the normal rainbow-delimiters-depth-N-face faces."
  2556. (save-match-data
  2557. (let* ((base-face (rainbow-delimiters-default-pick-face depth match loc))
  2558. (base-name (symbol-name base-face)))
  2559. (if (and evil-cleverparens-mode
  2560. (eq ?\) (char-syntax
  2561. (elt (buffer-substring-no-properties loc (1+ loc)) 0)))
  2562. (string-match (rx string-start "rainbow-delimiters-depth-"
  2563. (group (+ num))
  2564. "-face" string-end)
  2565. base-name))
  2566. (or (intern-soft (format "my/rainbow-delimiters-depth-%s-dark-face"
  2567. (match-string 1 base-name)))
  2568. base-face)
  2569. base-face))))
  2570. (setopt rainbow-delimiters-pick-face-function
  2571. 'my/rainbow-delimiters-parinfer-pick-face))
  2572. ;; make regexp look nicer
  2573. (use-package easy-escape
  2574. :hook ((emacs-lisp-mode reb-mode) . easy-escape-minor-mode)
  2575. :config
  2576. (face-spec-set 'easy-escape-face
  2577. '((t (:foreground unspecified
  2578. :weight bold
  2579. :inherit 'font-lock-regexp-grouping-backslash)))))
  2580. ;; auto-highlight-symbol
  2581. (use-package auto-highlight-symbol
  2582. :hook (lisp-data-mode . auto-highlight-symbol-mode)
  2583. :init
  2584. (setq ahs-face 'bold
  2585. ahs-face-unfocused 'bold
  2586. ahs-definition-face 'bold
  2587. ahs-definition-face-unfocused 'bold
  2588. ahs-plugin-default-face 'bold
  2589. ahs-plugin-default-face-unfocused 'bold)
  2590. :config
  2591. (keymap-unset auto-highlight-symbol-mode-map "C-x C-a" t))
  2592. ;; Theme (doom-themes)
  2593. (use-package doom-themes
  2594. :config
  2595. (load-theme 'doom-molokai t)
  2596. (doom-themes-org-config))
  2597. ;; solaire-mode
  2598. (use-package solaire-mode
  2599. :config
  2600. (solaire-global-mode 1))
  2601. ;; Highlight todos
  2602. (use-package hl-todo
  2603. :hook (prog-mode . hl-todo-mode))
  2604. (use-package magit-todos
  2605. :after (hl-todo magit)
  2606. :config
  2607. (magit-todos-mode 1))
  2608. ;; icons
  2609. (use-package nerd-icons)
  2610. (use-package nerd-icons-completion
  2611. :config
  2612. (nerd-icons-completion-mode))
  2613. (use-package nerd-icons-dired
  2614. :hook (dired-mode . my/-maybe-enable-nerd-icons-dired)
  2615. :init
  2616. (defun my/-maybe-enable-nerd-icons-dired ()
  2617. (unless (bound-and-true-p dirvish-override-dired-mode)
  2618. (nerd-icons-dired-mode))))
  2619. (use-package kind-icon
  2620. :after corfu
  2621. :init
  2622. (setq kind-icon-default-face 'corfu-default
  2623. kind-icon-default-style
  2624. '(:padding -1 :stroke 0 :margin 0 :radius 0 :height 0.5 :scale 1))
  2625. :config
  2626. (add-to-list 'corfu-margin-formatters #'kind-icon-margin-formatter))
  2627. ;; modeline (doom-modeline)
  2628. (use-package doom-modeline
  2629. :init
  2630. (setq doom-modeline-support-imenu t)
  2631. (doom-modeline-mode 1))
  2632. ;; dashboard.el
  2633. (use-package dashboard
  2634. :config
  2635. (defvar-local my/-dashboard-did-fix-image nil
  2636. "Weather or not the dashboard image has been fixed in this buffer.")
  2637. (defun my/-dashboard-fix-image ()
  2638. (unless my/-dashboard-did-fix-image
  2639. (dashboard-refresh-buffer)
  2640. (setq my/-dashboard-did-fix-image t)))
  2641. (defun my/-dashboard-setup-function ()
  2642. (add-hook 'window-configuration-change-hook 'my/-dashboard-fix-image nil t)
  2643. (setq-local display-line-numbers nil))
  2644. (add-hook 'dashboard-mode-hook 'my/-dashboard-setup-function)
  2645. (set-face-background 'dashboard-banner-logo-title nil)
  2646. (dashboard-setup-startup-hook)
  2647. (setq initial-buffer-choice (lambda () (get-buffer-create "*dashboard*"))
  2648. dashboard-force-refresh t
  2649. dashboard-display-icons-p t
  2650. dashboard-icon-type 'nerd-icons
  2651. dashboard-set-file-icons t
  2652. dashboard-projects-backend 'project-el
  2653. dashboard-items '((recents . 5)
  2654. (projects . 5)
  2655. (bookmarks . 5))))
  2656. ;; page break lines
  2657. (use-package page-break-lines
  2658. :config
  2659. (global-page-break-lines-mode 1)
  2660. (add-to-list 'page-break-lines-modes 'prog-mode)
  2661. (add-to-list 'page-break-lines-modes 'text-mode)
  2662. (add-to-list 'page-break-lines-modes 'helpful-mode))
  2663. ;; fun!
  2664. (use-package mines)
  2665. ;;; init.el ends here