Browse Source
Merge branch 'master' of https://gitea.cloudsphere.duckdns.org/marc/config_emacs
master
Merge branch 'master' of https://gitea.cloudsphere.duckdns.org/marc/config_emacs
master
Marc Pohling
5 years ago
5 changed files with 1573 additions and 57 deletions
Split View
Diff Options
-
62config.org
-
11create-init.sh
-
386init.el
-
551init.org
-
620user-local/elisp/beancount.el
@ -0,0 +1,11 @@ |
|||
#!/bin/sh |
|||
rm ./init.el |
|||
rm ./init.elc |
|||
|
|||
cat <<EOF >./init.el |
|||
(require 'org) |
|||
(find-file (concat user-emacs-directory "init.org")) |
|||
(org-babel-tangle) |
|||
(load-file (concat user-emacs-directory "init.el")) |
|||
(byte-compile-file (concat user-emacs-directory "init.el")) |
|||
EOF |
@ -1,55 +1,357 @@ |
|||
;; Garbage collection threshold |
|||
;; higher means less interuptions |
|||
(setq gc-cons-threshold 400000000) |
|||
|
|||
|
|||
;; Begin initialization |
|||
;; Turn off mouse interface early in startup to avoid momentary display |
|||
(when window-system |
|||
(menu-bar-mode -1) |
|||
(tool-bar-mode -1) |
|||
(scroll-bar-mode -1) |
|||
(tooltip-mode -1) |
|||
) |
|||
|
|||
(setq inhibit-startup-message t) |
|||
(setq initial-scratch-message "") |
|||
(defun me/tangle-init () |
|||
"If the current buffer is 'init.org', |
|||
the code blocks are tangled, and the tangled file is compiled." |
|||
(when (equal (buffer-file-name) |
|||
(expand-file-name (concat user-emacs-directory "init.org"))) |
|||
;; avoid running hooks |
|||
(let ((prog-mode-hook nil)) |
|||
(org-babel-tangle) |
|||
(byte-compile-file (concat user-emacs-directory "init.el")) |
|||
(load-file user-init-file)))) |
|||
(add-hook 'after-save-hook 'me/tangle-init) |
|||
|
|||
;; Setup package |
|||
(require 'package) |
|||
(add-to-list 'package-archives |
|||
'("melpa" . "https://melpa.org/packages/") t) |
|||
(add-to-list 'package-archives |
|||
'("melpa-stable" . "https://stable.melpa.org/packages/") t) |
|||
(add-to-list 'package-archives |
|||
'("gnu" . "https://elpa.gnu.org/packages/") t) |
|||
(add-to-list 'package-archives |
|||
'("org" . "https://orgmode.org/elpa/") t) |
|||
|
|||
(add-to-list 'package-archives '("melpa" . "https://melpa.org/packages/") t) |
|||
(add-to-list 'package-archives '("melpa-stable" . "https://stable.melpa.org/packages/") t) |
|||
(add-to-list 'package-archives '("org" . "https://orgmode.org/elpa/") t) |
|||
|
|||
(package-initialize) |
|||
|
|||
;; Bootstrap use-package |
|||
;; Install use-package if it's not already installed |
|||
;; use-package is used to configure the rest of the packages |
|||
(unless (package-installed-p 'use-package) |
|||
(package-refresh-contents) |
|||
(package-install 'use-package) |
|||
) |
|||
(package-refresh-contents) |
|||
(package-install 'use-package)) |
|||
|
|||
(setq use-package-verbose nil) |
|||
|
|||
;; From use-package README |
|||
(eval-when-compile |
|||
(require 'use-package)) |
|||
;; see https://github.com/jwiegley/use-package/issues/522 |
|||
(use-package diminish |
|||
:ensure t) |
|||
;(eval-when-compile |
|||
(require 'use-package);) |
|||
|
|||
(defvar MY--PATH_USER_LOCAL (expand-file-name "~/.emacs.d/user-local/")) |
|||
(defvar MY--PATH_USER_GLOBAL (expand-file-name "~/.emacs.d/user-global/")) |
|||
|
|||
(defvar MY--PATH_ORG_FILES (expand-file-name "~/Archiv/Organisieren/")) |
|||
(defvar MY--PATH_ORG_FILES_MOBILE (expand-file-name "~/Archiv/Organisieren/mobile/")) |
|||
(defvar MY--PATH_ORG_JOURNAl (expand-file-name "~/Archiv/Organisieren/Journal/")) |
|||
(setq bookmark-default-file (concat MY--PATH_USER_LOCAL "bookmarks")) |
|||
(setq recentf-save-file (concat MY--PATH_USER_LOCAL "recentf")) |
|||
(setq custom-file (concat MY--PATH_USER_LOCAL "custom.el")) ;; don't spam init.e with saved customization settings |
|||
(setq abbrev-file-name (concat MY--PATH_USER_GLOBAL "abbrev_defs")) |
|||
(setq backup-directory-alist `((".*" . ,temporary-file-directory))) |
|||
(setq auto-save-file-name-transforms `((".*" ,temporary-file-directory))) |
|||
(setq save-abbrevs 'silently) ;; don't bother me with asking for abbrev saving |
|||
(setq-default create-lockfiles nil) ;; disable lock files, can cause trouble in e.g. lsp-mode |
|||
(defalias 'yes-or-no-p 'y-or-n-p) ;; answer with y and n |
|||
(setq custom-safe-themes t) ;; don't ask me if I want to load a theme |
|||
(setq sentence-end-double-space nil) ;; don't coun two spaces after a period as the end of a sentence. |
|||
(delete-selection-mode t) ;; delete selected region when typing |
|||
|
|||
(setq locale-coding-system 'utf-8) |
|||
(set-terminal-coding-system 'utf-8) |
|||
(set-keyboard-coding-system 'utf-8) |
|||
(set-selection-coding-system 'utf-8) |
|||
(if (eq system-type 'windows-nt) |
|||
(prefer-coding-system 'utf-8dos) |
|||
(prefer-coding-system 'utf-8)) |
|||
(blink-cursor-mode -1) ;; turn off blinking cursor |
|||
(show-paren-mode t) ;; show other part of brackets |
|||
(column-number-mode t) |
|||
(setq uniquify-buffer-name-style 'forward) |
|||
(setq-default indent-tabs-mode nil) ;; avoid tabs in place of multiple spaces (they look bad in tex) |
|||
(setq-default indicate-empty-lines t) ;; show empty lines |
|||
(setq scroll-margin 5 ;; smooth scrolling |
|||
scroll-conservatively 10000 |
|||
scroll-preserve-screen-position 1 |
|||
scroll-step 1) |
|||
(global-hl-line-mode t) ;; highlight current line |
|||
(menu-bar-mode 0) ;; disable menu bar |
|||
(tool-bar-mode 0) ;; disable tool bar |
|||
(scroll-bar-mode 0) ;; disable scroll bar |
|||
|
|||
(set-face-font 'default "Hack Nerd Font Mono-10") |
|||
|
|||
(global-visual-line-mode) |
|||
(diminish 'visual-line-mode) |
|||
(use-package adaptive-wrap |
|||
:ensure t |
|||
:init |
|||
(when (fboundp 'adaptive-wrap-prefix-mode) |
|||
(defun my/activate-adaptive-wrap-prefix-mode () |
|||
"Toggle `visual-line-mode' and `adaptive-wrap-prefix-mode' simultaneously." |
|||
(adaptive-wrap-prefix-mode (if visual-line-mode 1 -1))) |
|||
(add-hook 'visual-line-mode-hook 'my/activate-adaptive-wrap-prefix-mode))) |
|||
|
|||
(use-package display-line-numbers |
|||
:init |
|||
(add-hook 'prog-mode-hook 'display-line-numbers-mode) |
|||
(add-hook 'org-src-mode-hook 'display-line-numbers-mode) |
|||
:config |
|||
(setq-default display-line-numbers-type 'visual |
|||
display-line-numbers-current-absolute t |
|||
display-line-numbers-with 4 |
|||
display-line-numbers-widen t)) |
|||
; (add-hook 'emacs-lisp-mode-hook 'display-line-numbers-mode) |
|||
|
|||
(use-package rainbow-mode |
|||
:diminish |
|||
:hook ((org-mode |
|||
emacs-lisp-mode) . rainbow-mode)) |
|||
|
|||
(require 'undo-tree) |
|||
(use-package undo-tree |
|||
:ensure t |
|||
:diminish undo-tree-mode |
|||
:init |
|||
(global-undo-tree-mode 1)) |
|||
|
|||
(use-package imenu-list |
|||
:ensure t |
|||
:config |
|||
(setq imenu-list-focus-after-activation t |
|||
imenu-list-auto-resize t |
|||
imenu-list-position 'right) |
|||
:bind |
|||
(:map global-map |
|||
([f9] . imenu-list-smart-toggle)) |
|||
) |
|||
|
|||
(require 'which-key) |
|||
(use-package which-key |
|||
:ensure t |
|||
:diminish which-key-mode |
|||
:config |
|||
(which-key-mode) |
|||
(which-key-setup-side-window-right-bottom) |
|||
(which-key-setup-minibuffer) |
|||
(setq which-key-idle-delay 0.5)) |
|||
|
|||
(use-package org |
|||
:ensure org-plus-contrib |
|||
:pin org) |
|||
:ensure org-plus-contrib |
|||
:init |
|||
(add-hook 'org-mode-hook 'company/org-mode-hook) |
|||
:config |
|||
;; (require 'org-id) |
|||
(add-to-list 'org-modules "org-id") |
|||
(setq org-default-notes-file (concat MY--PATH_ORG_FILES "notes.org") |
|||
org-agenda-files (list MY--PATH_ORG_FILES |
|||
MY--PATH_ORG_FILES_MOBILE) |
|||
org-id-locations-file (concat MY--PATH_USER_LOCAL ".org-id-locations") |
|||
org-log-into-drawer "LOGBOOK")) |
|||
(org-id-update-id-locations) |
|||
|
|||
(require 'org-habit) ;;TODO Lösung ohne require finden, scheint mir nicht ideal zu sein, nur um ein org-modul zu aktivieren |
|||
;; (add-to-list 'org-modules "org-habit") |
|||
(setq org-habit-graph-column 80 |
|||
org-habit-preceding-days 30 |
|||
org-habit-following-days 7 |
|||
org-habit-show-habits-only-for-today nil) |
|||
|
|||
(use-package org-journal |
|||
:ensure t |
|||
:defer t |
|||
:custom |
|||
(org-journal-dir MY--PATH_ORG_JOURNAl) |
|||
(org-journal-enable-agenda-integration t)) |
|||
|
|||
(require 'ivy) |
|||
(use-package ivy |
|||
:ensure t |
|||
:diminish |
|||
(ivy-mode . "") |
|||
:init |
|||
(ivy-mode 1) |
|||
:bind |
|||
("C-r" . ivy-resume) ;; overrides isearch-backwards binding |
|||
:config |
|||
(setq ivy-use-virtual-buffers t ;; recent files and bookmarks in ivy-switch-buffer |
|||
ivy-height 20 ;; height of ivy window |
|||
ivy-count-format "%d/%d" ;; current and total number |
|||
ivy-re-builders-alist ;; regex replaces spaces with * |
|||
'((t . ivy--regex-plus)))) |
|||
|
|||
(use-package counsel |
|||
:ensure t |
|||
:bind* |
|||
(("M-x" . counsel-M-x) |
|||
("C-x C-f" . counsel-find-file) |
|||
("C-x C-r" . counsel-recentf) |
|||
("C-c C-f" . counsel-git) |
|||
("C-c h f" . counsel-describe-function) |
|||
("C-c h v" . counsel-describe-variable) |
|||
("M-i" . counsel-imenu))) |
|||
|
|||
(use-package swiper |
|||
:ensure t |
|||
:bind |
|||
("C-s" . swiper)) |
|||
|
|||
(use-package ivy-hydra |
|||
:ensure t) |
|||
|
|||
(require 'company) |
|||
(use-package company |
|||
:defer 1 |
|||
:bind |
|||
(:map company-active-map |
|||
("RET" . nil) |
|||
([return] . nil) |
|||
("TAB" . company-complete-selection) |
|||
([tab] . company-complete-selection) |
|||
("<right>" . company-complete-common)) |
|||
:config |
|||
(global-company-mode 1) |
|||
(setq-default |
|||
company-idle-delay .2 |
|||
company-minimum-prefix-length 1 |
|||
company-require-match nil |
|||
company-show-numbers t |
|||
company-tooltip-align-annotations t)) |
|||
|
|||
(require 'company-statistics) |
|||
(use-package company-statistics |
|||
:ensure t |
|||
:after company |
|||
:init |
|||
(setq company-statistics-file (concat MY--PATH_USER_LOCAL "company-statistics-cache.el"));~/.emacs.d/user-dir/company-statistics-cache.el") |
|||
:config |
|||
(company-statistics-mode 1)) |
|||
|
|||
(use-package company-dabbrev |
|||
:ensure nil |
|||
:after company |
|||
:config |
|||
(setq-default company-dabbrev-downcase nil)) |
|||
|
|||
(use-package company-box |
|||
:ensure nil |
|||
:init |
|||
(add-hook 'company-mode-hook 'company-box-mode)) |
|||
|
|||
(defun company/org-mode-hook() |
|||
(set (make-local-variable 'company-backends) |
|||
'(company-capf company-files)) |
|||
(add-hook 'completion-at-point-functions 'pcomplete-completions-at-point nil t) |
|||
(message "company/org-mode-hook")) |
|||
|
|||
(defun company/elisp-mode-hook() |
|||
(set (make-local-variable 'company-backends) |
|||
'((company-elisp company-dabbrev) company-capf company-files)) |
|||
(message "company/elisp-mode-hook")) |
|||
|
|||
(defun company/beancount-mode-hook() |
|||
(set (make-local-variable 'company-backends) |
|||
'(company-beancount))) |
|||
|
|||
(use-package magit |
|||
:ensure t |
|||
:defer t |
|||
:init |
|||
; set git-path in work environment |
|||
(if (string-equal user-login-name "POH") |
|||
(setq magit-git-executable "P:/Eigene Dateien/Tools/Git/bin/git.exe") |
|||
) |
|||
:bind (("C-x g" . magit-status)) |
|||
) |
|||
|
|||
(use-package lsp-mode |
|||
:defer t |
|||
:commands lsp |
|||
:custom |
|||
(lsp-auto-guess-root nil) |
|||
(lsp-prefer-flymake nil) ; use flycheck instead |
|||
(lsp-file-watch-threshold 2000) |
|||
:bind (:map lsp-mode-map ("C-c C-f" . lsp-format-buffer)) |
|||
:hook ((python-mode |
|||
js-mode |
|||
js2-mode |
|||
typescript-mode |
|||
web-mode) . lsp)) |
|||
|
|||
(use-package lsp-ui |
|||
:after lsp-mode |
|||
:diminish |
|||
:commands lsp-ui-mode |
|||
:config |
|||
(setq lsp-ui-doc-enable t |
|||
lsp-ui-doc-header t |
|||
lsp-ui-doc-include-signature t |
|||
lsp-ui-doc-position 'top |
|||
lsp-ui-doc-border (face-foreground 'default) |
|||
lsp-ui-sideline-enable nil |
|||
lsp-ui-sideline-ignore-duplicate t |
|||
lsp-ui-sideline-show-code-actions nil) |
|||
(when (display-graphic-p) |
|||
(setq lsp-ui-doc-use-webkit t)) |
|||
;; workaround hide mode-line of lsp-ui-imenu buffer |
|||
(defadvice lsp-ui-imenu (after hide-lsp-ui-imenu-mode-line activate) |
|||
(setq mode-line-format nil))) |
|||
|
|||
(use-package company-lsp |
|||
:requires company |
|||
:defer t |
|||
:ensure t |
|||
:config |
|||
;;disable client-side cache because lsp server does a better job |
|||
(setq company-transformers nil |
|||
company-lsp-async t |
|||
company-lsp-cache-candidates nil)) |
|||
|
|||
(use-package flycheck |
|||
;;:ensure t |
|||
:hook |
|||
((css-mode . flycheck-mode) |
|||
(emacs-lisp-mode . flycheck-mode) |
|||
(python-mode . flycheck-mode)) |
|||
:init |
|||
(setq flycheck-emacs-lisp-load-path 'inherit) |
|||
:config |
|||
(setq-default |
|||
flycheck-check-synta-automatically '(save mode-enabled) |
|||
flycheck-disable-checkers '(emacs-lisp-checkdoc) |
|||
eldoc-idle-delay .1 ;; let eldoc echo faster than flycheck |
|||
flycheck-display-errors-delay .3)) ;; this way any errors will override eldoc messages |
|||
|
|||
(add-hook 'emacs-lisp-mode-hook 'company/elisp-mode-hook) |
|||
|
|||
(use-package web-mode |
|||
:ensure t |
|||
:defer t |
|||
:mode |
|||
("\\.phtml\\'" |
|||
"\\.tpl\\.php\\'" |
|||
"\\.djhtml\\'" |
|||
"\\.[t]?html?\\'")) |
|||
|
|||
(use-package pyvenv |
|||
:ensure t |
|||
:config |
|||
(setenv "WORKON_HOME" (expand-file-name "~/Archiv/Programmierprojekte/Python/virtualenv/")) |
|||
(add-hook 'pyvenv-post-activate-hooks #'my/postactivatehook)) |
|||
|
|||
(defun my/postactivatehook () |
|||
(setq lsp-python-ms-extra-paths pyvenv-virtual-env)) |
|||
|
|||
(use-package virtualenvwrapper |
|||
:ensure t |
|||
:hook (venv-postmkvirtualenv . (lambda() (shell-command "pip3 install importmagic epc"))) |
|||
:config |
|||
(setq venv-location (expand-file-name "~/Archiv/Programmierprojekte/Python/virtualenv/"))) |
|||
|
|||
;; Load the config |
|||
(org-babel-load-file (concat user-emacs-directory "config.org")) |
|||
(use-package lsp-python-ms |
|||
:ensure t |
|||
:after lsp-mode python) |
|||
; :custom (lsp-python-executable-cmd "python3")) |
|||
|
|||
(setq gc-cons-threshold 800000) |
|||
(use-package beancount |
|||
:load-path "user-local/elisp" |
|||
:defer t |
|||
:mode |
|||
("\\.beancount$" . beancount-mode) |
|||
:init |
|||
(add-hook 'beancount-mode-hook 'company/beancount-mode-hook) |
|||
; (add-hook 'beancount-mode-hook (pyvenv-activate "/opt/beancount")) |
|||
; (setenv "PATH" |
|||
; (concat "/opt/beancount/bin:" |
|||
; (getenv "PATH"))) |
|||
:config |
|||
(setq beancount-filename-main "/home/marc/Archiv/Finanzen/Transaktionen/transactions.beancount")) |
@ -0,0 +1,551 @@ |
|||
#+TITLE: Emacs configuration file |
|||
#+AUTHOR: Marc |
|||
#+BABEL: :cache yes |
|||
#+PROPERTY: header-args :tangle yes |
|||
|
|||
* TODOS |
|||
- Paket exec-path-from-shell, um PATH aus Linux auch in emacs zu haben |
|||
|
|||
* First start |
|||
When pulling the repository the first time, an initial init.el needs to be setup. After start it will replace itself with the configuration from init.org |
|||
|
|||
#+BEGIN_SRC emacs-lisp :tangle no |
|||
(require 'org') |
|||
(find-file (concat user-emacs-directory "init.org")) |
|||
(org-babel-tangle) |
|||
(load-file (concat user-emacs-directory "init.el")) |
|||
(byte-compile-file (concat user-emacs-directory "init.el")) |
|||
#+END_SRC |
|||
|
|||
This function updates init.el whenever changes in init.org are made. The update will be active after saving. |
|||
|
|||
#+BEGIN_SRC emacs-lisp |
|||
(defun me/tangle-init () |
|||
"If the current buffer is 'init.org', |
|||
the code blocks are tangled, and the tangled file is compiled." |
|||
(when (equal (buffer-file-name) |
|||
(expand-file-name (concat user-emacs-directory "init.org"))) |
|||
;; avoid running hooks |
|||
(let ((prog-mode-hook nil)) |
|||
(org-babel-tangle) |
|||
(byte-compile-file (concat user-emacs-directory "init.el")) |
|||
(load-file user-init-file)))) |
|||
(add-hook 'after-save-hook 'me/tangle-init) |
|||
#+END_SRC |
|||
|
|||
#+BEGIN_SRC emacs-lisp |
|||
(require 'package) |
|||
|
|||
;; bug before emacs 26.3 |
|||
(setq gnutls-algorithm-priority "NORMAL:-VERS-TLS1.3") |
|||
(add-to-list 'package-archives '("elpa" . "https://elpa.gnu.org/packages/") t) |
|||
(add-to-list 'package-archives '("melpa" . "https://melpa.org/packages/") t) |
|||
(add-to-list 'package-archives '("melpa-stable" . "https://stable.melpa.org/packages/") t) |
|||
(add-to-list 'package-archives '("org" . "https://orgmode.org/elpa/") t) |
|||
|
|||
(package-initialize) |
|||
#+END_SRC |
|||
|
|||
#+BEGIN_SRC emacs-lisp |
|||
(unless (package-installed-p 'use-package) |
|||
(package-refresh-contents) |
|||
(package-install 'use-package)) |
|||
|
|||
(setq use-package-verbose nil) |
|||
|
|||
(eval-when-compile |
|||
(require 'use-package)) |
|||
(require 'bind-key) |
|||
(use-package diminish |
|||
:ensure t) |
|||
#+END_SRC |
|||
|
|||
* Default settings |
|||
|
|||
#+BEGIN_SRC emacs-lisp |
|||
(defvar MY--PATH_USER_LOCAL (expand-file-name "~/.emacs.d/user-local/")) |
|||
(defvar MY--PATH_USER_GLOBAL (expand-file-name "~/.emacs.d/user-global/")) |
|||
|
|||
(defvar MY--PATH_ORG_FILES (expand-file-name "~/Archiv/Organisieren/")) |
|||
(defvar MY--PATH_ORG_FILES_MOBILE (expand-file-name "~/Archiv/Organisieren/mobile/")) |
|||
(defvar MY--PATH_ORG_JOURNAl (expand-file-name "~/Archiv/Organisieren/Journal/")) |
|||
(setq bookmark-default-file (concat MY--PATH_USER_LOCAL "bookmarks")) |
|||
(setq recentf-save-file (concat MY--PATH_USER_LOCAL "recentf")) |
|||
(setq custom-file (concat MY--PATH_USER_LOCAL "custom.el")) ;; don't spam init.e with saved customization settings |
|||
(setq abbrev-file-name (concat MY--PATH_USER_GLOBAL "abbrev_defs")) |
|||
(setq backup-directory-alist `((".*" . ,temporary-file-directory))) |
|||
(setq auto-save-file-name-transforms `((".*" ,temporary-file-directory))) |
|||
(setq save-abbrevs 'silently) ;; don't bother me with asking for abbrev saving |
|||
(setq-default create-lockfiles nil) ;; disable lock files, can cause trouble in e.g. lsp-mode |
|||
(defalias 'yes-or-no-p 'y-or-n-p) ;; answer with y and n |
|||
(setq custom-safe-themes t) ;; don't ask me if I want to load a theme |
|||
(setq sentence-end-double-space nil) ;; don't coun two spaces after a period as the end of a sentence. |
|||
(delete-selection-mode t) ;; delete selected region when typing |
|||
|
|||
(setq locale-coding-system 'utf-8) |
|||
(set-terminal-coding-system 'utf-8) |
|||
(set-keyboard-coding-system 'utf-8) |
|||
(set-selection-coding-system 'utf-8) |
|||
(if (eq system-type 'windows-nt) |
|||
(prefer-coding-system 'utf-8dos) |
|||
(prefer-coding-system 'utf-8)) |
|||
(blink-cursor-mode -1) ;; turn off blinking cursor |
|||
(show-paren-mode t) ;; show other part of brackets |
|||
(column-number-mode t) |
|||
(setq uniquify-buffer-name-style 'forward) |
|||
(setq-default indent-tabs-mode nil) ;; avoid tabs in place of multiple spaces (they look bad in tex) |
|||
(setq-default indicate-empty-lines t) ;; show empty lines |
|||
(setq scroll-margin 5 ;; smooth scrolling |
|||
scroll-conservatively 10000 |
|||
scroll-preserve-screen-position 1 |
|||
scroll-step 1) |
|||
(global-hl-line-mode t) ;; highlight current line |
|||
(menu-bar-mode 0) ;; disable menu bar |
|||
(tool-bar-mode 0) ;; disable tool bar |
|||
(scroll-bar-mode 0) ;; disable scroll bar |
|||
#+END_SRC |
|||
* visuals |
|||
** Font |
|||
#+BEGIN_SRC emacs-lisp |
|||
(set-face-font 'default "Hack-10") |
|||
#+END_SRC |
|||
|
|||
** line wrappings |
|||
#+BEGIN_SRC emacs-lisp |
|||
(global-visual-line-mode) |
|||
(diminish 'visual-line-mode) |
|||
(use-package adaptive-wrap |
|||
:ensure t |
|||
:init |
|||
(when (fboundp 'adaptive-wrap-prefix-mode) |
|||
(defun my/activate-adaptive-wrap-prefix-mode () |
|||
"Toggle `visual-line-mode' and `adaptive-wrap-prefix-mode' simultaneously." |
|||
(adaptive-wrap-prefix-mode (if visual-line-mode 1 -1))) |
|||
(add-hook 'visual-line-mode-hook 'my/activate-adaptive-wrap-prefix-mode))) |
|||
#+END_SRC |
|||
** line numbers |
|||
#+BEGIN_SRC emacs-lisp |
|||
(use-package display-line-numbers |
|||
:init |
|||
(add-hook 'prog-mode-hook 'display-line-numbers-mode) |
|||
(add-hook 'org-src-mode-hook 'display-line-numbers-mode) |
|||
:config |
|||
(setq-default display-line-numbers-type 'visual |
|||
display-line-numbers-current-absolute t |
|||
display-line-numbers-with 4 |
|||
display-line-numbers-widen t)) |
|||
; (add-hook 'emacs-lisp-mode-hook 'display-line-numbers-mode) |
|||
#+END_SRC |
|||
** misc |
|||
#+BEGIN_SRC emacs-lisp |
|||
(use-package rainbow-mode |
|||
:ensure t |
|||
:diminish |
|||
:hook ((org-mode |
|||
emacs-lisp-mode) . rainbow-mode)) |
|||
#+END_SRC |
|||
* undo |
|||
#+BEGIN_SRC emacs-lisp |
|||
(use-package undo-tree |
|||
:ensure t |
|||
:diminish undo-tree-mode |
|||
:init |
|||
(global-undo-tree-mode 1)) |
|||
#+END_SRC |
|||
* imenu-list |
|||
A minor mode to show imenu in a sidebar. |
|||
Call imenu-list-smart-toggle. |
|||
[[https://github.com/bmag/imenu-list][Source]] |
|||
|
|||
#+BEGIN_SRC emacs-lisp |
|||
(use-package imenu-list |
|||
:ensure t |
|||
:config |
|||
(setq imenu-list-focus-after-activation t |
|||
imenu-list-auto-resize t |
|||
imenu-list-position 'right) |
|||
:bind |
|||
(:map global-map |
|||
([f9] . imenu-list-smart-toggle)) |
|||
) |
|||
#+END_SRC |
|||
* which-key |
|||
#+BEGIN_SRC emacs-lisp |
|||
(use-package which-key |
|||
:ensure t |
|||
:diminish which-key-mode |
|||
:config |
|||
(which-key-mode) |
|||
(which-key-setup-side-window-right-bottom) |
|||
(which-key-setup-minibuffer) |
|||
(setq which-key-idle-delay 0.5)) |
|||
#+END_SRC |
|||
|
|||
* orgmode |
|||
** org |
|||
#+BEGIN_SRC emacs-lisp |
|||
(use-package org |
|||
:ensure org-plus-contrib |
|||
:init |
|||
(add-hook 'org-mode-hook 'company/org-mode-hook) |
|||
:config |
|||
;; (require 'org-id) |
|||
(add-to-list 'org-modules "org-id") |
|||
(setq org-default-notes-file (concat MY--PATH_ORG_FILES "notes.org") |
|||
org-agenda-files (list MY--PATH_ORG_FILES |
|||
MY--PATH_ORG_FILES_MOBILE) |
|||
org-id-locations-file (concat MY--PATH_USER_LOCAL ".org-id-locations") |
|||
org-log-into-drawer "LOGBOOK")) |
|||
(org-id-update-id-locations) |
|||
#+END_SRC |
|||
** habits |
|||
#+BEGIN_SRC emacs-lisp |
|||
(require 'org-habit) ;;TODO Lösung ohne require finden, scheint mir nicht ideal zu sein, nur um ein org-modul zu aktivieren |
|||
;; (add-to-list 'org-modules "org-habit") |
|||
(setq org-habit-graph-column 80 |
|||
org-habit-preceding-days 30 |
|||
org-habit-following-days 7 |
|||
org-habit-show-habits-only-for-today nil) |
|||
#+END_SRC |
|||
** *TODO* |
|||
org-super-agenda |
|||
|
|||
** org-caldav |
|||
Vorerst deaktiviert, Nutzen evtl. nicht vorhanden |
|||
BEGIN_SRC emacs-lisp |
|||
(use-package org-caldav |
|||
:ensure t |
|||
:config |
|||
(setq org-caldav-url "https://nextcloud.cloudsphere.duckdns.org/remote.php/dav/calendars/marc" |
|||
org-caldav-calendar-id "orgmode" |
|||
org-caldav-inbox (expand-file-name "~/Archiv/Organisieren/caldav-inbox") |
|||
org-caldav-files (concat MY--PATH_ORG_FILES "tasks"))) |
|||
END_SRC |
|||
** journal |
|||
[[https://github.com/bastibe/org-journal][Source]] |
|||
|
|||
#+BEGIN_SRC emacs-lisp |
|||
(use-package org-journal |
|||
:ensure t |
|||
:defer t |
|||
:custom |
|||
(org-journal-dir MY--PATH_ORG_JOURNAl) |
|||
(org-journal-enable-agenda-integration t)) |
|||
#+END_SRC |
|||
* ivy / counsel / swiper |
|||
|
|||
#+BEGIN_SRC emacs-lisp |
|||
; (require 'ivy) |
|||
(use-package ivy |
|||
:ensure t |
|||
:diminish |
|||
(ivy-mode . "") |
|||
:init |
|||
(ivy-mode 1) |
|||
:bind |
|||
("C-r" . ivy-resume) ;; overrides isearch-backwards binding |
|||
:config |
|||
(setq ivy-use-virtual-buffers t ;; recent files and bookmarks in ivy-switch-buffer |
|||
ivy-height 20 ;; height of ivy window |
|||
ivy-count-format "%d/%d" ;; current and total number |
|||
ivy-re-builders-alist ;; regex replaces spaces with * |
|||
'((t . ivy--regex-plus)))) |
|||
|
|||
(use-package counsel |
|||
:ensure t |
|||
:bind* |
|||
(("M-x" . counsel-M-x) |
|||
("C-x C-f" . counsel-find-file) |
|||
("C-x C-r" . counsel-recentf) |
|||
("C-c C-f" . counsel-git) |
|||
("C-c h f" . counsel-describe-function) |
|||
("C-c h v" . counsel-describe-variable) |
|||
("M-i" . counsel-imenu))) |
|||
|
|||
(use-package swiper |
|||
:ensure t |
|||
:bind |
|||
("C-s" . swiper)) |
|||
|
|||
(use-package ivy-hydra |
|||
:ensure t) |
|||
#+END_SRC |
|||
|
|||
* company |
|||
|
|||
#+BEGIN_SRC emacs-lisp |
|||
; (require 'company) |
|||
(use-package company |
|||
:defer 1 |
|||
:bind |
|||
(:map company-active-map |
|||
("RET" . nil) |
|||
([return] . nil) |
|||
("TAB" . company-complete-selection) |
|||
([tab] . company-complete-selection) |
|||
("<right>" . company-complete-common)) |
|||
:config |
|||
(global-company-mode 1) |
|||
(setq-default |
|||
company-idle-delay .2 |
|||
company-minimum-prefix-length 1 |
|||
company-require-match nil |
|||
company-show-numbers t |
|||
company-tooltip-align-annotations t)) |
|||
|
|||
; (require 'company-statistics) |
|||
(use-package company-statistics |
|||
:ensure t |
|||
:after company |
|||
:init |
|||
(setq company-statistics-file (concat MY--PATH_USER_LOCAL "company-statistics-cache.el"));~/.emacs.d/user-dir/company-statistics-cache.el") |
|||
:config |
|||
(company-statistics-mode 1)) |
|||
|
|||
(use-package company-dabbrev |
|||
:ensure nil |
|||
:after company |
|||
:config |
|||
(setq-default company-dabbrev-downcase nil)) |
|||
|
|||
(use-package company-box |
|||
:ensure t |
|||
:init |
|||
(add-hook 'company-mode-hook 'company-box-mode)) |
|||
#+END_SRC |
|||
|
|||
** company backends |
|||
|
|||
#+BEGIN_SRC emacs-lisp |
|||
(defun company/org-mode-hook() |
|||
(set (make-local-variable 'company-backends) |
|||
'(company-capf company-files)) |
|||
(add-hook 'completion-at-point-functions 'pcomplete-completions-at-point nil t) |
|||
(message "company/org-mode-hook")) |
|||
|
|||
(defun company/elisp-mode-hook() |
|||
(set (make-local-variable 'company-backends) |
|||
'((company-elisp company-dabbrev) company-capf company-files)) |
|||
(message "company/elisp-mode-hook")) |
|||
|
|||
(defun company/beancount-mode-hook() |
|||
(set (make-local-variable 'company-backends) |
|||
'(company-beancount))) |
|||
#+END_SRC |
|||
|
|||
* Programming |
|||
** Magit / Git |
|||
Little crash course in magit: |
|||
- magit-init to init a git project |
|||
- magit-status (C-x g) to call the status window |
|||
|
|||
In status buffer: |
|||
- s stage files |
|||
- u unstage files |
|||
- U unstage all files |
|||
- a apply changes to staging |
|||
- c c commit (type commit message, then C-c C-c to commit) |
|||
- b b switch to another branch |
|||
- P u git push |
|||
- F u git pull |
|||
|
|||
#+BEGIN_SRC emacs-lisp |
|||
(use-package magit |
|||
:ensure t |
|||
:defer t |
|||
:init |
|||
; set git-path in work environment |
|||
(if (string-equal user-login-name "POH") |
|||
(setq magit-git-executable "P:/Eigene Dateien/Tools/Git/bin/git.exe") |
|||
) |
|||
:bind (("C-x g" . magit-status)) |
|||
) |
|||
#+END_SRC |
|||
|
|||
** LSP |
|||
Configuration for the language server protocol |
|||
*ACHTUNG* Dateipfad muss absolut sein, symlink im Pfad führt zumindest beim ersten Start zu Fehlern beim lsp |
|||
Sobald der lsp einmal lief, kann zukünftig der symlink-Pfad genommen werden. |
|||
Getestet wurde die funktionierende Datei selbst und neu erstellte Dateien im selben Pfad. |
|||
TODO Unterverzeichnisse wurden noch nicht getestet |
|||
|
|||
#+BEGIN_SRC emacs-lisp |
|||
(use-package lsp-mode |
|||
:defer t |
|||
:commands lsp |
|||
:custom |
|||
(lsp-auto-guess-root nil) |
|||
(lsp-prefer-flymake nil) ; use flycheck instead |
|||
(lsp-file-watch-threshold 2000) |
|||
:bind (:map lsp-mode-map ("C-c C-f" . lsp-format-buffer)) |
|||
:hook ((python-mode |
|||
js-mode |
|||
js2-mode |
|||
typescript-mode |
|||
web-mode) . lsp)) |
|||
|
|||
(use-package lsp-ui |
|||
:after lsp-mode |
|||
:ensure t |
|||
:diminish |
|||
:commands lsp-ui-mode |
|||
:config |
|||
(setq lsp-ui-doc-enable t |
|||
lsp-ui-doc-header t |
|||
lsp-ui-doc-include-signature t |
|||
lsp-ui-doc-position 'top |
|||
lsp-ui-doc-border (face-foreground 'default) |
|||
lsp-ui-sideline-enable nil |
|||
lsp-ui-sideline-ignore-duplicate t |
|||
lsp-ui-sideline-show-code-actions nil) |
|||
(when (display-graphic-p) |
|||
(setq lsp-ui-doc-use-webkit t)) |
|||
;; workaround hide mode-line of lsp-ui-imenu buffer |
|||
(defadvice lsp-ui-imenu (after hide-lsp-ui-imenu-mode-line activate) |
|||
(setq mode-line-format nil))) |
|||
|
|||
(use-package company-lsp |
|||
:requires company |
|||
:defer t |
|||
:ensure t |
|||
:config |
|||
;;disable client-side cache because lsp server does a better job |
|||
(setq company-transformers nil |
|||
company-lsp-async t |
|||
company-lsp-cache-candidates nil)) |
|||
#+END_SRC |
|||
|
|||
** flycheck |
|||
#+BEGIN_SRC emacs-lisp |
|||
(use-package flycheck |
|||
:ensure t |
|||
:hook |
|||
((css-mode . flycheck-mode) |
|||
(emacs-lisp-mode . flycheck-mode) |
|||
(python-mode . flycheck-mode)) |
|||
:init |
|||
(setq flycheck-emacs-lisp-load-path 'inherit) |
|||
:config |
|||
(setq-default |
|||
flycheck-check-synta-automatically '(save mode-enabled) |
|||
flycheck-disable-checkers '(emacs-lisp-checkdoc) |
|||
eldoc-idle-delay .1 ;; let eldoc echo faster than flycheck |
|||
flycheck-display-errors-delay .3)) ;; this way any errors will override eldoc messages |
|||
#+END_SRC |
|||
|
|||
** lisp |
|||
#+BEGIN_SRC emacs-lisp |
|||
(add-hook 'emacs-lisp-mode-hook 'company/elisp-mode-hook) |
|||
#+END_SRC |
|||
** web |
|||
|
|||
apt install npm |
|||
sudo npm install -g vscode-html-languageserver-bin |
|||
evtl alternativ typescript-language-server? |
|||
|
|||
#+BEGIN_SRC emacs-lisp |
|||
(use-package web-mode |
|||
:ensure t |
|||
:defer t |
|||
:mode |
|||
("\\.phtml\\'" |
|||
"\\.tpl\\.php\\'" |
|||
"\\.djhtml\\'" |
|||
"\\.[t]?html?\\'")) |
|||
#+END_SRC |
|||
|
|||
** Python |
|||
Systemseitig muss python-language-server installiert sein: |
|||
pip3 install 'python-language-server[all]' |
|||
|
|||
für andere language servers |
|||
https://github.com/emacs-lsp/lsp-mode#install-language-server |
|||
|
|||
#+BEGIN_SRC emacs-lisp |
|||
(use-package pyvenv |
|||
:ensure t |
|||
:config |
|||
(setenv "WORKON_HOME" (expand-file-name "~/Archiv/Programmierprojekte/Python/virtualenv/")) |
|||
(add-hook 'pyvenv-post-activate-hooks #'my/postactivatehook)) |
|||
|
|||
(defun my/postactivatehook () |
|||
(setq lsp-python-ms-extra-paths pyvenv-virtual-env)) |
|||
|
|||
(use-package virtualenvwrapper |
|||
:ensure t |
|||
:hook (venv-postmkvirtualenv . (lambda() (shell-command "pip3 install importmagic epc"))) |
|||
:config |
|||
(setq venv-location (expand-file-name "~/Archiv/Programmierprojekte/Python/virtualenv/"))) |
|||
|
|||
(use-package lsp-python-ms |
|||
:ensure t |
|||
:after lsp-mode python) |
|||
; :custom (lsp-python-executable-cmd "python3")) |
|||
|
|||
|
|||
#+END_SRC |
|||
* beancount |
|||
** Installation |
|||
#+BEGIN_SRC shell |
|||
sudo su |
|||
cd /opt |
|||
python3 -m venv beancount |
|||
source ./beancount/bin/activate |
|||
pip3 install wheel |
|||
pip3 install beancount |
|||
sleep 100 |
|||
echo "shell running!" |
|||
deactivate |
|||
#+END_SRC |
|||
|
|||
#+BEGIN_SRC emacs-lisp |
|||
(use-package beancount |
|||
:load-path "user-local/elisp" |
|||
:ensure t |
|||
:defer t |
|||
:mode |
|||
("\\.beancount$" . beancount-mode) |
|||
:init |
|||
(add-hook 'beancount-mode-hook 'company/beancount-mode-hook) |
|||
; (add-hook 'beancount-mode-hook (pyvenv-activate "/opt/beancount")) |
|||
; (setenv "PATH" |
|||
; (concat "/opt/beancount/bin:" |
|||
; (getenv "PATH"))) |
|||
:config |
|||
(setq beancount-filename-main "/home/marc/Archiv/Finanzen/Transaktionen/transactions.beancount")) |
|||
#+END_SRC |
|||
|
|||
To support org-babel, check if it can find the symlink to ob-beancount.el |
|||
|
|||
#+BEGIN_SRC shell |
|||
orgpath=`find /home/marc/.emacs.d/elpa/ -type d -name "org-plus*" -print` |
|||
beansym="$orgpath/ob-beancount.el |
|||
bean="/home/marc/Archiv/Programmierprojekte/Lisp/beancount-mode/ob-beancount.el" |
|||
|
|||
if [ -h "$beansym" ] |
|||
then |
|||
echo "$beansym found" |
|||
elif [ -e "$bean" ] |
|||
then |
|||
echo "creating symlink" |
|||
ln -s "$bean" "$beansym" |
|||
else |
|||
echo "$bean not found, symlink creation aborted" |
|||
fi |
|||
#+END_SRC |
|||
|
|||
Fava is strongly recommended. |
|||
|
|||
#+BEGIN_SRC shell |
|||
cd /opt |
|||
python3 -m venv fava |
|||
source ./fava/bin/activate |
|||
pip3 install wheel |
|||
pip3 install fava |
|||
deactivate |
|||
#+END_SRC |
|||
|
|||
Start fava with fava my_file.beancount |
|||
|
|||
It is accessable on this URL: [[http://127.0.0.1:5000][Fava]] |
|||
Beancount-mode can start fava and open the URL right away. |
@ -0,0 +1,620 @@ |
|||
;;; package --- Summary |
|||
;;; Commentary: |
|||
;; A humble try to port leder-mode to beancount |
|||
|
|||
;;; Code: |
|||
;(require 'beancount-regex) |
|||
|
|||
(require 'font-lock) |
|||
(require 'company) ; for company-mode |
|||
(require 'pcomplete) |
|||
(require 'cl-lib) |
|||
|
|||
(defgroup beancount () |
|||
"Editing mode for beancount files." |
|||
:group 'beancount) |
|||
|
|||
(defconst beancount-timestamped-directive-names |
|||
'("balance" |
|||
"open" |
|||
"close" |
|||
"pad" |
|||
"document" |
|||
"note" |
|||
;; the ones below are not followed by an account name. |
|||
"event" |
|||
"price" |
|||
"commodity" |
|||
"query" |
|||
"txn") |
|||
"Directive names that can appear after a date.") |
|||
|
|||
(defconst beancount-account-categories |
|||
'("Assets" |
|||
"Liabilities" |
|||
"Equity" |
|||
"Income" |
|||
"Expenses" |
|||
; TODO: Fill changed root accounts automatically |
|||
"Aktiva" |
|||
"Verbindlichkeiten" |
|||
"Eigenkapital" |
|||
"Ertraege" |
|||
"Aufwendungen")) |
|||
|
|||
(defconst beancount-nontimestamped-directive-names |
|||
'("pushtag" |
|||
"poptag" |
|||
"option" |
|||
"include" |
|||
"plugin") |
|||
"Directive names that can appear after a date.") |
|||
|
|||
(defconst beancount-option-names |
|||
;; this list has to be kept in sync with the options definied in |
|||
;; beancount/parser/options.py |
|||
'("title" |
|||
"name_assets" |
|||
"name_equity" |
|||
"name_income" |
|||
"name_expenses" |
|||
"bookin_algorithm" |
|||
"bookin_method" |
|||
"account_previous_balances" |
|||
"account_previous_earnings" |
|||
"account_previous_conversions" |
|||
"account_current_earnings" |
|||
"account_current_conversions" |
|||
"account_rounding" |
|||
"conversion_currency" |
|||
"inferred_tolerance_default" |
|||
"inferred_tolerance_multiplier" |
|||
"infer_tolerance_from_cost" |
|||
"documents" |
|||
"operating_currency" |
|||
"render_commas" |
|||
"plugin_processing_mode" |
|||
"plugin" |
|||
"long_string_maxlines" |
|||
)) |
|||
|
|||
(defconst beancount-directive-names |
|||
(append beancount-nontimestamped-directive-names |
|||
beancount-timestamped-directive-names) |
|||
"A list of directive names.") |
|||
|
|||
(defconst beancount-tag-chars |
|||
"[:alnum:]-_/." |
|||
"Allowed tag characters.") |
|||
|
|||
(defconst beancount-account-chars |
|||
"[:alnum:]-_:" |
|||
"Allowed account characters.") |
|||
|
|||
(defconst beancount-account-regexp |
|||
(concat (regexp-opt beancount-account-categories) |
|||
"\\(?::[[:upper:]][" beancount-account-chars "]+\\)") |
|||
"A regular expression to match account names.") |
|||
|
|||
; TODO: currently shows all texts between "" |
|||
(defconst beancount-payee-regexp |
|||
"\"\\(.*?\\)\"") |
|||
|
|||
(defconst beancount-date-regexp |
|||
"^[12][901][0-9]\\{2\\}-\\(\\(0[1-9]\\)\\|\\(1[012]\\)\\)-\\(\\([012][0-9]\\)\\|\\(3[01]\\)\\)" |
|||
"Regular expression for dates.") |
|||
|
|||
(defconst beancount-number-regexp |
|||
"[-+]?[0-9,]+\\(?:\\.[0-9]*\\)" |
|||
"Regular expression to match decimal numbers.") |
|||
|
|||
(defconst beancount-tag-regexp |
|||
(concat "#" |
|||
"[" |
|||
beancount-tag-chars |
|||
"]+") |
|||
"Regular expression for valid tags.") |
|||
|
|||
(defconst beancount-currency-regexp |
|||
"[A-Z][A-Z-_'.]*" |
|||
"Regular expression to match currencies.") |
|||
|
|||
|
|||
(defconst beancount-timestamped-accounts-regexp |
|||
(concat beancount-date-regexp |
|||
" " ;"\\(\\s-+\\)" |
|||
(regexp-opt beancount-timestamped-directive-names) |
|||
" ") ;"\\(\\s-+\\)") |
|||
"A regular expression to match valid preceding characters before an account name.") |
|||
|
|||
(defconst beancount-amount-and-currency-regex |
|||
"\s-*[-]?[0-9,]+[.][0-9]\\{2\\}\s[A-Za-z0-9.]+" |
|||
"A regular expression for amounts including currency.") |
|||
|
|||
(defconst beancount-payee-any-status-regex |
|||
"^[0-9]+[-/][-/.=0-9]+\\(\\s-+\\*\\)?\\(\\s-+(.*?)\\)?\\s-+\\(.+?\\)\\s-*\\(;\\|$\\)") |
|||
|
|||
(defconst beancount-date-and-status-regex |
|||
(concat beancount-date-regexp |
|||
"\\(\\s-+[\\*!]\\)") |
|||
"Returns true for YYYY-MM-DD ! and YYYY-MM-DD *.") |
|||
|
|||
(defconst beancount-valid-prefix-for-directives-regex |
|||
(concat "^" |
|||
beancount-date-regexp |
|||
"[ ]+" |
|||
"[a-z]*$")) |
|||
|
|||
(defconst beancount-valid-prefix-for-payee-completion-regex |
|||
(concat "\\(" |
|||
beancount-date-and-status-regex |
|||
" \"" ; empty space and open quotes |
|||
"[^\"\n]*" ; any number of chars except quotes and newline |
|||
"\\(" ; start of optional second quoted term |
|||
"\"[ ]+\"" ; closing quotes, whitespaces, opening quotes |
|||
"[^\"\n]*" ; any number of chars except quotes and newline |
|||
"\\)?" ; end of optional second quoted term |
|||
"\\)$" ; the whole regex looks from the right side of the line |
|||
)) |
|||
|
|||
(defconst beancount-valid-prefix-for-tag-completion-regex |
|||
(concat ;"\\(" |
|||
beancount-date-and-status-regex |
|||
" \"" ; empty space and open quotes |
|||
"[^\"\n]*" ; any number of chars except quotes and newline |
|||
"\\(" ; start of optional second quoted term |
|||
"\"[ ]+\"" ; closing quotes, whitespaces, opening quotes |
|||
"[^\"\n]*" ; any number of chars except quotes and newline |
|||
"\\)?" ; end of optional second quoted term |
|||
"\"[ ]+#" |
|||
|
|||
;"\\)$" ; the whole regex looks from the right side of the line |
|||
)) |
|||
|
|||
(defconst beancount-comments-regex |
|||
(concat ";[^\"\n]*$")) ; right part of the line after a comment symbol if no quote or newline is included |
|||
|
|||
(defconst beancount-empty-line-regex |
|||
"^\\(\\s-+\\)" ;; maybe "^[ \t]+" is better |
|||
"Returns true for preceding whitespaces.") |
|||
|
|||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; |
|||
;; Faces |
|||
(defgroup beancount-faces nil "Beancount mode highlighting" :group 'beancount) |
|||
|
|||
(defface beancount-font-default-face |
|||
`((t :inherit default)) |
|||
"Default face" |
|||
:group 'beancount-faces) |
|||
|
|||
(defface beancount-font-xact-cleared-face |
|||
`((t :foreground "#AAAAAA" :weight normal)) |
|||
"Default face for cleared transactions" |
|||
:group 'beancount-faces) |
|||
|
|||
(defface beancount-font-xact-pending-face |
|||
`((t :foreground "#dc322f" :weight bold)) |
|||
"Default face for pending transactions" |
|||
:group 'beancount-faces) |
|||
|
|||
(defface beancount-font-payee-cleared-face |
|||
`((t :inherit beancount-font-other-face)) |
|||
"Default face for pending transactions" |
|||
:group 'beancount-faces) |
|||
|
|||
(defface beancount-font-payee-pending-face |
|||
`((t :foreground "#f24b61" :weight normal)) |
|||
"Default face for pending (!) transactions" |
|||
:group 'beancount-faces) |
|||
|
|||
(defface beancount-font-other-face |
|||
`((t :foreground "#657b83" :weight normal)) |
|||
"Default face for other transactions" |
|||
:group 'beancount-faces) |
|||
|
|||
(defface beancount-font-posting-date-face |
|||
`((t :foreground "#cb4b16" :weight normal)) |
|||
"Default face for dates" |
|||
:group 'beancount-faces) |
|||
|
|||
(defface beancount-font-amount-face |
|||
`((t :foreground "#cb4b16" :weight normal)) |
|||
"Default face for amounts" |
|||
:group 'beancount-faces) |
|||
|
|||
|
|||
(defvar beancount-font-lock-directives |
|||
`(;; reserved keywords |
|||
(,(regexp-opt beancount-directive-names) . font-lock-keyword-face) |
|||
;; tags & links |
|||
("[#\\^][A-Za-z0-9\-_/.]+" . font-lock-type-face) |
|||
;; comments |
|||
(,beancount-comments-regex (0 font-lock-comment-face)) |
|||
;; date |
|||
(,beancount-date-regexp . 'beancount-font-posting-date-face) |
|||
;; account |
|||
(,beancount-account-regexp . 'beancount-font-other-face) |
|||
;; payees |
|||
("\"\\(.*?\\)\"" . font-lock-comment-face) |
|||
;; txn flags |
|||
("! " . font-lock-warning-face) |
|||
)) |
|||
|
|||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; |
|||
;; Alignments |
|||
|
|||
(defmacro beancount-for-line-in-region (begin end &rest exprs) |
|||
"Iterate over each line in region from BEGIN to END (EXPRS) |
|||
until an empty line is encountered." |
|||
`(save-excursion |
|||
(let ((end-marker (copy-marker ,end))) |
|||
(goto-char ,begin) |
|||
(beginning-of-line) |
|||
(while (and (not (eobp)) (< (point) end-marker)) |
|||
(beginning-of-line) |
|||
(progn ,@exprs) |
|||
(forward-line 1) |
|||
)))) |
|||
|
|||
(defun beancount-align-numbers (begin end) |
|||
"Align all numbers in the current buffer from BEGIN to END." |
|||
(interactive "r") |
|||
|
|||
;; loop once in the buffer to find the length of the longest string before the |
|||
;; number. |
|||
(let (prefix-widths |
|||
number-widths |
|||
(number-padding " ")) |
|||
(beancount-for-line-in-region |
|||
begin end |
|||
(let ((line (thing-at-point 'line))) |
|||
(when (string-match (concat "\\(.*?\\)" |
|||
"[ \t]+" |
|||
"\\(" beancount-number-regexp "\\)" |
|||
"[ \t]+" |
|||
beancount-currency-regexp) |
|||
line) |
|||
(push (length (match-string 1 line)) prefix-widths) |
|||
(push (length (match-string 2 line)) number-widths) |
|||
))) |
|||
|
|||
(when prefix-widths |
|||
;; Loop again to make the adjustments to the numbers. |
|||
(let* ((number-width (apply 'max number-widths)) |
|||
(number-format (format "%%%ss" number-width)) |
|||
;; compute the rightmost column of prefix |
|||
(max-prefix-width (apply 'max prefix-widths)) |
|||
(prefix-format (format "%%-%ss" max-prefix-width)) |
|||
) |
|||
|
|||
(beancount-for-line-in-region |
|||
begin end |
|||
(let ((line (thing-at-point 'line))) |
|||
(when (string-match (concat "^\\([^\"]*?\\)" |
|||
"[ \t]+" |
|||
"\\(" beancount-number-regexp "\\)" |
|||
"[ \t]+" |
|||
"\\(.*\\)$") |
|||
line) |
|||
(delete-region (line-beginning-position) (line-end-position)) |
|||
(let* ((prefix (match-string 1 line)) |
|||
(number (match-string 2 line)) |
|||
(rest (match-string 3 line))) |
|||
(insert (format prefix-format prefix)) |
|||
(insert number-padding) |
|||
(insert (format number-format number)) |
|||
(insert " ") |
|||
(insert rest))))))))) |
|||
|
|||
|
|||
(defun beancount-hash-keys (hashtable) |
|||
"Extract all the keys of the given HASHTABLE. Return a sorted list." |
|||
(let (rlist) |
|||
(maphash (lambda (k _v) (push k rlist)) hashtable) |
|||
(sort rlist 'string<))) |
|||
|
|||
(defvar beancount-accounts nil |
|||
"A list of the accounts available in this buffer. |
|||
This is a cache of the value computed by `beancount-get-accounts'.") |
|||
(make-variable-buffer-local 'beancount-accounts) |
|||
|
|||
(defun beancount-init-accounts () |
|||
"Initialize or reset the list of accounts." |
|||
(interactive) |
|||
(setq beancount-accounts (beancount-get-accounts-in-buffer)) ;-new |
|||
(pcomplete-uniqify-list (nreverse beancount-accounts)) |
|||
;; (setq beancount-accounts (beancount-get-accounts)) |
|||
(message "Accounts updated.")) |
|||
|
|||
|
|||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; |
|||
;; Navigation |
|||
|
|||
(defun beancount-navigate-start-xact-or-directive-p () |
|||
"Return t if at the beginning of an empty or all-whitespace line." |
|||
(not (looking-at "[ \t]\\|\\(^$\\)"))) |
|||
|
|||
(defun beancount-navigate-prev-xact-or-directive () |
|||
"Move to the beginning of the next xact or directive." |
|||
(interactive) |
|||
(beginning-of-line) |
|||
(if (beancount-navigate-start-xact-or-directive-p) ;if we are at the start of an xact, move backward to the previous xact |
|||
(progn |
|||
(forward-line -1) |
|||
(if (not (beancount-navigate-start-xact-or-directive-p)) ; we have moved backward and are not at another xact, recurse backward |
|||
(beancount-navigate-prev-xact-or-directive))) |
|||
(while (not (or (bobp) |
|||
(beancount-navigate-start-xact-or-directive-p))) |
|||
(forward-line -1)))) |
|||
|
|||
(defun beancount-navigate-next-xact-or-directive () |
|||
"Move to the beginning of the next xact or directive." |
|||
(interactive) |
|||
(beginning-of-line) |
|||
(if (beancount-navigate-start-xact-or-directive-p) ; if we are at the start of an xact, move forward to the next xact |
|||
(progn |
|||
(forward-line) |
|||
(if (not (beancount-navigate-start-xact-or-directive-p)) ; we have moved forward and are not at another xact, recurse forward |
|||
(beancount-navigate-next-xact-or-directive))) |
|||
(while (not (or (eobp) ; we didn't stsrt off at the beginning of an xact |
|||
(beancount-navigate-start-xact-or-directive-p))) |
|||
(forward-line)))) |
|||
|
|||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; |
|||
;; Completion |
|||
|
|||
(defun beancount-thing-at-point () |
|||
"Describe thing at points. Return 'transaction, 'posting, or nil. |
|||
Leave point at the beginning of the thing under point.") |
|||
|
|||
(defun beancount-trim-trailing-whitespace (str) |
|||
"Replace trailing whitespaces in STR." |
|||
(replace-regexp-in-string "[ \t]*$" "" str)) |
|||
|
|||
;; (defun beancount-fully-complete-xact () |
|||
;; "Completes a transaction if there is another matching payee in the buffer." |
|||
;; (interactive) |
|||
;; (let* ((name (beancount-trim-trailing-whitespace (caar (beancount-parse-arguments)))) |
|||
;; xacf))) |
|||
|
|||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; |
|||
;; Functions for external programs |
|||
|
|||
(defvar beancount-filename-main buffer-file-name |
|||
"File name of the main beancount file for beancount-check.") |
|||
|
|||
(defvar beancount-terminal-name "urxvt" |
|||
"Name of the terminal emulator to run fava.") |
|||
|
|||
(defvar beancount-fava-exec "/opt/fava/bin/fava" |
|||
"Full path of fava executable.") |
|||
|
|||
(defvar beancount-install-dir nil |
|||
"Directory in which Beancount's source is located. |
|||
Only useful if you have not installed Beancount properly in your PATH") |
|||
|
|||
(defun beancount--run (prog &rest args) |
|||
"Random text PROG ARGS." |
|||
(let ((process-environment |
|||
(if beancount-install-dir |
|||
`(,(concat "PYTHONPATH=" beancount-install-dir) |
|||
,(concat "PATH=" |
|||
(expand-file-name "bin" beancount-install-dir) |
|||
":" |
|||
(getenv "PATH")) |
|||
,@process-environment) |
|||
process-environment)) |
|||
(compile-command (mapconcat (lambda (arg) |
|||
(if (stringp arg) |
|||
(shell-quote-argument arg) "")) |
|||
(cons prog args) |
|||
" "))) |
|||
(call-interactively 'compile))) |
|||
|
|||
(defvar beancount-check-program "bean-check" |
|||
"Program to run the parser and validator on an input file.") |
|||
|
|||
(defun beancount-check () |
|||
"Run `beancount-check-program'." |
|||
(interactive) |
|||
(let ((compilation-read-command nil)) |
|||
(beancount--run beancount-check-program beancount-filename-main))) |
|||
; (file-relative-name buffer-file-name)))) |
|||
|
|||
(defun beancount-fava () |
|||
"Run `beancount-fava' and open the URL in the default browser." |
|||
(interactive) |
|||
(start-process "termx" nil beancount-terminal-name "-e" beancount-fava-exec beancount-filename-main) |
|||
(sleep-for 0.5) ; necessary to prevent an error |
|||
(browse-url "127.0.0.1:5000")) |
|||
|
|||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; |
|||
;; Completions |
|||
|
|||
(defun beancount-get-accounts-in-buffer () |
|||
"Return a list of all accounts." |
|||
(let ((origin (point)) |
|||
accounts-list) |
|||
(save-excursion |
|||
(goto-char (point-min)) |
|||
(while (re-search-forward |
|||
beancount-account-regexp nil t) |
|||
(setq accounts-list (cons (match-string-no-properties 0) |
|||
accounts-list)))) |
|||
(pcomplete-uniqify-list (nreverse accounts-list)))) |
|||
|
|||
(defun beancount-get-payees-in-buffer () |
|||
"Return a list of all payees." |
|||
(let ((origin (point)) |
|||
payees-list) |
|||
(save-excursion |
|||
(goto-char (point-min)) |
|||
(while (re-search-forward |
|||
"^[0-9- *]*\"\\(.*?\\)\" \"\\(.*?\\)\"" nil t) ; matches are in brackets |
|||
(setq payees-list (cons (match-string-no-properties 1) ; get first match, generally the payee |
|||
payees-list)) |
|||
(setq payees-list (cons (match-string-no-properties 2) ; get second match, generally a description |
|||
payees-list)))) |
|||
(pcomplete-uniqify-list (nreverse payees-list)))) |
|||
|
|||
(defun beancount-get-tags-in-buffer () |
|||
"Return a list of all tags." |
|||
(let ((origin (point)) |
|||
tags-list) |
|||
(save-excursion |
|||
(goto-char (point-min)) |
|||
(while (re-search-forward |
|||
beancount-tag-regexp nil t) |
|||
(setq tags-list (cons (substring (match-string-no-properties 0) 1) |
|||
tags-list)))) |
|||
(pcomplete-uniqify-list (nreverse tags-list)))) |
|||
|
|||
(defun beancount-get-commodities-in-buffer () |
|||
"Return a list of all commodities / currencies." |
|||
(let ((origin (point)) |
|||
commodities-list) |
|||
(save-excursion |
|||
(goto-char (point-min)) |
|||
(while (re-search-forward |
|||
"^[ ]+[A-Za-z0-9-_:]+[ ]+[-]?[0-9,.]+[ ]+\\([A-Z][A-Z0-9-_.']*[A-Z0-9]\\)" nil t) |
|||
(setq commodities-list (cons (match-string-no-properties 1) |
|||
commodities-list)))) |
|||
(pcomplete-uniqify-list (nreverse commodities-list)))) |
|||
|
|||
|
|||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; |
|||
;; Finishing setup |
|||
|
|||
(defvar beancount-mode-map |
|||
(let ((map (make-sparse-keymap))) |
|||
(define-key map [(meta ?p)] #'beancount-navigate-prev-xact-or-directive) |
|||
(define-key map [(meta ?n)] #'beancount-navigate-next-xact-or-directive) |
|||
map) |
|||
"Keymap for `bean-mode'.") |
|||
|
|||
(easy-menu-define beancount-mode-menu beancount-mode-map |
|||
"Beancount menu" |
|||
'("Beancount" |
|||
["Customize Ledger Mode" (lambda () (interactive) (customize-group 'beanmode))])) |
|||
|
|||
|
|||
|
|||
;; (defun beancount-get-accounts (&optional string) |
|||
;; "Return list of account names with STRING infix present. |
|||
;; STRING can be multiple words separated by a space." |
|||
;; (let* ((accounts-string (shell-command-to-string |
|||
;; (concat "bean-query -q -f csv " |
|||
;; "/home/marc/Archiv/Finanzen/transactions_ingdiba.beancount " |
|||
;; "'SELECT DISTINCT account ORDER BY ACCOUNT ASC'"))) |
|||
;; (accounts-list (split-string accounts-string))) |
|||
;; accounts-list)) |
|||
|
|||
(defvar beancount-accounts-cache nil |
|||
"List of accounts cached for company mode.") |
|||
|
|||
(defvar beancount-payees-cache nil |
|||
"List of payees cached for company mode.") |
|||
|
|||
(defvar beancount-tags-cache nil |
|||
"List of tags cached for company mode.") |
|||
|
|||
(defvar beancount-commodities-cache nil |
|||
"List of commodities / currencies cached for company mode.") |
|||
|
|||
(defun beancount-update-accounts-and-payees () |
|||
"Initialize or reset the list of accounts." |
|||
(interactive) |
|||
(setq beancount-accounts-cache (beancount-get-accounts-in-buffer)) ;-new |
|||
(pcomplete-uniqify-list (nreverse beancount-accounts-cache)) |
|||
;; (setq beancount-accounts (beancount-get-accounts)) |
|||
(message "Accounts updated.") |
|||
(setq beancount-payees-cache (beancount-get-payees-in-buffer)) ;-new |
|||
(pcomplete-uniqify-list (nreverse beancount-payees-cache)) |
|||
(message "Payees updated.") |
|||
(setq beancount-commodities-cache (beancount-get-commodities-in-buffer)) |
|||
(pcomplete-uniqify-list (nreverse beancount-commodities-cache)) |
|||
(setq beancount-tags-cache (beancount-get-tags-in-buffer)) |
|||
(pcomplete-uniqify-list (nreverse beancount-tags-cache)) |
|||
(message "Tags updated.")) |
|||
;; first test for conditional completions |
|||
;; e.g. only account names when completion is after a date |
|||
;; (defun beancount-company-candidates (prefix) |
|||
;; "Check what PREFIX does." |
|||
;; (let* (accounts (beancount-accounts-cache)) |
|||
;; accounts)) |
|||
|
|||
;;;###autoload |
|||
(defun company-beancount (command &optional arg &rest ignored) |
|||
"Company backend for 'beancount-mode'. |
|||
COMMAND, ARG, and IGNORED the regular meanings." |
|||
(interactive (list 'interactive)) |
|||
(message arg) |
|||
(pcase command |
|||
(`interactive (company-begin-backend 'company-beancount)) |
|||
(`prefix (and (eq major-mode 'beancount-mode) |
|||
(company-grab-symbol))) |
|||
(`candidates |
|||
(cond |
|||
;; if line ends with date and status, at max one quoted text, and an open quote, offer payees and explainations |
|||
((string-match-p beancount-valid-prefix-for-payee-completion-regex (thing-at-point 'line)) |
|||
(delq nil |
|||
(mapcar (lambda (c) |
|||
; (and (string-prefix-p (substring arg 1) c) c) |
|||
(and (string-prefix-p arg c) c)) |
|||
beancount-payees-cache))) |
|||
;; if line starts with date, offer directives |
|||
((string-match-p beancount-valid-prefix-for-directives-regex (thing-at-point 'line)) |
|||
(delq nil |
|||
(mapcar (lambda (c) |
|||
(and (string-prefix-p arg c) c)) |
|||
beancount-timestamped-directive-names))) |
|||
;; if line starts with date, status and payees, offer tags |
|||
((string-match-p beancount-valid-prefix-for-tag-completion-regex (thing-at-point 'line)) |
|||
(delq nil |
|||
(mapcar (lambda (c) |
|||
(and (string-prefix-p (substring arg 1) c) c)) |
|||
beancount-tags-cache))) |
|||
;; if line starts with accounts and amounts, offer commodities |
|||
((string-match-p (concat "^[ ]+[A-Za-z0-9-_:]+[ ]+[-]?[0-9,.]+[ ]+" beancount-currency-regexp) (thing-at-point 'line)) |
|||
(delq nil |
|||
(mapcar (lambda (c) |
|||
(and (string-prefix-p arg c) c)) |
|||
beancount-commodities-cache))) |
|||
;; if line is empty, offer accounts |
|||
((or (string-match-p "^\\s-+" (thing-at-point 'line)) |
|||
;; if the preceding text is allowed before an account, offer accounts |
|||
;; TODO: Not yet working! |
|||
(string-match-p beancount-timestamped-accounts-regexp (thing-at-point 'line))) |
|||
(delq nil |
|||
(mapcar (lambda (c) |
|||
(message c) |
|||
(and (string-prefix-p arg c) c)) |
|||
beancount-accounts-cache))) |
|||
)))) |
|||
|
|||
;;;###autoload |
|||
(define-derived-mode beancount-mode text-mode "Beancount" |
|||
"A mode for editing beancount files. |
|||
\\{beancount-mode-map}" |
|||
:init-value nil |
|||
:lighter " Beancount" |
|||
:group 'beancount |
|||
|
|||
;; customize font-lock for beancount. |
|||
(set-syntax-table beancount-mode-syntax-table) |
|||
(font-lock-add-keywords nil beancount-font-lock-directives) |
|||
|
|||
(or beancount-accounts-cache |
|||
(setq beancount-accounts-cache (beancount-get-accounts-in-buffer))) ;-new |
|||
(or beancount-payees-cache |
|||
(setq beancount-payees-cache (beancount-get-payees-in-buffer))) |
|||
(or beancount-commodities-cache |
|||
(setq beancount-commodities-cache (beancount-get-commodities-in-buffer))) |
|||
(or beancount-tags-cache |
|||
(setq beancount-tags-cache (beancount-get-tags-in-buffer))) |
|||
) |
|||
|
|||
|
|||
(provide 'beancount) |
|||
|
|||
;;; beancount-mode.el ends here |