You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

2052 lines
64 KiB

5 years ago
2 years ago
3 years ago
3 years ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
6 years ago
10 months ago
10 months ago
10 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
1 month ago
1 year ago
8 months ago
6 months ago
6 months ago
8 months ago
6 months ago
6 months ago
6 months ago
6 months ago
6 months ago
6 months ago
6 months ago
6 months ago
8 months ago
6 months ago
11 months ago
11 months ago
11 months ago
6 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
10 months ago
11 months ago
11 months ago
3 months ago
3 months ago
3 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
1 year ago
6 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
6 years ago
11 months ago
11 months ago
11 months ago
6 years ago
11 months ago
6 years ago
6 years ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
11 months ago
1 year ago
1 year ago
1 year ago
  1. #+TITLE: Emacs configuration file
  2. #+AUTHOR: Marc
  3. #+BABEL: :cache yes
  4. #+PROPERTY: header-args :tangle init.el
  5. #+OPTIONS: ^:nil
  6. * TODOS
  7. - early-init.el? What to outsource here?
  8. - Paket exec-path-from-shell, um PATH aus Linux auch in emacs zu haben
  9. - Smart mode line?
  10. - Theme
  11. - flymake instead of flycheck?
  12. - Hydra
  13. - General
  14. - (defalias 'list-buffers 'ibuffer) ;; change default to ibuffer
  15. - ido?
  16. - treemacs (for linux)
  17. windmove?
  18. - tramp (in linux)
  19. - visual-regexp
  20. - org configuration: paths
  21. - org custom agenda
  22. - org-ql (related to org agendas)
  23. - org configuration: everything else
  24. - beancount configuration from config.org
  25. - CONTINUE TODO from config.org at Programming
  26. - all-the-icons?
  27. - lispy? [[https://github.com/abo-abo/lispy]]
  28. * Header
  29. Emacs variables are dynamically scoped. That's unusual for most languages, so disable it here, too
  30. #+begin_src emacs-lisp
  31. ;;; init.el --- -*- lexical-binding: t -*-
  32. #+end_src
  33. * First start
  34. These functions updates config.el whenever changes in config.org are made. The update will be active after saving.
  35. #+BEGIN_SRC emacs-lisp
  36. (defun my/tangle-config ()
  37. "Export code blocks from the literate config file."
  38. (interactive)
  39. ;; prevent emacs from killing until tangle-process has finished
  40. (add-to-list 'kill-emacs-query-functions
  41. (lambda ()
  42. (or (not (process-live-p (get-process "tangle-process")))
  43. (y-or-n-p "\"my/tangle-config\" is running; kill it? "))))
  44. (org-babel-tangle-file config-org init-el)
  45. (message "reloading user-init-file")
  46. (load-file init-el))
  47. (add-hook 'org-mode-hook
  48. (lambda ()
  49. (if (equal (buffer-file-name) config-org)
  50. (my--add-local-hook 'after-save-hook 'my/tangle-config))))
  51. (defun my--add-local-hook (hook function)
  52. "Add buffer-local hook."
  53. (add-hook hook function :local t))
  54. #+END_SRC
  55. A small function to measure start up time.
  56. Compare that to
  57. emacs -q --eval='(message "%s" (emacs-init-time))'
  58. (roughly 0.27s)
  59. https://blog.d46.us/advanced-emacs-startup/
  60. #+begin_src emacs-lisp
  61. (add-hook 'emacs-startup-hook
  62. (lambda ()
  63. (message "Emacs ready in %s with %d garbage collections."
  64. (format "%.2f seconds"
  65. (float-time
  66. (time-subtract after-init-time before-init-time)))
  67. gcs-done)))
  68. ;(setq gc-cons-threshold (* 50 1000 1000))
  69. #+end_src
  70. * Default settings
  71. ** paths
  72. #+BEGIN_SRC emacs-lisp
  73. (defconst *sys/gui*
  74. (display-graphic-p)
  75. "Is emacs running in a gui?")
  76. (defconst *sys/linux*
  77. (string-equal system-type 'gnu/linux)
  78. "Is the system running Linux?")
  79. (defconst *sys/windows*
  80. (string-equal system-type 'windows-nt)
  81. "Is the system running Windows?")
  82. (defconst *home_desktop*
  83. (string-equal (system-name) "marc")
  84. "Is emacs running on my desktop?")
  85. (defconst *home_laptop*
  86. (string-equal (system-name) "laptop")
  87. "Is emacs running on my laptop?")
  88. (defconst *work_local*
  89. (string-equal (system-name) "PMPCNEU08")
  90. "Is emacs running at work on the local system?")
  91. (defconst *work_remote*
  92. (or (string-equal (system-name) "PMTS01")
  93. (string-equal (system-name) "PMTSNEU01"))
  94. "Is emacs running at work on the remote system?")
  95. #+END_SRC
  96. #+BEGIN_SRC emacs-lisp
  97. (defvar MY--PATH_USER_LOCAL (concat user-emacs-directory "user-local/"))
  98. (defvar MY--PATH_USER_GLOBAL (concat user-emacs-directory "user-global/"))
  99. (add-to-list 'custom-theme-load-path (concat MY--PATH_USER_GLOBAL "themes"))
  100. (when *sys/linux*
  101. (defconst MY--PATH_ORG_FILES (expand-file-name "~/archiv/Organisieren/"))
  102. (defconst MY--PATH_ORG_FILES_MOBILE (expand-file-name "~/archiv/Organisieren/mobile/"))
  103. (defconst MY--PATH_ORG_JOURNAl (expand-file-name "~/archiv/Organisieren/Journal/"))
  104. (defconst MY--PATH_ORG_ROAM (file-truename "~/archiv/Organisieren/notes/")))
  105. (when *work_remote*
  106. (defconst MY--PATH_ORG_FILES "p:/Eigene Dateien/Notizen/")
  107. (defconst MY--PATH_ORG_FILES_MOBILE nil) ;; hacky way to prevent "free variable" compiler error
  108. (defconst MY--PATH_ORG_JOURNAL nil) ;; hacky way to prevent "free variable" compiler error
  109. (defconst MY--PATH_START "p:/Eigene Dateien/Notizen/")
  110. (defconst MY--PATH_ORG_ROAM (expand-file-name "p:/Eigene Dateien/Notizen/")))
  111. (setq custom-file (concat MY--PATH_USER_LOCAL "custom.el")) ;; don't spam init.e with saved customization settings
  112. (setq backup-directory-alist `((".*" . ,temporary-file-directory)))
  113. (setq auto-save-file-name-transforms `((".*" ,temporary-file-directory)))
  114. (customize-set-variable 'auth-sources (list (concat MY--PATH_USER_LOCAL "authinfo")
  115. (concat MY--PATH_USER_LOCAL "authinfo.gpg")
  116. (concat MY--PATH_USER_LOCAL "netrc")))
  117. #+end_src
  118. ** Browser
  119. #+begin_src emacs-lisp
  120. (setq browse-url-function 'browse-url-generic
  121. browse-url-generic-program "firefox")
  122. #+end_src* Package Management
  123. ** Elpaca
  124. Boilerplate for Elpaca
  125. #+begin_src emacs-lisp
  126. (defvar elpaca-installer-version 0.8)
  127. (defvar elpaca-directory (expand-file-name "elpaca/" user-emacs-directory))
  128. (defvar elpaca-builds-directory (expand-file-name "builds/" elpaca-directory))
  129. (defvar elpaca-repos-directory (expand-file-name "repos/" elpaca-directory))
  130. (defvar elpaca-order '(elpaca :repo "https://github.com/progfolio/elpaca.git"
  131. :ref nil :depth 1
  132. :files (:defaults "elpaca-test.el" (:exclude "extensions"))
  133. :build (:not elpaca--activate-package)))
  134. (let* ((repo (expand-file-name "elpaca/" elpaca-repos-directory))
  135. (build (expand-file-name "elpaca/" elpaca-builds-directory))
  136. (order (cdr elpaca-order))
  137. (default-directory repo))
  138. (add-to-list 'load-path (if (file-exists-p build) build repo))
  139. (unless (file-exists-p repo)
  140. (make-directory repo t)
  141. (when (< emacs-major-version 28) (require 'subr-x))
  142. (condition-case-unless-debug err
  143. (if-let* ((buffer (pop-to-buffer-same-window "*elpaca-bootstrap*"))
  144. ((zerop (apply #'call-process `("git" nil ,buffer t "clone"
  145. ,@(when-let* ((depth (plist-get order :depth)))
  146. (list (format "--depth=%d" depth) "--no-single-branch"))
  147. ,(plist-get order :repo) ,repo))))
  148. ((zerop (call-process "git" nil buffer t "checkout"
  149. (or (plist-get order :ref) "--"))))
  150. (emacs (concat invocation-directory invocation-name))
  151. ((zerop (call-process emacs nil buffer nil "-Q" "-L" "." "--batch"
  152. "--eval" "(byte-recompile-directory \".\" 0 'force)")))
  153. ((require 'elpaca))
  154. ((elpaca-generate-autoloads "elpaca" repo)))
  155. (progn (message "%s" (buffer-string)) (kill-buffer buffer))
  156. (error "%s" (with-current-buffer buffer (buffer-string))))
  157. ((error) (warn "%s" err) (delete-directory repo 'recursive))))
  158. (unless (require 'elpaca-autoloads nil t)
  159. (require 'elpaca)
  160. (elpaca-generate-autoloads "elpaca" repo)
  161. (load "./elpaca-autoloads")))
  162. (add-hook 'after-init-hook #'elpaca-process-queues)
  163. (elpaca `(,@elpaca-order))
  164. #+end_src
  165. some elpaca customization after the boilerplate
  166. #+begin_src emacs-lisp
  167. (elpaca elpaca-use-package
  168. (elpaca-use-package-mode))
  169. (when *work_remote*
  170. (setq elpaca-queue-limit 12)
  171. (elpaca-no-symlink-mode))
  172. (elpaca-wait)
  173. #+end_src
  174. stolen from
  175. https://github.com/progfolio/.emacs.d/blob/cff07d4454d327a4df1915a2cdf8ac6bc5dfde23/init.org?plain=1#L276
  176. usage e.g. here. Not sure though what the benefits are
  177. https://github.com/justinbarclay/.emacs.d
  178. #+begin_src emacs-lisp
  179. (defmacro use-feature (name &rest args)
  180. "Like `use-package' but accounting for asynchronous installation.
  181. NAME and ARGS are in `use-package'."
  182. (declare (indent defun))
  183. `(use-package ,name
  184. :ensure nil
  185. ,@args))
  186. #+end_src
  187. * use-package keywords general / diminish
  188. Needs to be loaded before any other package which uses the :general keyword
  189. #+BEGIN_SRC emacs-lisp
  190. (use-package general
  191. :ensure t
  192. :demand t)
  193. (use-package diminish
  194. :ensure t
  195. :demand t)
  196. ;;wait for elpaca any time a use-package keyword is added
  197. (elpaca-wait)
  198. #+END_SRC
  199. * sane defaults
  200. #+begin_src emacs-lisp
  201. (setq-default create-lockfiles nil) ;; disable lock files, can cause trouble in e.g. lsp-mode
  202. (defalias 'yes-or-no-p 'y-or-n-p) ;; answer with y and n
  203. (setq custom-safe-themes t) ;; don't ask me if I want to load a theme
  204. (setq sentence-end-double-space nil) ;; don't coun two spaces after a period as the end of a sentence.
  205. (delete-selection-mode t) ;; delete selected region when typing
  206. (use-package saveplace
  207. :ensure nil
  208. :config
  209. (save-place-mode 1) ;; saves position in file when it's closed
  210. :custom
  211. (save-place-file (concat MY--PATH_USER_LOCAL "places")))
  212. (setq save-place-forget-unreadable-files nil) ;; checks if file is readable before saving position
  213. (global-set-key (kbd "RET") 'newline-and-indent) ;; indent after newline
  214. (setq save-interprogram-paste-before-kill t) ;; put replaced text into killring
  215. ;; https://emacs.stackexchange.com/questions/3673/how-to-make-vc-and-magit-treat-a-symbolic-link-to-a-real-file-in-git-repo-just
  216. (setq find-file-visit-truename t) ;; some programs like lsp have trouble following symlinks, maybe vc-follow-symlinks would be enough
  217. #+END_SRC
  218. * Performance Optimization
  219. ** Garbage Collection
  220. Make startup faster by reducing the frequency of garbage collection.
  221. Set gc-cons-threshold (default is 800kb) to maximum value available, to prevent any garbage collection from happening during load time.
  222. #+BEGIN_SRC emacs-lisp :tangle early-init.el
  223. (setq gc-cons-threshold most-positive-fixnum)
  224. #+END_SRC
  225. Restore it to reasonable value after init. Also stop garbage collection during minibuffer interaction (helm etc.)
  226. #+begin_src emacs-lisp
  227. (defconst 1mb 1048576)
  228. (defconst 20mb 20971520)
  229. (defconst 30mb 31457280)
  230. (defconst 50mb 52428800)
  231. (defun my--defer-garbage-collection ()
  232. (setq gc-cons-threshold most-positive-fixnum))
  233. (defun my--restore-garbage-collection ()
  234. (run-at-time 1 nil (lambda () (setq gc-cons-threshold 30mb))))
  235. (add-hook 'emacs-startup-hook 'my--restore-garbage-collection 100)
  236. (add-hook 'minibuffer-setup-hook 'my--defer-garbage-collection)
  237. (add-hook 'minibuffer-exit-hook 'my--restore-garbage-collection)
  238. (setq read-process-output-max 1mb) ;; lsp-mode's performance suggest
  239. #+end_src
  240. ** File Handler
  241. #+begin_src emacs-lisp :tangle early-init.el
  242. (defvar default-file-name-handler-alist file-name-handler-alist)
  243. (setq file-name-handler-alist nil)
  244. (add-hook 'emacs-startup-hook
  245. (lambda ()
  246. (setq file-name-handler-alist default-file-name-handler-alist)) 100)
  247. #+end_src
  248. ** Others
  249. #+begin_src emacs-lisp :tangle early-init.el
  250. ;; Resizing the emacs frame can be a terriblu expensive part of changing the font.
  251. ;; By inhibiting this, we easily hale startup times with fonts that are larger
  252. ;; than the system default.
  253. (setq package-enable-at-startup nil)
  254. (setq frame-inhibit-implied-resize t)
  255. #+end_src
  256. * Appearance
  257. ** Defaults
  258. #+begin_src emacs-lisp
  259. (set-charset-priority 'unicode)
  260. (setq-default locale-coding-system 'utf-8
  261. default-process-coding-system '(utf-8-unix . utf-8-unix))
  262. (set-terminal-coding-system 'utf-8)
  263. (set-keyboard-coding-system 'utf-8)
  264. (set-selection-coding-system 'utf-8)
  265. (if *sys/windows*
  266. (progn
  267. (prefer-coding-system 'utf-8-dos)
  268. (set-clipboard-coding-system 'utf-16-le)
  269. (set-selection-coding-system 'utf-16-le))
  270. (prefer-coding-system 'utf-8))
  271. (setq-default bidi-paragraph-direction 'left-to-right
  272. bidi-inhibit-bpa t ;; both settings reduce line rescans
  273. uniquify-buffer-name-style 'forward
  274. indent-tabs-mode nil ;; avoid tabs in place of multiple spaces (they look bad in tex)
  275. indicate-empty-lines t ;; show empty lines
  276. scroll-margin 5 ;; smooth scrolling
  277. scroll-conservatively 10000
  278. scroll-preserve-screen-position 1
  279. scroll-step 1
  280. ring-bell-function 'ignore ;; disable pc speaker bell
  281. visible-bell t)
  282. (global-hl-line-mode t) ;; highlight current line
  283. (blink-cursor-mode -1) ;; turn off blinking cursor
  284. (column-number-mode t)
  285. #+end_src
  286. ** Remove redundant UI
  287. #+begin_src emacs-lisp :tangle early-init.el
  288. (menu-bar-mode -1) ;; disable menu bar
  289. (tool-bar-mode -1) ;; disable tool bar
  290. (scroll-bar-mode -1) ;; disable scroll bar
  291. #+end_src
  292. ** Font
  293. #+BEGIN_SRC emacs-lisp
  294. (when *sys/linux*
  295. (set-face-font 'default "Hack-10"))
  296. (when *work_remote*
  297. (set-face-font 'default "Lucida Sans Typewriter-11"))
  298. #+END_SRC
  299. ** Themes
  300. #+BEGIN_SRC emacs-lisp
  301. (defun my/toggle-theme ()
  302. (interactive)
  303. (when (or *sys/windows* *sys/linux*)
  304. (if (eq (car custom-enabled-themes) 'plastic)
  305. (progn (disable-theme 'plastic)
  306. (load-theme 'leuven))
  307. (progn
  308. (disable-theme 'leuven)
  309. (load-theme 'plastic)))))
  310. (bind-key "C-c t" 'my/toggle-theme)
  311. #+END_SRC
  312. Windows Theme:
  313. #+BEGIN_SRC emacs-lisp
  314. (when *sys/windows*
  315. (mapcar #'disable-theme custom-enabled-themes)
  316. (load-theme 'tango))
  317. (when *sys/linux*
  318. (mapcar #'disable-theme custom-enabled-themes)
  319. (load-theme 'plastic))
  320. #+END_SRC
  321. ** line wrappings
  322. #+BEGIN_SRC emacs-lisp
  323. (global-visual-line-mode)
  324. ;(diminish 'visual-line-mode)
  325. (use-package adaptive-wrap
  326. :ensure t
  327. :hook
  328. (visual-line-mode . adaptive-wrap-prefix-mode))
  329. ; :init
  330. ; (when (fboundp 'adaptive-wrap-prefix-mode)
  331. ; (defun me/activate-adaptive-wrap-prefix-mode ()
  332. ; "Toggle `visual-line-mode' and `adaptive-wrap-prefix-mode' simultaneously."
  333. ; (adaptive-wrap-prefix-mode (if visual-line-mode 1 -1)))
  334. ; (add-hook 'visual-line-mode-hook 'me/activate-adaptive-wrap-prefix-mode)))
  335. #+END_SRC
  336. ** line numbers
  337. #+BEGIN_SRC emacs-lisp
  338. (use-package display-line-numbers
  339. :ensure nil
  340. :init
  341. :hook
  342. ((prog-mode
  343. org-src-mode) . display-line-numbers-mode)
  344. :config
  345. (setq-default display-line-numbers-type 'visual
  346. display-line-numbers-current-absolute t
  347. display-line-numbers-with 4
  348. display-line-numbers-widen t))
  349. #+END_SRC
  350. ** misc
  351. Delight can replace mode names with custom names ,
  352. e.g. python-mode with just "π ".
  353. #+BEGIN_SRC emacs-lisp
  354. (use-package rainbow-mode
  355. :ensure t
  356. :diminish
  357. :hook
  358. ((org-mode
  359. emacs-lisp-mode) . rainbow-mode))
  360. (use-package delight
  361. :if *sys/linux*
  362. :ensure t)
  363. (show-paren-mode t) ;; show other part of brackets
  364. (setq blink-matching-paren nil) ;; not necessary with show-paren-mode, bugs out on C-s counsel-line
  365. (use-package rainbow-delimiters
  366. :ensure t
  367. :hook
  368. (prog-mode . rainbow-delimiters-mode))
  369. #+END_SRC
  370. * dired
  371. #+begin_src emacs-lisp
  372. (use-package dired
  373. :ensure nil
  374. :custom
  375. (dired-kill-when-opening-new-dired-buffer t))
  376. #+end_src
  377. * Bookmarks
  378. Usage:
  379. - C-x r m (bookmark-set): add bookmark
  380. - C-x r l (list-bookmark): list bookmarks
  381. - C-x r b (bookmark-jump): open bookmark
  382. Edit bookmarks (while in bookmark file):
  383. - d: mark current item
  384. - x: delete marked items
  385. - r: rename current item
  386. - s: save changes
  387. #+begin_src emacs-lisp
  388. (use-package bookmark
  389. :ensure nil
  390. :custom
  391. (bookmark-default-file (concat MY--PATH_USER_LOCAL "bookmarks")))
  392. ;;do I really want this?
  393. (use-package bookmark+
  394. :ensure (:host github :repo "emacsmirror/bookmark-plus"))
  395. #+end_src
  396. Some windows specific stuff
  397. #+BEGIN_SRC emacs-lisp
  398. (when *sys/windows*
  399. (remove-hook 'find-file-hook 'vc-refresh-state)
  400. ; (progn
  401. ; (setq gc-cons-threshold (* 511 1024 1024)
  402. ; gc-cons-percentage 0.5
  403. ; garbage-collection-messages t
  404. ; (run-with-idle-timer 5 t #'garbage-collect))
  405. (when (boundp 'w32-pipe-read-delay)
  406. (setq w32-pipe-read-delay 0))
  407. (when (boundp 'w32-get-true-file-attributes)
  408. (setq w32-get-true-file-attributes nil)))
  409. #+END_SRC
  410. * burly
  411. [[https://github.com/alphapapa/burly.el][Github]]
  412. Store window configuration and save them as a bookmark
  413. burly-bookmark-windows: bookmarks the current window layout
  414. #+begin_src emacs-lisp
  415. (use-package burly
  416. :ensure t
  417. :config
  418. (burly-tabs-mode) ;;open a burly window bookbark in a new tab
  419. )
  420. #+end_src
  421. * recentf
  422. Exclude some dirs from spamming recentf
  423. #+begin_src emacs-lisp
  424. (use-package recentf
  425. :ensure nil
  426. ; :defer 1
  427. :config
  428. (recentf-mode)
  429. :custom
  430. (recentf-exclude '(".*-autoloads\\.el\\'"
  431. "[/\\]\\elpa/"
  432. "COMMIT_EDITMSG\\'"))
  433. (recentf-save-file (concat MY--PATH_USER_LOCAL "recentf"))
  434. (recentf-max-menu-items 600)
  435. (recentf-max-saved-items 600))
  436. #+end_src
  437. * savehist
  438. #+begin_src emacs-lisp
  439. (use-package savehist
  440. :ensure nil
  441. :config
  442. (savehist-mode)
  443. :custom
  444. (savehist-file (concat MY--PATH_USER_LOCAL "history")))
  445. #+end_src
  446. * undo
  447. #+BEGIN_SRC emacs-lisp
  448. (use-package undo-tree
  449. :ensure t
  450. :diminish undo-tree-mode
  451. :init
  452. (global-undo-tree-mode 1)
  453. :custom
  454. (undo-tree-auto-save-history nil))
  455. #+END_SRC
  456. * COMMENT ace-window (now avy)
  457. #+begin_src emacs-lisp
  458. (use-package ace-window
  459. :ensure t
  460. :bind
  461. (:map global-map
  462. ("C-x o" . ace-window)))
  463. #+end_src
  464. * which-key
  465. #+BEGIN_SRC emacs-lisp
  466. (use-package which-key
  467. :ensure t
  468. :diminish which-key-mode
  469. :custom
  470. (which-key-idle-delay 0.5)
  471. (which-key-sort-order 'which-key-description-order)
  472. :config
  473. (which-key-mode)
  474. (which-key-setup-side-window-bottom))
  475. #+END_SRC
  476. * abbrev
  477. #+begin_src emacs-lisp
  478. (use-package abbrev
  479. :ensure nil
  480. :diminish abbrev-mode
  481. :hook
  482. ((text-mode org-mode) . abbrev-mode)
  483. :init
  484. (setq abbrev-file-name (concat MY--PATH_USER_GLOBAL "abbrev_tables.el"))
  485. :config
  486. (if (file-exists-p abbrev-file-name)
  487. (quietly-read-abbrev-file))
  488. (setq save-abbrevs 'silently)) ;; don't bother me with asking for abbrev saving
  489. #+end_src
  490. * imenu-list
  491. A minor mode to show imenu in a sidebar.
  492. Call imenu-list-smart-toggle.
  493. [[https://github.com/bmag/imenu-list][Source]]
  494. #+BEGIN_SRC emacs-lisp
  495. (use-package imenu-list
  496. :ensure t
  497. :demand t ; otherwise mode loads too late and won't work on first file it's being activated on
  498. :config
  499. (setq imenu-list-focus-after-activation t
  500. imenu-list-auto-resize t
  501. imenu-list-position 'right)
  502. :general
  503. ([f9] 'imenu-list-smart-toggle)
  504. (:states '(normal insert)
  505. :keymaps 'imenu-list-major-mode-map
  506. "RET" '(imenu-list-goto-entry :which-key "goto")
  507. "TAB" '(hs-toggle-hiding :which-key "collapse")
  508. "v" '(imenu-list-display-entry :which-key "show") ; also prevents visual mode
  509. "q" '(imenu-list-quit-window :which-key "quit"))
  510. :custom
  511. (org-imenu-depth 4))
  512. #+END_SRC
  513. * COMMENT Evil
  514. See also
  515. https://github.com/noctuid/evil-guide
  516. Use C-z (evil-toggle-key) to switch between evil and emacs keybindings,
  517. in case evil is messing something up.
  518. #+BEGIN_SRC emacs-lisp
  519. (use-package evil
  520. :ensure t
  521. :defer .1
  522. :custom
  523. (evil-want-C-i-jump nil) ;; prevent evil from blocking TAB in org tree expanding
  524. (evil-want-integration t)
  525. (evil-want-keybinding nil)
  526. :config
  527. ;; example for using emacs default key map in a certain mode
  528. ;; (evil-set-initial-state 'dired-mode 'emacs)
  529. (evil-mode 1))
  530. #+END_SRC
  531. * Eldoc
  532. use builtin version
  533. #+begin_src emacs-lisp
  534. (use-package eldoc
  535. :ensure nil
  536. :diminish eldoc-mode
  537. :defer t)
  538. #+end_src
  539. * COMMENT Eldoc Box
  540. Currently corfu-popupinfo displays eldoc in highlighted completion candidate. Maybe that's good enough.
  541. #+begin_src emacs-lisp
  542. (use-package eldoc-box
  543. :ensure t)
  544. #+end_src
  545. * Meow
  546. #+begin_src emacs-lisp
  547. (use-package meow
  548. :ensure t
  549. :config
  550. (setq meow-cheatsheet-layout meow-cheatsheet-layout-qwerty)
  551. (meow-motion-overwrite-define-key
  552. '("j" . meow-next)
  553. '("k" . meow-prev)
  554. '("<escape>" . ignore))
  555. (meow-leader-define-key
  556. ;; SPC j/k will run the original command in MOTION state.
  557. '("j" . "H-j")
  558. '("k" . "H-k")
  559. ;; Use SPC (0-9) for digit arguments.
  560. '("1" . meow-digit-argument)
  561. '("2" . meow-digit-argument)
  562. '("3" . meow-digit-argument)
  563. '("4" . meow-digit-argument)
  564. '("5" . meow-digit-argument)
  565. '("6" . meow-digit-argument)
  566. '("7" . meow-digit-argument)
  567. '("8" . meow-digit-argument)
  568. '("9" . meow-digit-argument)
  569. '("0" . meow-digit-argument)
  570. '("/" . meow-keypad-describe-key)
  571. '("?" . meow-cheatsheet))
  572. (meow-normal-define-key
  573. '("0" . meow-expand-0)
  574. '("9" . meow-expand-9)
  575. '("8" . meow-expand-8)
  576. '("7" . meow-expand-7)
  577. '("6" . meow-expand-6)
  578. '("5" . meow-expand-5)
  579. '("4" . meow-expand-4)
  580. '("3" . meow-expand-3)
  581. '("2" . meow-expand-2)
  582. '("1" . meow-expand-1)
  583. '("-" . negative-argument)
  584. '(";" . meow-reverse)
  585. '("," . meow-inner-of-thing)
  586. '("." . meow-bounds-of-thing)
  587. '("[" . meow-beginning-of-thing)
  588. '("]" . meow-end-of-thing)
  589. '("a" . meow-append)
  590. '("A" . meow-open-below)
  591. '("b" . meow-back-word)
  592. '("B" . meow-back-symbol)
  593. '("c" . meow-change)
  594. '("d" . meow-delete)
  595. '("D" . meow-backward-delete)
  596. '("e" . meow-next-word)
  597. '("E" . meow-next-symbol)
  598. '("f" . meow-find)
  599. '("g" . meow-cancel-selection)
  600. '("G" . meow-grab)
  601. '("h" . meow-left)
  602. '("H" . meow-left-expand)
  603. '("i" . meow-insert)
  604. '("I" . meow-open-above)
  605. '("j" . meow-next)
  606. '("J" . meow-next-expand)
  607. '("k" . meow-prev)
  608. '("K" . meow-prev-expand)
  609. '("l" . meow-right)
  610. '("L" . meow-right-expand)
  611. '("m" . meow-join)
  612. '("n" . meow-search)
  613. '("o" . meow-block)
  614. '("O" . meow-to-block)
  615. '("p" . meow-yank)
  616. '("q" . meow-quit)
  617. '("Q" . meow-goto-line)
  618. '("r" . meow-replace)
  619. '("R" . meow-swap-grab)
  620. '("s" . meow-kill)
  621. '("t" . meow-till)
  622. '("u" . meow-undo)
  623. '("U" . meow-undo-in-selection)
  624. '("v" . meow-visit)
  625. '("w" . meow-mark-word)
  626. '("W" . meow-mark-symbol)
  627. '("x" . meow-line)
  628. '("X" . meow-goto-line)
  629. '("y" . meow-save)
  630. '("Y" . meow-sync-grab)
  631. '("z" . meow-pop-selection)
  632. '("'" . repeat)
  633. '("<escape>" . ignore))
  634. ; :config
  635. (meow-global-mode t))
  636. #+end_src
  637. * avy
  638. Search, move, copy, delete text within all visible buffers.
  639. Also replaces ace-window for buffer switching.
  640. [[https://github.com/abo-abo/avy]]
  641. #+BEGIN_SRC emacs-lisp
  642. (use-package avy
  643. :ensure t
  644. :general
  645. (:prefix "M-s"
  646. "" '(:ignore t :which-key "avy")
  647. "w" '(avy-goto-char-2 :which-key "avy-jump")
  648. "s" '(avy-goto-char-timer :which-key "avy-timer")
  649. "c" '(:ignore t :which-key "avy copy")
  650. "c l" '(avy-copy-line :which-key "avy copy line")
  651. "c r" '(avy-copy-region :which-key "avy copy region")
  652. "m" '(:ignore t :which-key "avy move")
  653. "m l" '(avy-move-line :which-key "avy move line")
  654. "m r" '(avy-move-region :which-key "avy move region")))
  655. #+END_SRC
  656. * Vertico
  657. Vertico is a completion ui for the minibuffer.
  658. [[https://github.com/minad/vertico][Vertico Github]]
  659. #+begin_src emacs-lisp
  660. ;; completion ui
  661. (use-package vertico
  662. :ensure t
  663. :init
  664. (vertico-mode))
  665. #+end_src
  666. * Corfu
  667. Completion ui for the buffer.
  668. [[https://github.com/minad/corfu][Corfu Github]]
  669. #+begin_src emacs-lisp
  670. (use-package corfu
  671. :ensure t
  672. :after savehist
  673. :custom
  674. (corfu-popupinfo-delay t)
  675. (corfu-auto t)
  676. (corfu-cycle t)
  677. (corfu-auto-delay 0.3)
  678. (corfu-preselect-first nil)
  679. (corfu-popupinfo-delay '(1.0 . 0.0)) ;1s for first popup, instant for subsequent popups
  680. (corfu-popupinfo-max-width 70)
  681. (corfu-popupinfo-max-height 20)
  682. :init
  683. (global-corfu-mode)
  684. ; (corfu-popupinfo-mode) ; causes corfu window to stay
  685. (corfu-history-mode)
  686. ;; belongs to emacs
  687. (add-to-list 'savehist-additional-variables 'corfu-history)
  688. :hook
  689. (corfu-mode . corfu-popupinfo-mode))
  690. ; :bind
  691. ; (:map corfu-map
  692. ; ("TAB" . corfu-next)
  693. ; ("<C-return>" . corfu-insert)
  694. ; ("C-TAB" . corfu-popupinfo-toggle)))
  695. ;; (general-define-key
  696. ;; :states 'insert
  697. ;; :definer 'minor-mode
  698. ;; :keymaps 'completion-in-region-mode
  699. ;; :predicate 'corfu-mode
  700. ;; "C-d" 'corfu-info-documentation)
  701. (use-package emacs
  702. :ensure nil
  703. :init
  704. ;; hide commands in M-x which do not apply to current mode
  705. (setq read-extended-command-predicate #'command-completion-default-include-p)
  706. ;; enable indentation + completion using TAB
  707. (setq tab-always-indent 'complete))
  708. #+end_src
  709. * Cape
  710. [[https://github.com/minad/cape][Cape Github]]
  711. Backend completions for the buffer (not minibuffer).
  712. Available functions:
  713. dabbrev, file, history, keyword, tex, sgml, rfc1345, abbrev, ispell, dict, symbol, line
  714. #+begin_src emacs-lisp
  715. (use-package cape
  716. :ensure t
  717. :bind
  718. (("C-c p p" . completion-at-point) ;; capf
  719. ("C-c p t" . complete-tag) ;; etags
  720. ("C-c p d" . cape-dabbrev)
  721. ("C-c p h" . cape-history)
  722. ("C-c p f" . cape-file))
  723. :init
  724. (advice-add #'lsp-completion-at-point :around #'cape-wrap-noninterruptible) ;; for performance issues with lsp
  725. (add-to-list 'completion-at-point-functions #'cape-dabbrev)
  726. (add-to-list 'completion-at-point-functions #'cape-file)
  727. (add-to-list 'completion-at-point-functions #'cape-history))
  728. #+end_src
  729. * kind-icon
  730. Make corfu pretty
  731. [[https://github.com/jdtsmith/kind-icon][kind-icon Github]]
  732. #+begin_src emacs-lisp
  733. (use-package kind-icon
  734. :ensure t
  735. :after corfu
  736. :custom
  737. (kind-icon-default-face 'corfu-default) ;; to compute blended backgrounds correctly
  738. :config
  739. (add-to-list 'corfu-margin-formatters #'kind-icon-margin-formatter))
  740. #+end_src
  741. * Orderless
  742. [[https://github.com/oantolin/orderless][Orderless Github]]
  743. Orderless enables customizing the way how input is completed, orders the suggestions by recency. Works for vertico and corfu.
  744. #+begin_src emacs-lisp
  745. (use-package orderless
  746. :ensure t
  747. :init
  748. (setq completion-styles '(orderless partial-completion basic)
  749. completion-category-defaults nil
  750. completion-category-overrides nil))
  751. ; completion-category-overrides '((file (styles partial-completion)))))
  752. #+end_src
  753. * Consult
  754. [[https://github.com/minad/consult][Github]]
  755. Backend completion functions for the minibuffer.
  756. Default preview key: M-.
  757. #+begin_src emacs-lisp
  758. (use-package consult
  759. :ensure t
  760. :bind
  761. (("C-x C-r" . consult-recent-file)
  762. ("C-x b" . consult-buffer)
  763. ("C-s" . consult-line)
  764. ("C-x r b" . consult-bookmark)) ;replace bookmark-jump
  765. :config
  766. ;; disable preview for some commands and buffers
  767. ;; and enable it by M-.
  768. ;; see https://github.com/minad/consult#use-package-example
  769. (consult-customize
  770. consult-theme :preview-key '(debounce 0.2 any)
  771. consult-ripgrep consult-git-grep consult-grep
  772. consult-bookmark consult-recent-file consult-xref
  773. consult--source-bookmark consult--source-file-register
  774. consult--source-recent-file consult--source-project-recent-file
  775. :preview-key '(:debounce 0.2 any)))
  776. #+end_src
  777. * Marginalia
  778. [[https://github.com/minad/marginalia/][Github]]
  779. Adds additional information/annotations to the minibuffer
  780. #+begin_src emacs-lisp
  781. (use-package marginalia
  782. :ensure t
  783. :init
  784. (marginalia-mode)
  785. :bind
  786. (:map minibuffer-local-map
  787. ("M-A" . marginalia-cycle))
  788. :custom
  789. ;; switch by 'marginalia-cycle
  790. (marginalia-annotators '(marginalia-annotators-heavy
  791. marginalia-annotators-light
  792. nil)))
  793. #+end_src
  794. * Embark
  795. Actions on the completion buffer.
  796. #+begin_src emacs-lisp
  797. (use-package embark
  798. :ensure t
  799. :bind
  800. (("C-S-a" . embark-act)
  801. ("C-h B" . embark-bindings))
  802. :init
  803. (setq prefix-help-command #'embark-prefix-help-command)
  804. :config
  805. ;; hide modeline of the embark live/completions buffers
  806. (add-to-list 'display-buffer-alist
  807. '("\\`\\*Embark Collect \\(Live\\|Completions\\)\\*"
  808. nil
  809. (window-parameters (mode-line-format . none)))))
  810. (use-package embark-consult
  811. :ensure t
  812. :after (embark consult)
  813. :demand t
  814. :hook
  815. (embark-collect-mode . embark-consult-preview-minor-mode))
  816. #+end_src
  817. * Tree-sitter
  818. #+begin_src emacs-lisp
  819. (when *sys/linux*
  820. (use-package tree-sitter
  821. :ensure t
  822. :init
  823. (global-tree-sitter-mode t)
  824. :hook
  825. (tree-sitter-after-on . tree-sitter-hl-mode))
  826. (use-package tree-sitter-langs
  827. :ensure t
  828. :after tree-sitter)
  829. )
  830. #+end_src
  831. * COMMENT Xeft (needs xapian, not really windows compatible)
  832. Fast full text search for stuff org-ql cannot cover
  833. #+begin_src emacs-lisp
  834. (use-package xeft
  835. :ensure t
  836. :custom
  837. (xeft-recursive 'follow-symlinks))
  838. #+end_src
  839. * COMMENT Helm
  840. As an alternative if I'm not happy with selectrum & co
  841. #+begin_src emacs-lisp
  842. (use-package helm
  843. :ensure t
  844. :hook
  845. (helm-mode . helm-autoresize-mode)
  846. ;; :bind
  847. ;; (("M-x" . helm-M-x)
  848. ;; ("C-s" . helm-occur)
  849. ;; ("C-x C-f" . helm-find-files)
  850. ;; ("C-x C-b" . helm-buffers-list)
  851. ;; ("C-x b" . helm-buffers-list)
  852. ;; ("C-x C-r" . helm-recentf)
  853. ;; ("C-x C-i" . helm-imenu))
  854. :config
  855. (helm-mode)
  856. :custom
  857. (helm-split-window-inside-p t) ;; open helm buffer inside current window
  858. (helm-move-to-line-cycle-in-source t)
  859. (helm-echo-input-in-header-line t)
  860. (helm-autoresize-max-height 20)
  861. (helm-autoresize-min-height 5)
  862. )
  863. #+end_src
  864. * Emails
  865. Requires on system
  866. - isync to sync emails between host and local
  867. - mu / mu4e4e (apt nstall mu4e/bookworm-backports)
  868. After installing mu4e it needs to be initialized
  869. #+begin_src shell :tangle no
  870. mu init --maildir=/path/to/mail/folder --my-address=mu@adress.com --my-address=another@adress.com...
  871. mu index
  872. #+end_src
  873. for credentials see
  874. https://www.emacswiki.org/emacs/SmtpAuth
  875. #+begin_src emacs-lisp
  876. ;(use-package notmuch
  877. ; :ensure t)
  878. (use-feature mu4e
  879. :if *sys/linux*
  880. ; :ensure nil
  881. :after (org)
  882. :init
  883. (require 'mu4e)
  884. :custom
  885. ;; this is set to 't' to avoid mail syncing issues when using mbsync
  886. (mu4e-change-filenames-when-moving t)
  887. (smtpmail-servers-requiring-authorization ".*")
  888. (mu4e-update-interval (* 10 60))
  889. (mu4e-get-mail-command "mbsync -a")
  890. (mu4e-maildir "~/archiv/Dokumente/email")
  891. (mu4e-attachment-dir "/mnt/backup/downloads")
  892. (mu4e-org-support t)
  893. (mu4e-completion-read-function 'completion-read)
  894. (mu4e-use-fancy-chars t)
  895. (mu4e-view-show-addresses t)
  896. (mu4e-view-show-images t)
  897. (mu4e-sent-messages-behaviour 'sent) ;works better with mbsync
  898. (mu4e-headers-fields
  899. '((:human-date . 12)
  900. (:flags . 6)
  901. (:from . 22)
  902. (:to . 25)
  903. (:subject)))
  904. :config
  905. (setq mu4e-contexts
  906. `( ,(make-mu4e-context
  907. :name "mail.de"
  908. :enter-func (lambda () (mu4e-message "switch to mail.de context"))
  909. :match-func (lambda (msg)
  910. (when msg
  911. (string-match-p "^/mailde" (mu4e-message-field msg :maildir))))
  912. :vars '((user-mail-address . "marc.pohling@mail.de")
  913. (user-full-name . "Marc Pohling")
  914. (message-send-mail-function . smtpmail-send-it)
  915. (smtpmail-smtp-user . "marc.pohling@mail.de")
  916. (smtpmail-smtp-server . "smtp.mail.de")
  917. (smtpmail-smtp-service . 587)
  918. (smtpmail-stream-type . starttls)
  919. (mu4e-drafts-folder . "/mailde/drafts")
  920. (mu4e-sent-folder . "/mailde/sent")
  921. (mu4e-refile-folder . "/mailde/archive")
  922. (mu4e-trash-folder . "/mailde/trash")))
  923. ,(make-mu4e-context
  924. :name "web.de"
  925. :enter-func (lambda () (mu4e-message "switch to web.de context"))
  926. :match-func (lambda (msg)
  927. (when msg
  928. (string-match-p "^/mailde" (mu4e-message-field msg :maildir))))
  929. :vars '((user-mail-address . "marc.pohling@web.de")
  930. (user-full-name . "Marc Pohling")
  931. (mu4e-drafts-folder . "/webde/drafts")
  932. (mu4e-sent-folder . "/webde/sent")
  933. (mu4e-refile-folder . "/webde/archive")
  934. (mu4e-trash-folder . "/webde/trash")))
  935. ,(make-mu4e-context
  936. :name "gmail loui"
  937. :enter-func (lambda () (mu4e-message "switch to gmail loui context"))
  938. :match-func (lambda (msg)
  939. (when msg
  940. (string-match-p "^/gmailloui" (mu4e-message-field msg :maildir))))
  941. :vars '((user-mail-address . "louithelou1@gmail.com")
  942. (user-full-name . "Loui")
  943. (message-send-mail-function . smtpmail-send-it)
  944. (smtpmail-smtp-user . "louithelou1@gmail.com")
  945. (smtpmail-smtp-server . "smtp.gmail.com")
  946. (smtpmail-smtp-service . 587)
  947. (smtpmail-stream-type . starttls)
  948. (mu4e-drafts-folder . "/gmailloui/drafts")
  949. (mu4e-sent-folder . "/gmailloui/sent")
  950. (mu4e-refile-folder . "/gmailloui/archive")
  951. (mu4e-trash-folder . "/gmailloui/trash")))))
  952. )
  953. #+end_src
  954. mu4e-dashboard
  955. https://github.com/rougier/mu4e-dashboard
  956. #+begin_src emacs-lisp
  957. (push 'mu4e elpaca-ignored-dependencies)
  958. (use-package mu4e-dashboard
  959. :if *sys/linux*
  960. :ensure (:host github :repo "rougier/mu4e-dashboard")
  961. :after mu4e
  962. :hook
  963. (mu4e-dashboard-mode . (lambda () (display-line-numbers-mode -1)))
  964. :custom
  965. (mu4e-dashboard-file (concat MY--PATH_USER_GLOBAL "mu4e-dashboard.org"))
  966. :config
  967. ; (require 'mu4e)
  968. (defun mu4e-dashboard-edit ()
  969. (interactive)
  970. (let ((edit-buffer "*edit-mu4e-dashboard*"))
  971. (when (get-buffer edit-buffer)
  972. (kill-buffer (get-buffer edit-buffer)))
  973. (make-indirect-buffer (current-buffer) edit-buffer)
  974. (switch-to-buffer-other-window (get-buffer edit-buffer))
  975. (org-mode 1)))
  976. (display-line-numbers-mode -1)
  977. (flyspell-mode -1))
  978. #+end_src
  979. * outlook
  980. In outlook a macro is necessary, also a reference to FM20.DLL
  981. (Microsoft Forms 2.0 Object Library, in c:\windows\syswow64\fm20.dll)
  982. The macro copies the GUID of the email to the clipboard
  983. Attention: the GUID changes when the email is moved to another folder!
  984. The macro:
  985. #+BEGIN_SRC
  986. Sub AddLinkToMessageInClipboard()
  987. 'Adds a link to the currently selected message to the clipboard
  988. Dim objMail As Outlook.MailItem
  989. Dim doClipboard As New DataObject
  990. 'One and ONLY one message muse be selected
  991. If Application.ActiveExplorer.Selection.Count <> 1 Then
  992. MsgBox ("Select one and ONLY one message.")
  993. Exit Sub
  994. End If
  995. Set objMail = Application.ActiveExplorer.Selection.Item(1)
  996. doClipboard.SetText "[[outlook:" + objMail.EntryID + "][MESSAGE: " + objMail.Subject + " (" + objMail.SenderName + ")]]"
  997. doClipboard.PutInClipboard
  998. End Sub
  999. #+END_SRC
  1000. #+BEGIN_SRC emacs-lisp
  1001. ;(org-add-link-type "outlook" 'my--org-outlook-open)
  1002. ;(org-add-link-type "outlooknewmail" 'my--org-outlook-new-mail)
  1003. (defun my--org-outlook-open (id)
  1004. (w32-shell-execute "open" "outlook" (concat " /select outlook:" id)))
  1005. (defun my--org-outlook-new-mail (receipients)
  1006. ; separate RECEIPIENTS for main, cc and bcc by |
  1007. (setq recp (split-string receipients "|"))
  1008. (setq mailstring (concat " /c ipm.note /m " (nth 0 recp)))
  1009. (if (nth 1 recp)
  1010. ; this check has tp be separated because (and..) would stumble over a nil
  1011. (if (not (equal "" (nth 1 recp)))
  1012. (setq mailstring (concat mailstring "&cc=" (nth 1 recp)))))
  1013. (if (nth 2 recp)
  1014. (setq mailstring (concat mailstring "&bcc=" (nth 2 recp))))
  1015. (w32-shell-execute "open" "outlook" mailstring))
  1016. (defun my/org-outlook-open-test ()
  1017. (interactive)
  1018. (w32-shell-execute "open" "outlook" " /select outlook:000000008A209C397CEF2C4FBA9E54AEB5B1F97F0700846D043B407C5B43A0C05AFC46DC5C630587BE5E020900006E48FF8F6027694BA6593777F542C19E0002A6434D000000"))'
  1019. #+END_SRC
  1020. * misc
  1021. #+begin_src emacs-lisp
  1022. (use-package autorevert
  1023. :diminish auto-revert-mode)
  1024. #+end_src
  1025. * orgmode
  1026. ** some notes
  1027. *** copy file path within emacs
  1028. Enter dired-other-window
  1029. place cursor on the file
  1030. M-0 w (copy absolute path)
  1031. C-u w (copy relative path)
  1032. *** Archiving
  1033. C-c C-x C-a
  1034. To keep the subheading structure when archiving, set the properties of the superheading.
  1035. #+begin_src org :tangle no
  1036. ,* FOO
  1037. :PROPERTIES:
  1038. :ARCHIVE: %s_archive::* FOO
  1039. ,** DONE BAR
  1040. ,** TODO BAZ
  1041. #+end_src
  1042. When moving BAR to archive, it will go to FILENAME.org_archive below the heading FOO.
  1043. [[http://doc.endlessparentheses.com/Var/org-archive-location.html][Other examples]]
  1044. ** org
  1045. This seems necessary to prevent 'org is already installed' error
  1046. https://github.com/jwiegley/use-package/issues/319
  1047. #+begin_src emacs-lisp
  1048. ;(assq-delete-all 'org package--builtins)'
  1049. ;(assq-delete-all 'org package--builtin-versions)
  1050. #+end_src
  1051. #+BEGIN_SRC emacs-lisp
  1052. (defun my--buffer-prop-set (name value)
  1053. "Set a file property called NAME to VALUE in buffer file.
  1054. If the property is already set, replace its value."
  1055. (setq name (downcase name))
  1056. (org-with-point-at 1
  1057. (let ((case-fold-search t))
  1058. (if (re-search-forward (concat "^#\\+" name ":\\(.*\\)")
  1059. (point-max) t)
  1060. (replace-match (concat "#+" name ": " value) 'fixedcase)
  1061. (while (and (not (eobp))
  1062. (looking-at "^[#:]"))
  1063. (if (save-excursion (end-of-line) (eobp))
  1064. (progn
  1065. (end-of-line)
  1066. (insert "\n"))
  1067. (forward-line)
  1068. (beginning-of-line)))
  1069. (insert "#+" name ": " value "\n")))))
  1070. (defun my--buffer-prop-remove (name)
  1071. "Remove a buffer property called NAME."
  1072. (org-with-point-at 1
  1073. (when (re-search-forward (concat "\\(^#\\+" name ":.*\n?\\)")
  1074. (point-max) t)
  1075. (replace-match ""))))
  1076. (use-package org
  1077. :ensure t
  1078. ; :pin gnu
  1079. :mode (("\.org$" . org-mode))
  1080. :diminish org-indent-mode
  1081. :defer 1
  1082. :hook
  1083. (org-mode . org-indent-mode)
  1084. (org-source-mode . smartparens-mode)
  1085. :bind (("C-c l" . org-store-link)
  1086. ("C-c c" . org-capture)
  1087. ("C-c a" . org-agenda)
  1088. :map org-mode-map ("S-<right>" . org-shiftright)
  1089. ("S-<left>" . org-shiftleft))
  1090. :init
  1091. (defun my--org-agenda-files-set ()
  1092. "Sets default agenda files.
  1093. Necessary when updating roam agenda todos."
  1094. (setq org-agenda-files (list (concat MY--PATH_ORG_FILES "notes.org")
  1095. (concat MY--PATH_ORG_FILES "projects.org")
  1096. (concat MY--PATH_ORG_FILES "tasks.org")))
  1097. (when *sys/linux*
  1098. (nconc org-agenda-files
  1099. (directory-files-recursively MY--PATH_ORG_FILES_MOBILE "\\.org$"))))
  1100. (my--org-agenda-files-set)
  1101. (defun my--org-skip-subtree-if-priority (priority)
  1102. "Skip an agenda subtree if it has a priority of PRIORITY.
  1103. PRIORITY may be one of the characters ?A, ?B, or ?C."
  1104. (let ((subtree-end (save-excursion (org-end-of-subtree t)))
  1105. (pri-value (* 1000 (- org-lowest-priority priority)))
  1106. (pri-current (org-get-priority (thing-at-point 'line t))))
  1107. (if (= pri-value pri-current)
  1108. subtree-end
  1109. nil)))
  1110. :config
  1111. (when *work_remote*
  1112. (org-add-link-type "outlook" 'my--org-outlook-open)
  1113. (org-add-link-type "outlooknewmail" 'my--org-outlook-new-mail)
  1114. (setq org-todo-keywords
  1115. '((sequence "WAIT" "OPEN" "NEXT" "TODO" "|" "DONE" "DROP")))
  1116. (setq org-capture-templates
  1117. '(("t" "telephone call" entry
  1118. ; (file+olp+datetree (concat MY--PATH_ORG_FILES "phone_calls.org"))
  1119. (file+datetree "p:/Eigene Dateien/Notizen/phone_calls.org")
  1120. "* [%<%Y-%m-%d %H:%M>] %?"
  1121. :empty-lines 0 :jump-to-captured t))))
  1122. (when *sys/linux*
  1123. (setq org-pretty-entities t))
  1124. :custom
  1125. (org-startup-truncated t)
  1126. (org-startup-align-all-tables t)
  1127. (org-src-fontify-natively t) ;; use syntax highlighting in code blocks
  1128. (org-src-preserve-indentation t) ;; no extra indentation
  1129. (org-src-window-setup 'current-window) ;; C-c ' opens in current window
  1130. (org-modules (quote (org-id
  1131. org-habit
  1132. org-tempo))) ;; easy templates
  1133. (org-default-notes-file (concat MY--PATH_ORG_FILES "notes.org"))
  1134. (org-id-locations-file (concat MY--PATH_USER_LOCAL ".org-id-locations"))
  1135. (org-log-into-drawer "LOGBOOK")
  1136. (org-log-done 'time) ;; create timestamp when task is done
  1137. (org-blank-before-new-entry '((heading) (plain-list-item))) ;; prevent new line before new item
  1138. (org-src-tab-acts-natively t)
  1139. ;;Sort agenda by deadline and priority
  1140. (org-agenda-sorting-strategy
  1141. (quote
  1142. ((agenda deadline-up priority-down)
  1143. (todo priority-down category-keep)
  1144. (tags priority-down category-keep)
  1145. (search category-keep))))
  1146. (org-agenda-custom-commands
  1147. '(("c" "Simple agenda view"
  1148. ((tags "PRIORITY=\"A\""
  1149. ((org-agenda-skip-function '(org-agenda-skip-entry-if 'todo 'done))
  1150. (org-agenda-overriding-header "Hohe Priorität:")))
  1151. (agenda ""
  1152. ((org-agenda-skip-function '(org-agenda-skip-entry-if 'todo 'done))
  1153. (org-agenda-span 7)
  1154. (org-agenda-start-on-weekday nil)
  1155. (org-agenda-overriding-header "Nächste 7 Tage:")))
  1156. (alltodo ""
  1157. ((org-agenda-skip-function '(or (my--org-skip-subtree-if-priority ?A)
  1158. (org-agenda-skip-if nil '(scheduled deadline))))
  1159. (org-agenda-overriding-header "Sonstige Aufgaben:"))))))))
  1160. #+END_SRC
  1161. ** COMMENT languages
  1162. Set some languages and disable confirmation for evaluating code blocks C-c C-c
  1163. Elpaca cant find it, though it's built in org
  1164. #+begin_src emacs-lisp
  1165. (use-package ob-python
  1166. ; :ensure nil
  1167. :defer t
  1168. :after org
  1169. ; :ensure org-contrib
  1170. :commands
  1171. (org-babel-execute:python))
  1172. #+end_src
  1173. ** COMMENT habits
  1174. #+BEGIN_SRC emacs-lisp
  1175. (require 'org-habit) ;;TODO Lösung ohne require finden, scheint mir nicht ideal zu sein, nur um ein org-modul zu aktivieren
  1176. ;; (add-to-list 'org-modules "org-habit")
  1177. (setq org-habit-graph-column 80
  1178. org-habit-preceding-days 30
  1179. org-habit-following-days 7
  1180. org-habit-show-habits-only-for-today nil)
  1181. #+END_SRC
  1182. ** contacts
  1183. #+BEGIN_SRC emacs-lisp
  1184. (use-package org-contacts
  1185. :ensure (:host nil :repo "https://repo.or.cz/org-contacts.git")
  1186. :after org
  1187. :custom
  1188. (org-contacts-files (list (concat MY--PATH_ORG_ROAM "contacts.org"))))
  1189. #+END_SRC
  1190. ** *TODO*
  1191. [[https://github.com/nobiot/org-transclusion][org-transclusion]]?
  1192. ** COMMENT journal
  1193. [[https://github.com/bastibe/org-journal][Source]]
  1194. Ggf. durch org-roam-journal ersetzen
  1195. #+BEGIN_SRC emacs-lisp
  1196. (use-package org-journal
  1197. :if *sys/linux*
  1198. :ensure t
  1199. :defer t
  1200. :config
  1201. ;; feels hacky, but this way compiler error "assignment to free variable" disappears
  1202. (when (and (boundp 'org-journal-dir)
  1203. (boundp 'org-journal-enable-agenda-integration))
  1204. (setq org-journal-dir MY--PATH_ORG_JOURNAl
  1205. org-journal-enable-agenda-integration t)))
  1206. #+END_SRC
  1207. ** org-roam
  1208. [[https://github.com/org-roam/org-roam][Github]]
  1209. Um Headings innerhalb einer Datei zu verlinken:
  1210. - org-id-get-create im Heading,
  1211. - org-roam-node-insert in der verweisenden Datei
  1212. Bei Problemen wie unique constraint
  1213. org-roam-db-clear-all
  1214. org-roam-db-sync
  1215. #+BEGIN_SRC emacs-lisp
  1216. ;(use-package emacsql-sqlite-builtin
  1217. ; :ensure t)
  1218. (use-package org-roam
  1219. ; :requires emacsql-sqlite-builtin
  1220. :ensure t
  1221. :defer 2
  1222. :after org
  1223. :init
  1224. (setq org-roam-v2-ack t)
  1225. (defun my--roamtodo-p ()
  1226. "Return non-nil if current buffer has any todo entry.
  1227. TODO entries marked as done are ignored, meaning this function
  1228. returns nil if current buffer contains only completed tasks."
  1229. (seq-find
  1230. (lambda (type)
  1231. (eq type 'todo))
  1232. (org-element-map
  1233. (org-element-parse-buffer 'headline)
  1234. 'headline
  1235. (lambda (h)
  1236. (org-element-property :todo-type h)))))
  1237. (defun my--roamtodo-update-tag ()
  1238. "Update ROAMTODO tag in the current buffer."
  1239. (when (and (not (active-minibuffer-window))
  1240. (my--buffer-roam-note-p))
  1241. (save-excursion
  1242. (goto-char (point-min))
  1243. (let* ((tags (my--buffer-tags-get))
  1244. (original-tags tags))
  1245. (if (my--roamtodo-p)
  1246. (setq tags (cons "roamtodo" tags))
  1247. (setq tags (remove "roamtodo" tags)))
  1248. ;;cleanup duplicates
  1249. (when (or (seq-difference tags original-tags)
  1250. (seq-difference original-tags tags))
  1251. (apply #'my--buffer-tags-set tags))))))
  1252. (defun my--buffer-tags-get ()
  1253. "Return filetags value in current buffer."
  1254. (my--buffer-prop-get-list "filetags" "[ :]"))
  1255. (defun my--buffer-tags-set (&rest tags)
  1256. "Set TAGS in current buffer.
  1257. If filetags value is already set, replace it."
  1258. (if tags
  1259. (my--buffer-prop-set
  1260. "filetags" (concat ":" (string-join tags ":") ":"))
  1261. (my--buffer-prop-remove "filetags")))
  1262. (defun my--buffer-tags-add (tag)
  1263. "Add a TAG to filetags in current buffer."
  1264. (let* ((tags (my--buffer-tags-get))
  1265. (tags (append tags (list tag))))
  1266. (apply #'my--buffer-tags-set tags)))
  1267. (defun my--buffer-tags-remove (tag)
  1268. "Remove a TAG from filetags in current buffer."
  1269. (let* ((tags (my--buffer-tags-get))
  1270. (tags (delete tag tags)))
  1271. (apply #'my--buffer-tags-set tags)))
  1272. (defun my--buffer-prop-set (name value)
  1273. "Set a file property called NAME to VALUE in buffer file.
  1274. If the property is already set, replace its value."
  1275. (setq name (downcase name))
  1276. (org-with-point-at 1
  1277. (let ((case-fold-search t))
  1278. (if (re-search-forward (concat "^#\\+" name ":\\(.*\\)")
  1279. (point-max) t)
  1280. (replace-match (concat "#+" name ": " value) 'fixedcase)
  1281. (while (and (not (eobp))
  1282. (looking-at "^[#:]"))
  1283. (if (save-excursion (end-of-line) (eobp))
  1284. (progn
  1285. (end-of-line)
  1286. (insert "\n"))
  1287. (forward-line)
  1288. (beginning-of-line)))
  1289. (insert "#+" name ": " value "\n")))))
  1290. (defun my--buffer-prop-set-list (name values &optional separators)
  1291. "Set a file property called NAME to VALUES in current buffer.
  1292. VALUES are quoted and combined into single string using
  1293. `combine-and-quote-strings'.
  1294. If SEPARATORS is non-nil, it should be a regular expression
  1295. matching text that separates, but is not part of, the substrings.
  1296. If nil it defaults to `split-string-and-unquote', normally
  1297. \"[ \f\t\n\r\v]+\", and OMIT-NULLS is forced to t.
  1298. If the property is already set, replace its value."
  1299. (my--buffer-prop-set
  1300. name (combine-and-quote-strings values separators)))
  1301. (defun my--buffer-prop-get (name)
  1302. "Get a buffer property called NAME as a string."
  1303. (org-with-point-at 1
  1304. (when (re-search-forward (concat "^#\\+" name ": \\(.*\\)")
  1305. (point-max) t)
  1306. (buffer-substring-no-properties
  1307. (match-beginning 1)
  1308. (match-end 1)))))
  1309. (defun my--buffer-prop-get-list (name &optional separators)
  1310. "Get a buffer property NAME as a list using SEPARATORS.
  1311. If SEPARATORS is non-nil, it should be a regular expression
  1312. matching text that separates, but is not part of, the substrings.
  1313. If nil it defaults to `split-string-default-separators', normally
  1314. \"[ \f\t\n\r\v]+\", and OMIT-NULLS is forced to t."
  1315. (let ((value (my--buffer-prop-get name)))
  1316. (when (and value (not (string-empty-p value)))
  1317. (split-string-and-unquote value separators))))
  1318. (defun my--buffer-prop-remove (name)
  1319. "Remove a buffer property called NAME."
  1320. (org-with-point-at 1
  1321. (when (re-search-forward (concat "\\(^#\\+" name ":.*\n?\\)")
  1322. (point-max) t)
  1323. (replace-match ""))))
  1324. (defun my--buffer-roam-note-p ()
  1325. "Return non-nil if the currently visited buffer is a note."
  1326. (and buffer-file-name
  1327. (string-prefix-p
  1328. (expand-file-name (file-name-as-directory MY--PATH_ORG_ROAM))
  1329. (file-name-directory buffer-file-name))))
  1330. (defun my--org-roam-filter-by-tag (tag-name)
  1331. (lambda (node)
  1332. (member tag-name (org-roam-node-tags node))))
  1333. (defun my--org-roam-list-notes-by-tag (tag-name)
  1334. (mapcar #'org-roam-node-file
  1335. (seq-filter
  1336. (my--org-roam-filter-by-tag tag-name)
  1337. (org-roam-node-list))))
  1338. (defun my/org-roam-refresh-agenda-list ()
  1339. "Add all org roam files with #+filetags: roamtodo"
  1340. (interactive)
  1341. (my--org-agenda-files-set)
  1342. (nconc org-agenda-files
  1343. (my--org-roam-list-notes-by-tag "roamtodo"))
  1344. (setq org-agenda-files (delete-dups org-agenda-files)))
  1345. (add-hook 'find-file-hook #'my--roamtodo-update-tag)
  1346. (add-hook 'before-save-hook #'my--roamtodo-update-tag)
  1347. (advice-add 'org-agenda :before #'my/org-roam-refresh-agenda-list)
  1348. (advice-add 'org-todo-list :before #'my/org-roam-refresh-agenda-list)
  1349. (add-to-list 'org-tags-exclude-from-inheritance "roamtodo")
  1350. :config
  1351. (require 'org-roam-dailies) ;; ensure the keymap is available
  1352. (org-roam-db-autosync-mode)
  1353. ;; build the agenda list the first ime for the session
  1354. (my/org-roam-refresh-agenda-list)
  1355. (when *work_remote*
  1356. (setq org-roam-capture-templates
  1357. '(("n" "note" plain
  1358. "%?"
  1359. :if-new (file+head "notes/%<%Y%m%d%H%M%S>-${slug}.org" "#+title: ${title}\n")
  1360. :unnarrowed t)
  1361. ("i" "idea" plain
  1362. "%?"
  1363. :if-new (file+head "ideas/%<%Y%m%d%H%M%S>-${slug}.org" "#+title: ${title}\n")
  1364. :unnarrowed t)
  1365. ("p" "project" plain
  1366. "%?"
  1367. :target (file+head "projects/${slug}.org" "#+title: ${title}\n#+filetags: :project:\n")
  1368. :unnarrowed t)
  1369. ("s" "Sicherheitenmeldung" plain
  1370. "*** TODO [#A] Sicherheitenmeldung ${title}\n :PROPERTIES:\n :ID: %(org-id-uuid)\n:END:\n%u\n"
  1371. :target (file+olp "tasks.org" ("Todos" "Sicherheitenmeldungen")))
  1372. ("m" "Monatsbericht" plain
  1373. "*** TODO [#A] Monatsbericht ${title}\n :PROPERTIES:\n :ID: %(org-id-uuid)\n:END:\n%u\n"
  1374. :target (file+olp "tasks.org" ("Todos" "Monatsberichte"))))))
  1375. :custom
  1376. (org-roam-database-connector 'sqlite-builtin)
  1377. (org-roam-directory MY--PATH_ORG_ROAM)
  1378. (org-roam-completion-everywhere t)
  1379. (org-roam-capture-templates
  1380. '(("n" "note" plain
  1381. "%?"
  1382. :if-new (file+head "notes/%<%Y%m%d%H%M%S>-${slug}.org" "#+title: ${title}\n")
  1383. :unnarrowed t)
  1384. ("i" "idea" plain
  1385. "%?"
  1386. :if-new (file+head "ideas/%<%Y%m%d%H%M%S>-${slug}.org" "#+title: ${title}\n")
  1387. :unnarrowed t)
  1388. ))
  1389. :bind (("C-c n l" . org-roam-buffer-toggle)
  1390. ("C-c n f" . org-roam-node-find)
  1391. ("C-c n i" . org-roam-node-insert)
  1392. :map org-mode-map
  1393. ("C-M-i" . completion-at-point)
  1394. :map org-roam-dailies-map
  1395. ("Y" . org-roam-dailies-capture-yesterday)
  1396. ("T" . org-roam-dailies-capture-tomorrow))
  1397. :bind-keymap
  1398. ("C-c n d" . org-roam-dailies-map))
  1399. #+END_SRC
  1400. ** org-node (alternative to org-roam)
  1401. https://github.com/meedstrom/org-node
  1402. no sqlite, just plain text and linking with ids
  1403. not sure if actually better
  1404. #+begin_src emacs-lisp
  1405. (use-package org-node
  1406. :ensure t
  1407. :after org
  1408. :config
  1409. (org-node-cache-mode)
  1410. (org-node-backlink-mode))
  1411. #+end_src
  1412. ** org-ql
  1413. [[https://github.com/alphapapa/org-ql][org-ql]]
  1414. Run queries on org files
  1415. #+begin_src emacs-lisp
  1416. (use-package org-ql
  1417. :ensure t
  1418. :bind (:map org-ql-view-map
  1419. ("<tab>" . org-agenda-show-and-scroll-up)
  1420. ("TAB" . org-agenda-show-and-scroll-up)))
  1421. :custom
  1422. (org-directory MY--PATH_ORG_FILES)
  1423. (org-ql-search-directories-files-recursive t) ;;might hit performance but will search recursively in org-directory
  1424. )
  1425. #+end_src
  1426. ** org-sidebar
  1427. https://github.com/alphapapa/org-sidebar
  1428. #+begin_src emacs-lisp
  1429. (use-package org-sidebar
  1430. :ensure t
  1431. :bind (:map org-sidebar-map
  1432. ("<tab>" . org-agenda-show-and-scroll-up)
  1433. ("TAB" . org-agenda-show-and-scroll-up)))
  1434. #+end_src
  1435. ** TODO Verzeichnis außerhalb roam zum Archivieren (u.a. für erledigte Monatsmeldungen etc.)
  1436. * Programming
  1437. ** Magit / Git
  1438. Little crash course in magit:
  1439. - magit-init to init a git project
  1440. - magit-status (C-x g) to call the status window
  1441. In status buffer:
  1442. - s stage files
  1443. - u unstage files
  1444. - U unstage all files
  1445. - a apply changes to staging
  1446. - c c commit (type commit message, then C-c C-c to commit)
  1447. - b b switch to another branch
  1448. - P u git push
  1449. - F u git pull
  1450. #+BEGIN_SRC emacs-lisp
  1451. ;; updated version needed for magit, at least on windows
  1452. (use-package transient
  1453. :ensure t)
  1454. (use-package magit
  1455. :ensure t
  1456. ; :pin melpa-stable
  1457. :defer t
  1458. :init
  1459. ; set git-path in work environment
  1460. (if (string-equal user-login-name "POH")
  1461. (setq magit-git-executable "P:/Tools/Git/bin/git.exe")
  1462. )
  1463. :bind (("C-x g" . magit-status)))
  1464. #+END_SRC
  1465. ** COMMENT Eglot (can't do dap-mode, maybe dape?)
  1466. for python pyls (in env: pip install python-language-server) seems to work better than pyright (npm install -g pyright),
  1467. at least pandas couldnt be resolved in pyright
  1468. #+begin_src emacs-lisp
  1469. (use-package eglot
  1470. :ensure t
  1471. :init
  1472. (setq completion-category-overrides '((eglot (styles orderless))))
  1473. :config
  1474. (add-to-list 'eglot-server-programs '(python-mode . ("pyright-langserver" "--stdio")))
  1475. (with-eval-after-load 'eglot
  1476. (load-library "project"))
  1477. :hook
  1478. (python-mode . eglot-ensure)
  1479. :custom
  1480. (eglot-ignored-server-capabilities '(:documentHighlightProvider))
  1481. (eglot-autoshutdown t)
  1482. (eglot-events-buffer-size 0)
  1483. )
  1484. ;; performance stuff if necessary
  1485. ;(fset #'jsonrpc--log-event #'ignore)
  1486. #+end_src
  1487. ** LSP-Mode
  1488. #+begin_src emacs-lisp
  1489. (defun corfu-lsp-setup ()
  1490. (setf (alist-get 'styles (alist-get 'lsp-capf completion-category-defaults))
  1491. '(orderless)))
  1492. (use-package lsp-mode
  1493. :ensure t
  1494. ; :hook
  1495. ; ((python-mode . lsp))
  1496. :custom
  1497. (lsp-completion-provider :none)
  1498. (lsp-enable-suggest-server-download nil)
  1499. :hook
  1500. (lsp-completion-mode #'corfu-lsp-setup))
  1501. ;(use-package lsp-ui
  1502. ; :ensure t
  1503. ; :commands lsp-ui-mode)
  1504. (use-package lsp-pyright
  1505. :ensure t
  1506. :after (python lsp-mode)
  1507. :custom
  1508. (lsp-pyright-multi-root nil)
  1509. :hook
  1510. (python-mode-hook . (lambda ()
  1511. (require 'lsp-pyright) (lsp))))
  1512. #+end_src
  1513. ** flymake
  1514. python in venv: pip install pyflake (or ruff?)
  1515. TODO: if ruff active, sideline stops working
  1516. #+begin_src emacs-lisp
  1517. (setq python-flymake-command '("ruff" "--quiet" "--stdin-filename=stdin" "-"))
  1518. #+end_src
  1519. ** sideline
  1520. show flymake errors on the right of code window
  1521. #+begin_src emacs-lisp
  1522. (use-package sideline
  1523. :ensure t)
  1524. (use-package sideline-flymake
  1525. :ensure t
  1526. :requires sideline
  1527. :hook
  1528. (flymake-mode . sideline-mode)
  1529. :init
  1530. (setq sideline-flymake-display-mode 'line ; 'point or 'line
  1531. ; sideline-backends-left '(sideline-lsp)
  1532. sideline-backends-right '(sideline-flymake)))
  1533. #+end_src
  1534. ** yasnippet
  1535. For useful snippet either install yasnippet-snippets or get them from here
  1536. [[https://github.com/AndreaCrotti/yasnippet-snippets][Github]]
  1537. #+begin_src emacs-lisp
  1538. (use-package yasnippet
  1539. :ensure t
  1540. :defer t
  1541. :diminish yas-minor-mode
  1542. :config
  1543. (setq yas-snippet-dirs (list (concat MY--PATH_USER_GLOBAL "snippets")))
  1544. (yas-global-mode t)
  1545. (yas-reload-all)
  1546. (unbind-key "TAB" yas-minor-mode-map)
  1547. (unbind-key "<tab>" yas-minor-mode-map))
  1548. #+end_src
  1549. ** hippie expand
  1550. With hippie expand I am able to use yasnippet and emmet at the same time with the same key.
  1551. #+begin_src emacs-lisp
  1552. (use-package hippie-exp
  1553. :ensure nil
  1554. :defer t
  1555. :bind
  1556. ("C-<return>" . hippie-expand)
  1557. :config
  1558. (setq hippie-expand-try-functions-list
  1559. '(yas-hippie-try-expand emmet-expand-line)))
  1560. #+end_src
  1561. ** COMMENT flycheck (now flymake)
  1562. #+BEGIN_SRC emacs-lisp
  1563. (use-package flycheck
  1564. :ensure t
  1565. :hook
  1566. ((css-mode . flycheck-mode)
  1567. (emacs-lisp-mode . flycheck-mode)
  1568. (python-mode . flycheck-mode))
  1569. :defer 1.0
  1570. :init
  1571. (setq flycheck-emacs-lisp-load-path 'inherit)
  1572. :config
  1573. (setq-default
  1574. flycheck-check-synta-automatically '(save mode-enabled)
  1575. flycheck-disable-checkers '(emacs-lisp-checkdoc)
  1576. eldoc-idle-delay .1 ;; let eldoc echo faster than flycheck
  1577. flycheck-display-errors-delay .3)) ;; this way any errors will override eldoc messages
  1578. #+END_SRC
  1579. ** smartparens
  1580. #+BEGIN_SRC emacs-lisp
  1581. (use-package smartparens
  1582. :ensure t
  1583. :diminish smartparens-mode
  1584. :bind
  1585. (:map smartparens-mode-map
  1586. ("C-M-f" . sp-forward-sexp)
  1587. ("C-M-b" . sp-backward-sexp)
  1588. ("C-M-a" . sp-backward-down-sexp)
  1589. ("C-M-e" . sp-up-sexp)
  1590. ("C-M-w" . sp-copy-sexp)
  1591. ("M-k" . sp-kill-sexp)
  1592. ("C-M-<backspace>" . sp-slice-sexp-killing-backward)
  1593. ("C-S-<backspace>" . sp-slice-sexp-killing-around)
  1594. ("C-]" . sp-select-next-thing-exchange))
  1595. :config
  1596. (setq sp-show-pair-from-inside nil
  1597. sp-escape-quotes-after-insert nil)
  1598. (require 'smartparens-config))
  1599. #+END_SRC
  1600. ** lisp
  1601. #+BEGIN_SRC emacs-lisp
  1602. (use-package elisp-mode
  1603. :ensure nil
  1604. :defer t)
  1605. #+END_SRC
  1606. ** web
  1607. apt install npm
  1608. sudo npm install -g vscode-html-languageserver-bin
  1609. evtl alternativ typescript-language-server?
  1610. Unter Windows:
  1611. Hier runterladen: https://nodejs.org/dist/latest/
  1612. und in ein Verzeichnis entpacken.
  1613. Optional: PATH erweitern unter Windows (so kann exec-path-from-shell den Pfad ermitteln):
  1614. PATH=P:\path\to\node;%path%
  1615. *** web-mode
  1616. #+BEGIN_SRC emacs-lisp
  1617. (use-package web-mode
  1618. :ensure t
  1619. :defer t
  1620. :mode
  1621. ("\\.phtml\\'"
  1622. "\\.tpl\\.php\\'"
  1623. "\\.djhtml\\'"
  1624. "\\.[t]?html?\\'")
  1625. :hook
  1626. (web-mode . smartparens-mode)
  1627. :init
  1628. (if *work_remote*
  1629. (setq exec-path (append exec-path '("P:/Tools/node"))))
  1630. :config
  1631. (setq web-mode-enable-auto-closing t
  1632. web-mode-enable-auto-pairing t))
  1633. #+END_SRC
  1634. Emmet offers snippets, similar to yasnippet.
  1635. Default completion is C-j
  1636. [[https://github.com/smihica/emmet-mode#usage][Github]]
  1637. #+begin_src emacs-lisp
  1638. (use-package emmet-mode
  1639. :ensure t
  1640. :defer t
  1641. :hook
  1642. ((web-mode . emmet-mode)
  1643. (css-mode . emmet-mode))
  1644. :config
  1645. (unbind-key "C-<return>" emmet-mode-keymap))
  1646. #+end_src
  1647. *** JavaScript
  1648. npm install -g typescript-language-server typescript
  1649. maybe only typescript?
  1650. npm install -g prettier
  1651. #+begin_src emacs-lisp
  1652. (use-package rjsx-mode
  1653. :ensure t
  1654. :mode ("\\.js\\'"
  1655. "\\.jsx'"))
  1656. ; :config
  1657. ; (setq js2-mode-show-parse-errors nil
  1658. ; js2-mode-show-strict-warnings nil
  1659. ; js2-basic-offset 2
  1660. ; js-indent-level 2)
  1661. ; (setq-local flycheck-disabled-checkers (cl-union flycheck-disable-checkers
  1662. ; '(javascript-jshint)))) ; jshint doesn"t work for JSX
  1663. (use-package tide
  1664. :ensure t
  1665. :after (rjsx-mode company flycheck)
  1666. ; :hook (rjsx-mode . setup-tide-mode)
  1667. :config
  1668. (defun setup-tide-mode ()
  1669. "Setup function for tide."
  1670. (interactive)
  1671. (tide-setup)
  1672. (flycheck-mode t)
  1673. (setq flycheck-check-synta-automatically '(save mode-enabled))
  1674. (tide-hl-identifier-mode t)))
  1675. ;; needs npm install -g prettier
  1676. (use-package prettier-js
  1677. :ensure t
  1678. :after (rjsx-mode)
  1679. :defer t
  1680. :diminish prettier-js-mode
  1681. :hook ((js2-mode rsjx-mode) . prettier-js-mode))
  1682. #+end_src
  1683. ** YAML
  1684. #+begin_src emacs-lisp
  1685. (use-package yaml-mode
  1686. :if *sys/linux*
  1687. :ensure t
  1688. :defer t
  1689. :mode ("\\.yml$" . yaml-mode))
  1690. #+end_src
  1691. ** R
  1692. #+BEGIN_SRC emacs-lisp
  1693. (use-package ess
  1694. :ensure t
  1695. :defer t
  1696. :init
  1697. (if *work_remote*
  1698. (setq exec-path (append exec-path '("P:/Tools/R/bin/x64"))
  1699. org-babel-R-command "P:/Tools/R/bin/x64/R --slave --no-save")))
  1700. #+END_SRC
  1701. ** project.el
  1702. #+begin_src emacs-lisp
  1703. (use-package project
  1704. :custom
  1705. (project-vc-extra-root-markers '(".project.el" ".project" )))
  1706. #+end_src
  1707. ** Python
  1708. Preparations:
  1709. - create env:
  1710. m-x shell
  1711. python3 -m venv ./.env
  1712. - Install language server in *each* projects venv
  1713. source ./.env/bin/activate
  1714. pip install pyright
  1715. - in project root:
  1716. touch .project.el
  1717. echo "((nil . (pyvenv-activate . "/path/to/project/.env")))" >> .dir-locals.el
  1718. für andere language servers
  1719. https://github.com/emacs-lsp/lsp-mode#install-language-server
  1720. TODO if in a project, set venv automatically
  1721. (when-let ((project (project-current))) (project-root project))
  1722. returns project path from project.el
  1723. to recognize a project, either have git or
  1724. place a .project.el file in project root and
  1725. (setq project-vc-extra-root-markers '(".project.el" "..." ))
  1726. #+begin_src emacs-lisp
  1727. (use-package python
  1728. :if *sys/linux*
  1729. :delight "π "
  1730. :defer t
  1731. :bind (("M-[" . python-nav-backward-block)
  1732. ("M-]" . python-nav-forward-block))
  1733. :mode
  1734. (("\\.py\\'" . python-mode)))
  1735. (use-package pyvenv
  1736. ; :if *sys/linux*
  1737. :ensure t
  1738. :defer t
  1739. :after python
  1740. :hook
  1741. (python-mode . pyvenv-mode)
  1742. :custom
  1743. (pyvenv-default-virtual-env-name ".env")
  1744. (pyvenv-mode-line-indicator '(pyvenv-virtual-env-name ("[venv:" pyvenv-virtual-env-name "]"))))
  1745. ;; formatting to pep8
  1746. ;; requires pip install black
  1747. ;(use-package blacken
  1748. ; :ensure t)
  1749. #+end_src
  1750. TODO python mode hook:
  1751. - activate venv
  1752. - activate eglot with proper ls
  1753. - activate tree-sitter?
  1754. - have some fallback if activations fail
  1755. * beancount
  1756. ** Installation
  1757. #+BEGIN_SRC shell :tangle no
  1758. sudo su
  1759. cd /opt
  1760. python3 -m venv beancount
  1761. source ./beancount/bin/activate
  1762. pip3 install wheel
  1763. pip3 install beancount
  1764. sleep 100
  1765. echo "shell running!"
  1766. deactivate
  1767. #+END_SRC
  1768. #+begin_src emacs-lisp
  1769. (use-package beancount
  1770. :ensure nil
  1771. :if *sys/linux*
  1772. :load-path "user-global/elisp/"
  1773. ; :ensure t
  1774. :defer t
  1775. :mode
  1776. ("\\.beancount$" . beancount-mode)
  1777. :hook
  1778. (beancount-mode . my/beancount-company)
  1779. :config
  1780. (defun my/beancount-company ()
  1781. (setq-local completion-at-point-functions #'beancount-completion-at-point))
  1782. (setq beancount-filename-main "/home/marc/archiv/Finanzen/Transaktionen/transactions.beancount"))
  1783. #+end_src
  1784. +BEGIN_SRC emacs-lisp
  1785. (use-package beancount
  1786. :if *sys/linux*
  1787. :load-path "user-global/elisp"
  1788. ; :ensure t
  1789. :defer t
  1790. :mode
  1791. ("\\.beancount$" . beancount-mode)
  1792. ; :hook
  1793. ; (beancount-mode . my/beancount-company)
  1794. ; :init
  1795. ; (add-hook 'beancount-mode-hook 'company/beancount-mode-hook)
  1796. :config
  1797. (defun my/beancount-company ()
  1798. (setq-local completion-at-point-functions #'beancount-complete-at-point nil t))
  1799. ; (mapcar #'cape-company-to-capf
  1800. ; (list #'company-beancount #'company-dabbrev))))
  1801. (defun my--beancount-companyALT ()
  1802. (set (make-local-variable 'company-backends)
  1803. '(company-beancount)))
  1804. (setq beancount-filename-main "/home/marc/Archiv/Finanzen/Transaktionen/transactions.beancount"))
  1805. +END_SRC
  1806. To support org-babel, check if it can find the symlink to ob-beancount.el
  1807. #+BEGIN_SRC shell :tangle no
  1808. orgpath=`find /home/marc/.emacs.d/elpa/ -type d -name "org-plus*" -print`
  1809. beansym="$orgpath/ob-beancount.el
  1810. bean="/home/marc/Archiv/Programmierprojekte/Lisp/beancount-mode/ob-beancount.el"
  1811. if [ -h "$beansym" ]
  1812. then
  1813. echo "$beansym found"
  1814. elif [ -e "$bean" ]
  1815. then
  1816. echo "creating symlink"
  1817. ln -s "$bean" "$beansym"
  1818. else
  1819. echo "$bean not found, symlink creation aborted"
  1820. fi
  1821. #+END_SRC
  1822. Fava is strongly recommended.
  1823. #+BEGIN_SRC shell :tangle no
  1824. cd /opt
  1825. python3 -m venv fava
  1826. source ./fava/bin/activate
  1827. pip3 install wheel
  1828. pip3 install fava
  1829. deactivate
  1830. #+END_SRC
  1831. Start fava with fava my_file.beancount
  1832. It is accessable on this URL: [[http://127.0.0.1:5000][Fava]]
  1833. Beancount-mode can start fava and open the URL right away.
  1834. * Stuff after everything else
  1835. Set garbage collector to a smaller value to let it kick in faster.
  1836. Maybe a problem on Windows?
  1837. #+begin_src emacs-lisp
  1838. ;(setq gc-cons-threshold (* 2 1000 1000))
  1839. #+end_src
  1840. Rest of early-init.el
  1841. #+begin_src emacs-lisp :tangle early-init.el
  1842. (defconst config-org (expand-file-name "config.org" user-emacs-directory))
  1843. (defconst init-el (expand-file-name "init.el" user-emacs-directory))
  1844. (unless (file-exists-p init-el)
  1845. (require 'org)
  1846. (org-babel-tangle-file config-org init-el))
  1847. #+end_src