BREAKING CHANGE: This deprecates the IS-(MAC|WINDOWS|LINUX|BSD) family of global constants in favor of a native `featurep` check: IS-MAC -> (featurep :system 'macos) IS-WINDOWS -> (featurep :system 'windows) IS-LINUX -> (featurep :system 'linux) IS-BSD -> (featurep :system 'bsd) The constants will stick around until the v3 release so folks can still use it -- and there are still some modules that use it, but I'll phase those uses out gradually. Fix: #7479
248 lines
9.7 KiB
EmacsLisp
248 lines
9.7 KiB
EmacsLisp
;;; term/eshell/config.el -*- lexical-binding: t; -*-
|
|
|
|
;; see:
|
|
;; + `+eshell/here': open eshell in the current window
|
|
;; + `+eshell/toggle': toggles an eshell popup
|
|
;; + `+eshell/frame': converts the current frame into an eshell-dedicated
|
|
;; frame. Once the last eshell process is killed, the old frame configuration
|
|
;; is restored.
|
|
|
|
(defvar +eshell-config-dir
|
|
(expand-file-name "eshell/" doom-user-dir)
|
|
"Where to store eshell configuration files, as opposed to
|
|
`eshell-directory-name', which is where Doom will store temporary/data files.")
|
|
|
|
(defvar eshell-directory-name (concat doom-data-dir "eshell")
|
|
"Where to store temporary/data files, as opposed to `eshell-config-dir',
|
|
which is where Doom will store eshell configuration files.")
|
|
|
|
(defvar +eshell-enable-new-shell-on-split t
|
|
"If non-nil, spawn a new eshell session after splitting from an eshell
|
|
buffer.")
|
|
|
|
(defvar +eshell-kill-window-on-exit nil
|
|
"If non-nil, eshell will close windows along with its eshell buffers.")
|
|
|
|
(defvar +eshell-aliases
|
|
'(("q" "exit") ; built-in
|
|
("f" "find-file $1")
|
|
("ff" "find-file-other-window $1")
|
|
("d" "dired $1")
|
|
("bd" "eshell-up $1")
|
|
("rg" "rg --color=always $*")
|
|
("l" "ls -lh $*")
|
|
("ll" "ls -lah $*")
|
|
("git" "git --no-pager $*")
|
|
("gg" "magit-status")
|
|
("cdp" "cd-to-project")
|
|
("clear" "clear-scrollback")) ; more sensible than default
|
|
"An alist of default eshell aliases, meant to emulate useful shell utilities,
|
|
like fasd and bd. Note that you may overwrite these in your
|
|
`eshell-aliases-file'. This is here to provide an alternative, elisp-centric way
|
|
to define your aliases.
|
|
|
|
You should use `set-eshell-alias!' to change this.")
|
|
|
|
;; These files are exceptions, because they may contain configuration
|
|
(defvar eshell-aliases-file (concat +eshell-config-dir "aliases"))
|
|
(defvar eshell-rc-script (concat +eshell-config-dir "profile"))
|
|
(defvar eshell-login-script (concat +eshell-config-dir "login"))
|
|
|
|
(defvar +eshell--default-aliases nil)
|
|
|
|
|
|
;;
|
|
;;; Packages
|
|
|
|
(after! eshell ; built-in
|
|
(setq eshell-banner-message
|
|
'(format "%s %s\n"
|
|
(propertize (format " %s " (string-trim (buffer-name)))
|
|
'face 'mode-line-highlight)
|
|
(propertize (current-time-string)
|
|
'face 'font-lock-keyword-face))
|
|
eshell-scroll-to-bottom-on-input 'all
|
|
eshell-scroll-to-bottom-on-output 'all
|
|
eshell-kill-processes-on-exit t
|
|
eshell-hist-ignoredups t
|
|
;; don't record command in history if prefixed with whitespace
|
|
;; TODO Use `eshell-input-filter-initial-space' when Emacs 25 support is dropped
|
|
eshell-input-filter (lambda (input) (not (string-match-p "\\`\\s-+" input)))
|
|
;; em-prompt
|
|
eshell-prompt-regexp "^[^#$\n]* [#$λ] "
|
|
eshell-prompt-function #'+eshell-default-prompt-fn
|
|
;; em-glob
|
|
eshell-glob-case-insensitive t
|
|
eshell-error-if-no-glob t)
|
|
|
|
;; Consider eshell buffers real
|
|
(add-hook 'eshell-mode-hook #'doom-mark-buffer-as-real-h)
|
|
|
|
;; Keep track of open eshell buffers
|
|
(add-hook 'eshell-mode-hook #'+eshell-init-h)
|
|
(add-hook 'eshell-exit-hook #'+eshell-cleanup-h)
|
|
|
|
;; Enable autopairing in eshell
|
|
(add-hook 'eshell-mode-hook #'smartparens-mode)
|
|
|
|
;; Persp-mode/workspaces integration
|
|
(when (modulep! :ui workspaces)
|
|
(add-hook 'persp-activated-functions #'+eshell-switch-workspace-fn)
|
|
(add-hook 'persp-before-switch-functions #'+eshell-save-workspace-fn))
|
|
|
|
;; UI enhancements
|
|
(add-hook! 'eshell-mode-hook
|
|
(defun +eshell-remove-fringes-h ()
|
|
(set-window-fringes nil 0 0)
|
|
(set-window-margins nil 1 nil))
|
|
(defun +eshell-enable-text-wrapping-h ()
|
|
(visual-line-mode +1)
|
|
(set-display-table-slot standard-display-table 0 ?\ )))
|
|
|
|
(add-hook 'eshell-mode-hook #'hide-mode-line-mode)
|
|
|
|
;; Remove hscroll-margin in shells, otherwise you get jumpiness when the
|
|
;; cursor comes close to the left/right edges of the window.
|
|
(setq-hook! 'eshell-mode-hook hscroll-margin 0)
|
|
|
|
;; Recognize prompts as Imenu entries.
|
|
(setq-hook! 'eshell-mode-hook
|
|
imenu-generic-expression
|
|
`((,(propertize "λ" 'face 'eshell-prompt)
|
|
,(concat eshell-prompt-regexp "\\(.*\\)") 1)))
|
|
|
|
;; Don't auto-write our aliases! Let us manage our own `eshell-aliases-file'
|
|
;; or configure `+eshell-aliases' via elisp.
|
|
(advice-add #'eshell-write-aliases-list :override #'ignore)
|
|
|
|
(add-to-list 'eshell-modules-list 'eshell-tramp)
|
|
|
|
;; Visual commands require a proper terminal. Eshell can't handle that, so
|
|
;; it delegates these commands to a term buffer.
|
|
(after! em-term
|
|
(pushnew! eshell-visual-commands "tmux" "htop" "vim" "nvim" "ncmpcpp"))
|
|
|
|
(after! em-alias
|
|
(setq +eshell--default-aliases eshell-command-aliases-list
|
|
eshell-command-aliases-list
|
|
(append eshell-command-aliases-list
|
|
+eshell-aliases)))
|
|
|
|
;; HACK: Fixes #3817, where eshell completion after quotes is broken on Emacs
|
|
;; 28 and older.
|
|
;; CREDIT: Extracted from `cape''s cape-wrap-silent and cape-wrap-purify.
|
|
;; REVIEW: Remove when Doom drops 28 support.
|
|
(when (< emacs-major-version 29)
|
|
(defadvice! +eshell--silent-a (capf)
|
|
"Call CAPF and silence it (no messages, no errors).
|
|
This function can be used as an advice around an existing Capf."
|
|
:around #'pcomplete-completions-at-point
|
|
(letf! ((defmacro silent (&rest body) `(quiet! (ignore-errors ,@body)))
|
|
(defmacro wrapped-table (wrap body)
|
|
`(lambda (str pred action)
|
|
(,@body
|
|
(let ((result (complete-with-action action table str pred)))
|
|
(when
|
|
(and (eq action 'completion--unquote)
|
|
(functionp (cadr result)))
|
|
(cl-callf ,wrap (cadr result)))
|
|
result))))
|
|
(defun* silent-table (table) (wrapped-table silent-table (silent))))
|
|
(pcase (silent (funcall capf))
|
|
(`(,beg ,end ,table . ,plist)
|
|
`(,beg ,end ,(silent-table table) ,@plist)))))
|
|
|
|
(defadvice! +eshell--purify-a (capf)
|
|
"Call CAPF and ensure that it does not illegally modify the buffer. This
|
|
function can be used as an advice around an existing Capf. It has been
|
|
introduced mainly to fix the broken `pcomplete-completions-at-point' function in
|
|
Emacs versions < 29."
|
|
;; bug#50470: Fix Capfs which illegally modify the buffer or which
|
|
;; illegally call `completion-in-region'. The workaround here was proposed
|
|
;; by @jakanakaevangeli and is used in his capf-autosuggest package.
|
|
:around #'pcomplete-completions-at-point
|
|
(catch 'illegal-completion-in-region
|
|
(condition-case nil
|
|
(let ((buffer-read-only t)
|
|
(inhibit-read-only nil)
|
|
(completion-in-region-function
|
|
(lambda (beg end coll pred)
|
|
(throw 'illegal-completion-in-region
|
|
(list beg end coll :predicate pred)))))
|
|
(funcall capf))
|
|
(buffer-read-only nil))))))
|
|
|
|
|
|
(after! esh-mode
|
|
(map! :map eshell-mode-map
|
|
:n "RET" #'+eshell/goto-end-of-prompt
|
|
:n [return] #'+eshell/goto-end-of-prompt
|
|
:ni "C-j" #'eshell-next-matching-input-from-input
|
|
:ni "C-k" #'eshell-previous-matching-input-from-input
|
|
:ig "C-d" #'+eshell/quit-or-delete-char
|
|
:i "C-c h" #'evil-window-left
|
|
:i "C-c j" #'evil-window-down
|
|
:i "C-c k" #'evil-window-up
|
|
:i "C-c l" #'evil-window-right
|
|
"C-s" #'+eshell/search-history
|
|
;; Emacs bindings
|
|
"C-e" #'end-of-line
|
|
;; Tmux-esque prefix keybinds
|
|
"C-c s" #'+eshell/split-below
|
|
"C-c v" #'+eshell/split-right
|
|
"C-c x" #'+eshell/kill-and-close
|
|
[remap split-window-below] #'+eshell/split-below
|
|
[remap split-window-right] #'+eshell/split-right
|
|
[remap doom/backward-to-bol-or-indent] #'eshell-bol
|
|
[remap doom/backward-kill-to-bol-and-indent] #'eshell-kill-input
|
|
[remap evil-delete-back-to-indentation] #'eshell-kill-input
|
|
[remap evil-window-split] #'+eshell/split-below
|
|
[remap evil-window-vsplit] #'+eshell/split-right
|
|
;; To emulate terminal keybinds
|
|
"C-l" (cmd! (eshell/clear-scrollback) (eshell-emit-prompt))
|
|
(:localleader
|
|
"b" #'eshell-insert-buffer-name
|
|
"e" #'eshell-insert-envvar
|
|
"s" #'+eshell/search-history)))
|
|
|
|
|
|
(use-package! eshell-up
|
|
:commands eshell-up eshell-up-peek)
|
|
|
|
|
|
(use-package! eshell-z
|
|
:after eshell
|
|
:config
|
|
;; Use zsh's db if it exists, otherwise, store it in `doom-cache-dir'
|
|
(unless (file-exists-p eshell-z-freq-dir-hash-table-file-name)
|
|
(setq eshell-z-freq-dir-hash-table-file-name
|
|
(expand-file-name "z" eshell-directory-name))))
|
|
|
|
|
|
(use-package! esh-help
|
|
:after eshell
|
|
:config (setup-esh-help-eldoc))
|
|
|
|
|
|
(use-package! eshell-did-you-mean
|
|
:after esh-mode ; Specifically esh-mode, not eshell
|
|
:config
|
|
(eshell-did-you-mean-setup)
|
|
;; HACK There is a known issue with `eshell-did-you-mean' where it does not
|
|
;; work on first invocation, so we invoke it once manually by setting the
|
|
;; last command and then calling the output filter.
|
|
(setq eshell-last-command-name "catt")
|
|
(eshell-did-you-mean-output-filter "catt: command not found"))
|
|
|
|
|
|
(use-package eshell-syntax-highlighting
|
|
:hook (eshell-mode . eshell-syntax-highlighting-mode)
|
|
:init
|
|
(add-hook 'eshell-syntax-highlighting-elisp-buffer-setup-hook #'highlight-quoted-mode))
|
|
|
|
|
|
(use-package! fish-completion
|
|
:unless (featurep :system 'windows)
|
|
:hook (eshell-mode . fish-completion-mode)
|
|
:init (setq fish-completion-fallback-on-bash-p t
|
|
fish-completion-inhibit-missing-fish-command-warning t))
|