aboutsummaryrefslogtreecommitdiff
path: root/config/emacs.org
blob: ff6342e13e515e688c7c88495ac999f27bfb4a57 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
#+TITLE: Emacs Configuration
#+AUTHOR: Preston Pan
#+DESCRIPTION: my personal emacs configuration for nixOS
#+html_head: <link rel="stylesheet" type="text/css" href="../style.css" />

* Introduction
This is my Vanilla Emacs configuration, made to work with my NixOS configuration. For that
reason, you will not see :ensure t inside any use-package declaration, for emacs packages
are all compiled natively and reproducibly on the NixOS side. This configuration uses the
emacs-lisp language only to configure variables for said packages, for the most part.
** UI Elements
This section contains important UI elements and starting customization variables to make
emacs work in a semi-sane way and make it not look completely ugly:
#+begin_src emacs-lisp
  (display-battery-mode 1)
  (setq display-time-24hr-format t)
  (display-time-mode 1)
  (menu-bar-mode -1)
  (scroll-bar-mode -1)
  (tool-bar-mode -1)
  (load-theme 'catppuccin :no-confirm)
  (add-hook 'prog-mode-hook #'display-line-numbers-mode)
  (add-hook 'org-mode-hook #'display-line-numbers-mode)
  (set-face-attribute 'default nil :height 120)
  (setq use-short-answers t)
  (setq make-backup-files nil)
  (setq org-export-with-broken-links t)
  (setq org-src-fontify-natively t)
  (setq warning-minimum-level :emergency)
  (add-hook 'text-mode-hook 'visual-line-mode)
  (add-hook 'after-save-hook 'eglot-format)
  (setq debug-ignored-errors
      (cons 'remote-file-error debug-ignored-errors))
#+end_src
* Transparency
My NixOS configuration uses Hyprland to make things transparent:
#+begin_src emacs-lisp
  (set-frame-parameter nil 'alpha-background 90)
  (add-to-list 'default-frame-alist '(alpha-background . 90))
#+end_src
* Scrolling
Make emacs scroll in a sane way:
#+begin_src emacs-lisp
  (setq mouse-wheel-scroll-amount '(1 ((shift) . 1)))
  (setq mouse-wheel-progressive-speed nil)
  (setq mouse-wheel-follow-mouse 't)
  (setq scroll-step 1)
#+end_src
* Agenda
Configure org agenda variables:
#+begin_src emacs-lisp
  (require 'org-habit)
  (setq org-agenda-files (list "~/org/agenda.org"
			       "~/org/notes.org"))
  (setq org-default-notes-file (concat org-directory "/notes.org"))
  (setq org-habit-preceding-days 1)
#+end_src
* Publishing
This is the configuration required to publish my website:
#+begin_src emacs-lisp
  (require 'ox-publish)
  (setq org-publish-project-alist
	'(("website-org"
	   :base-directory "~/org/website"
	   :base-extension "org"
	   :publishing-directory "~/website_html"
	   :recursive t
	   :publishing-function org-html-publish-to-html
	   :headline-levels 4
	   :html-preamble t
	   :html-preamble-format (("en" "<p class=\"preamble\"><a href=\"/index.html\">home</a> | <a href=\"./index.html\">section main page</a></p><hr>")))
	  ("website-static"
	   :base-directory "~/org/website"
	   :base-extension "css\\|js\\|png\\|jpg\\|gif\\|pdf\\|mp3\\|ogg\\|swf\\|ico"
	   :publishing-directory "~/website_html/"
	   :recursive t
	   :publishing-function org-publish-attachment)
	  ("website" :auto-sitemap t :components ("website-org" "website-static"))))
  ;; (setq org-export-html-postamble-format '(("en" "<p class=\"preamble\"><a href=\"../index.html\">previous page</a> | <a href=\"/index.html\">home</a></p>")))
  (setq org-html-postamble "Copyright © 2024 Preston Pan")
#+end_src
* Autopair
Use electric-pair to automatically complete pairs of things. We need to change
what electric-pair does based on the mode.
#+begin_src emacs-lisp
  (defun electric-pair ()
    "If at end of line, insert character pair without surrounding spaces.
  Otherwise, just insert the typed character."
    (interactive)
    (if (eolp) (let (parens-require-spaces) (insert-pair)) (self-insert-command 1)))
    (add-hook 'org-mode-hook
	      (lambda ()
		(define-key org-mode-map "\"" 'electric-pair)
		(define-key org-mode-map "\'" 'electric-pair)
		(define-key org-mode-map "(" 'electric-pair)
		(define-key org-mode-map "[" 'electric-pair)
		(define-key org-mode-map "{" 'electric-pair)))
#+end_src
* Completion
Company-mode! We need this to do autocomplete stuff.
#+begin_src emacs-lisp
  (add-hook 'after-init-hook 'global-company-mode)
#+end_src
* Org Babel
For some reason, org-babel doesn't load these languages by default:
#+begin_src emacs-lisp
(org-babel-do-load-languages 'org-babel-load-languages
    '(
        (shell . t)
    )
)
#+end_src
* Packages
First, some small configurations and some evil-mode initilaization because I like vim keybindings:
#+begin_src emacs-lisp
  (require 'org-tempo)
  (use-package evil
    :init
    (setq evil-want-keybinding nil)
    :config
    (evil-mode 1)
    (evil-set-undo-system 'undo-redo))
  (use-package evil-collection
    :init
    (setq evil-want-keybinding nil)
    :config
    (evil-collection-init))
  (with-eval-after-load 'evil-maps
  (define-key evil-motion-state-map (kbd "SPC") nil)
  (define-key evil-motion-state-map (kbd "RET") nil)
  (define-key evil-motion-state-map (kbd "TAB") nil))
  (use-package evil-commentary
    :config
    (evil-commentary-mode))
  (use-package evil-org
    :after org
    :hook (org-mode . (lambda () evil-org-mode))
    :config
    (require 'evil-org-agenda)
    (evil-org-agenda-set-keys))

  (use-package which-key
    :config
    (which-key-mode))
  (use-package page-break-lines
    :init
    (page-break-lines-mode))
#+end_src
** Journal
I use org-journal to journal about my life, and it's a part of my website:
#+begin_src emacs-lisp
  (use-package org-journal
    :init
      (setq org-journal-dir "~/org/website/journal/")
      (setq org-journal-date-format "%A, %d %B %Y")

      (defun org-journal-file-header-func (time)
      "Custom function to create journal header."
      (concat
	(pcase org-journal-file-type
	  (`daily "#+TITLE: Daily Journal\n#+STARTUP: showeverything\n#+DESCRIPTION: My daily journal entry\n#+AUTHOR: Preston Pan\n#+HTML_HEAD: <link rel=\"stylesheet\" type=\"text/css\" href=\"../style.css\" />\n#+html_head: <script src=\"https://polyfill.io/v3/polyfill.min.js?features=es6\"></script>\n#+html_head: <script id=\"MathJax-script\" async src=\"https://cdn.jsdelivr.net/npm/mathjax@3/es5/tex-mml-chtml.js\"></script>\n#+options: broken-links:t")
	  (`weekly "#+TITLE: Weekly Journal\n#+STARTUP: folded")
	  (`monthly "#+TITLE: Monthly Journal\n#+STARTUP: folded")
	  (`yearly "#+TITLE: Yearly Journal\n#+STARTUP: folded"))))

    (setq org-journal-file-header 'org-journal-file-header-func)
    (setq org-journal-file-format "%Y%m%d.org")
    (setq org-journal-enable-agenda-integration t)
  )
#+end_src
** Doom Modeline
The default modeline is ugly.
#+begin_src emacs-lisp
  (use-package doom-modeline
  :config
  (doom-modeline-mode 1))
#+end_src
** Make Org Look Better
Org superstar adds those nice looking utf-8 bullets:
#+begin_src emacs-lisp
  (use-package org-superstar
  :config
  (add-hook 'org-mode-hook (lambda () (org-superstar-mode 1))))
#+end_src
** LSP
We set up eglot, the LSP manager for emacs, now built in:
#+begin_src emacs-lisp
  (use-package eglot 
    :config
    (add-hook 'prog-mode-hook 'eglot-ensure))
#+end_src
** Dashboard
We want our emacs initialization to be pretty and display useful things.
#+begin_src emacs-lisp
  (use-package dashboard
    :init
    (setq dashboard-banner-logo-title "Welcome, Commander!")
    (setq dashboard-icon-type 'nerd-icons)
    (setq dashboard-vertically-center-content t)
    (setq dashboard-set-init-info t)
    (setq dashboard-week-agenda t)
    (setq dashboard-items '((recents   . 5)
			(bookmarks . 5)
			(projects  . 5)
			(agenda    . 5)
			(registers . 5)))
    :config
    (dashboard-setup-startup-hook))
#+end_src
** Projectile
Manages projects and shit.
#+begin_src emacs-lisp
  (use-package projectile
    :config
    (projectile-mode +1))
#+end_src
** Ivy
Ivy is a pretty cool general program for displaying stuff:
#+begin_src emacs-lisp
  (use-package counsel)
  (use-package ivy
    :init
    (setq ivy-use-virtual-buffers t)
    (setq enable-recursive-minibuffers t)
    ;; enable this if you want `swiper' to use it
    ;; (setq search-default-mode #'char-fold-to-regexp)
    (global-set-key "\C-s" 'swiper)
    (global-set-key (kbd "C-c C-r") 'ivy-resume)
    (global-set-key (kbd "<f6>") 'ivy-resume)
    (global-set-key (kbd "M-x") 'counsel-M-x)
    (global-set-key (kbd "C-x C-f") 'counsel-find-file)
    (global-set-key (kbd "<f1> f") 'counsel-describe-function)
    (global-set-key (kbd "<f1> v") 'counsel-describe-variable)
    (global-set-key (kbd "<f1> o") 'counsel-describe-symbol)
    (global-set-key (kbd "<f1> l") 'counsel-find-library)
    (global-set-key (kbd "<f2> i") 'counsel-info-lookup-symbol)
    (global-set-key (kbd "<f2> u") 'counsel-unicode-char)
    (global-set-key (kbd "C-c g") 'counsel-git)
    (global-set-key (kbd "C-c j") 'counsel-git-grep)
    (global-set-key (kbd "C-c k") 'counsel-ag)
    (global-set-key (kbd "C-x l") 'counsel-locate)
    (global-set-key (kbd "C-S-o") 'counsel-rhythmbox)
    (define-key minibuffer-local-map (kbd "C-r") 'counsel-minibuffer-history)
    :config
    (ivy-mode))
#+end_src
** Magit
#+begin_src emacs-lisp
(use-package magit)
#+end_src
** Keybindings
#+begin_src emacs-lisp
  (use-package general
    :config
    (general-create-definer leader-key
      :prefix "SPC")
    (leader-key 'normal
      "o a" 'org-agenda
      "c b" 'counsel-bookmark
      "o c" 'org-capture
      "n j j" 'org-journal-new-entry
      "n r f" 'org-roam-node-find
      "n r i" 'org-roam-node-insert
      "n r g" 'org-roam-graph
      "r s s" 'elfeed
      "." 'counsel-find-file
      "g c /" 'magit-dispatch
      "g c c" 'magit-commit
      "o t" 'vterm-other-window
      "o e" 'eshell
      "o m" 'mu4e
      "e w" 'eww
      "p w" 'ivy-pass
      "m P p" 'org-publish
      "s e" 'sudo-edit
      "m m" 'emms
      "h m" '(woman :wk "Manual")
      "h r r" '(lambda () (interactive) (org-babel-load-file (expand-file-name "~/org/website/config/emacs.org")))
      ))
#+end_src
** RSS Feed
I use really simple syndication (RSS) in order to read news. As a result, I use
elfeed to fetch feeds found on my website:
#+begin_src emacs-lisp
  (use-package elfeed
    :init
    (add-hook 'elfeed-search-mode-hook #'elfeed-update)
    (setq elfeed-search-filter "@1-month-ago +unread")
    )
  (use-package elfeed-org
    :init
    (setq rmh-elfeed-org-files '("~/org/website/config/elfeed.org"))
    :config
    (elfeed-org))
#+end_src
** Eww
Used only for the purpose of viewing RSS feed items in emacs if I can, only resorting
to Firefox if I have to:
#+begin_src emacs-lisp
(setq search-engines
      '(
        (("google" "g") "https://google.com/search?q=%s")
        (("duckduckgo" "d" "ddg") "https://duckduckgo.com/?q=%s")
        (("rfc" "r") "https://www.rfc-editor.org/rfc/rfc%s.txt")
        (("rfc-kw" "rk") "https://www.rfc-editor.org/search/rfc_search_detail.php?title=%s")))

(setq search-engine-default "google")
(setq eww-search-prefix "https://google.com/search?q=")
(setq browse-url-secondary-browser-function 'browse-url-generic browse-url-generic-program "firefox")
(setq browse-url-browser-function 'eww-browse-url)
(add-hook 'eww-mode-hook
          (lambda () (local-set-key (kbd "y Y") #'eww-copy-page-url)))
#+end_src
** Org Roam
For all my mathematics and programming notes:
#+begin_src emacs-lisp
  (use-package org-roam
    :init
    (setq org-roam-graph-viewer "firefox")
    (setq org-roam-directory (file-truename "~/org/website/mindmap"))
    (setq org-roam-capture-templates '(("d" "default" plain "%?"
				      :target (file+head "${title}.org"
							 "#+title: ${title}\n#+author: Preston Pan\n#+html_head: <link rel=\"stylesheet\" type=\"text/css\" href=\"../style.css\" />\n#+html_head: <script src=\"https://polyfill.io/v3/polyfill.min.js?features=es6\"></script>\n#+html_head: <script id=\"MathJax-script\" async src=\"https://cdn.jsdelivr.net/npm/mathjax@3/es5/tex-mml-chtml.js\"></script>\n#+options: broken-links:t")
				      :unnarrowed t))))
#+end_src
** Pinentry
Set up pinentry so that I can use emacs as my pinentry frontend:
#+begin_src emacs-lisp
  (use-package pinentry
    :init (setq epa-pinentry-mode `loopback)
    :config (pinentry-start))
#+end_src
** Auctex
Make LaTeX a litle better:
#+begin_src emacs-lisp
  (setq TeX-PDF-mode t)
  (setq org-format-latex-options (plist-put org-format-latex-options :scale 1.5))
  (setq org-return-follows-link t)
#+end_src
** Email
Email in emacs can be done with Mu4e.
#+begin_src emacs-lisp
    ;; SMTP settings:
  (setq user-mail-address "preston@nullring.xyz")
  (setq user-full-name "Preston Pan")
  (setq sendmail-program "msmtp"
      send-mail-function 'smtpmail-send-it
      message-sendmail-f-is-evil t
      message-sendmail-extra-arguments '("--read-envelope-from")
      message-send-mail-function 'message-send-mail-with-sendmail)

  (require 'smtpmail)
  (use-package mu4e
    :init
    (setq mu4e-drafts-folder "/Drafts")
    (setq mu4e-sent-folder   "/Sent")
    (setq mu4e-trash-folder  "/Trash")
    (setq mu4e-attachment-dir  "~/Downloads")
    (setq mu4e-view-show-addresses 't)
    (setq mu4e-confirm-quit nil)


    (setq message-kill-buffer-on-exit t)
    (setq mu4e-compose-dont-reply-to-self t)
    (setq mu4e-change-filenames-when-moving t)
    (setq mu4e-get-mail-command "mbsync prestonpan")
    (setq mu4e-compose-reply-ignore-address '("no-?reply" "preston@nullring.xyz"))
    (setq mu4e-html2text-command "w3m -T text/html" ; how to hanfle html-formatted emails
	  mu4e-update-interval 300                  ; seconds between each mail retrieval
	  mu4e-headers-auto-update t                ; avoid to type `g' to update
	  mu4e-view-show-images t                   ; show images in the view buffer
	  mu4e-compose-signature-auto-include nil   ; I don't want a message signature
	  mu4e-use-fancy-chars t))
#+end_src
** Password Manager
I use ~pass~ in order to manage my passwords on linux, and this is an ivy frontend for it:
#+begin_src emacs-lisp
(use-package ivy-pass)
#+end_src
** Music
Set up emms in order to play music from my music directory:
#+begin_src emacs-lisp
  (use-package emms
    :init
    (emms-all)
    (setq emms-source-file-default-directory (expand-file-name "~/music/"))
    (setq emms-player-mpd-music-directory "/home/preston/music/")
    (setq emms-player-mpd-server-name "localhost")
    (setq emms-player-mpd-server-port "6600")
    (setq emms-player-list '(emms-player-mpd))
    (add-to-list 'emms-info-functions 'emms-info-mpd)
    (add-to-list 'emms-player-list 'emms-player-mpd)
  :config
    (emms-player-mpd-connect))
#+end_src
** Stem
My own programming language.
#+begin_src emacs-lisp
  (use-package stem-mode)
  (add-to-list 'auto-mode-alist '("\\.stem\\'" . stem-mode))
#+end_src