doomemacs/modules/feature/evil/config.el

387 lines
14 KiB
EmacsLisp
Raw Normal View History

;;; feature/evil/config.el -*- lexical-binding: t; -*-
2017-01-31 19:50:02 -05:00
;; I'm a vimmer at heart. Its modal philosophy suits me better, and this module
;; strives to make Emacs a much better vim than vim was.
(def-setting! :evil-state (modes state)
2017-02-04 21:09:33 -05:00
"Set the initialize STATE of MODE using `evil-set-initial-state'."
(let ((unquoted-modes (doom-unquote modes)))
(if (listp unquoted-modes)
`(progn
,@(cl-loop for mode in unquoted-modes
collect `(evil-set-initial-state ',mode ,state)))
`(evil-set-initial-state ,modes ,state))))
2017-02-04 21:09:33 -05:00
2017-01-31 19:50:02 -05:00
;;
;; evil-mode
;;
(autoload 'goto-last-change "goto-chg")
(autoload 'goto-last-change-reverse "goto-chg")
2017-06-14 21:03:20 +02:00
(def-package! evil
:demand t
2017-01-31 19:50:02 -05:00
:init
(setq evil-want-C-u-scroll t
evil-want-visual-char-semi-exclusive t
evil-want-Y-yank-to-eol t
evil-magic t
evil-echo-state t
evil-indent-convert-tabs t
2017-01-31 19:50:02 -05:00
evil-ex-search-vim-style-regexp t
evil-ex-substitute-global t
evil-ex-visual-char-range t ; column range for ex commands
2017-02-21 00:47:34 -05:00
evil-insert-skip-empty-lines t
2017-05-15 13:52:22 +02:00
evil-mode-line-format 'nil
;; more vim-like behavior
evil-symbol-word-search t
2017-02-21 00:47:34 -05:00
;; don't activate mark on shift-click
shift-select-mode nil)
2017-01-31 19:50:02 -05:00
:config
(add-hook 'doom-init-hook #'evil-mode)
2017-02-19 18:14:46 -05:00
(evil-select-search-module 'evil-search-module 'evil-search)
(set! :popup
2017-02-21 16:04:35 -05:00
'("*evil-registers*" :size 0.3)
'("*Command Line*" :size 8))
;; Don't interfere with localleader key
(define-key evil-motion-state-map "\\" nil)
2017-05-15 13:52:22 +02:00
;; Set cursor colors later, once theme is loaded
2017-04-22 01:49:15 -04:00
(defun +evil*init-cursors (&rest _)
2017-05-10 05:28:50 +02:00
(setq evil-default-cursor (face-background 'cursor nil t)
2017-02-19 18:14:46 -05:00
evil-normal-state-cursor 'box
2017-05-10 05:28:50 +02:00
evil-emacs-state-cursor `(,(face-foreground 'warning) box)
2017-02-19 18:14:46 -05:00
evil-insert-state-cursor 'bar
evil-visual-state-cursor 'hollow))
2017-04-22 01:49:15 -04:00
(advice-add #'load-theme :after #'+evil*init-cursors)
2017-02-19 18:14:46 -05:00
2017-05-15 13:52:22 +02:00
;; default modes
(dolist (mode '(tabulated-list-mode view-mode comint-mode term-mode calendar-mode Man-mode grep-mode))
(evil-set-initial-state mode 'emacs))
2017-05-15 13:52:22 +02:00
(dolist (mode '(help-mode debugger-mode))
(evil-set-initial-state mode 'normal))
;; make `try-expand-dabbrev' from `hippie-expand' work in minibuffer
;; @see `he-dabbrev-beg', so we need re-define syntax for '/'
(defun minibuffer-inactive-mode-hook-setup ()
(set-syntax-table (let* ((table (make-syntax-table)))
(modify-syntax-entry ?/ "." table)
table)))
2017-05-15 13:52:22 +02:00
(add-hook 'minibuffer-inactive-mode-hook #'minibuffer-inactive-mode-hook-setup)
2017-01-31 19:50:02 -05:00
2017-05-15 13:52:22 +02:00
;; --- keybind fixes ----------------------
(map! (:after wgrep
;; a wrapper that invokes `wgrep-mark-deletion' across lines
;; you use `evil-delete' on.
:map wgrep-mode-map [remap evil-delete] #'+evil-delete)
;; replace native folding commands
:n "zo" #'+evil:fold-open
:n "zO" #'+evil:fold-open
:n "zc" #'+evil:fold-close
:n "za" #'+evil:fold-toggle
:n "zr" #'+evil:fold-open-all
:n "zm" #'+evil:fold-close-all)
2017-01-31 19:50:02 -05:00
2017-05-15 13:52:22 +02:00
;; --- evil hacks -------------------------
2017-05-25 12:22:05 +02:00
(defvar +evil-esc-hook '(t)
2017-05-15 13:52:22 +02:00
"A hook run after ESC is pressed in normal mode (invoked by
2017-05-25 12:22:05 +02:00
`evil-force-normal-state'). If a hook returns non-nil, all hooks after it are
ignored.")
2017-05-15 13:52:22 +02:00
(defun +evil*attach-escape-hook ()
"Run the `+evil-esc-hook'."
2017-05-25 12:22:05 +02:00
(cond ((minibuffer-window-active-p (minibuffer-window))
;; quit the minibuffer if open.
(abort-recursive-edit))
((evil-ex-hl-active-p 'evil-ex-search)
;; disable ex search buffer highlights.
(evil-ex-nohighlight))
(t
;; Run all escape hooks. If any returns non-nil, then stop there.
(run-hook-with-args-until-success '+evil-esc-hook))))
2017-05-15 13:52:22 +02:00
(advice-add #'evil-force-normal-state :after #'+evil*attach-escape-hook)
2017-05-15 13:52:22 +02:00
(defun +evil*restore-normal-state-on-windmove (orig-fn &rest args)
"If in anything but normal or motion mode when moving to another window,
restore normal mode. This prevents insert state from bleeding into other modes
across windows."
2017-05-15 13:52:22 +02:00
(unless (memq evil-state '(normal motion))
(evil-normal-state +1))
(apply orig-fn args))
(advice-add #'windmove-do-window-select :around #'+evil*restore-normal-state-on-windmove)
(defun +evil*static-reindent (orig-fn &rest args)
"Don't move cursor on indent."
(save-excursion (apply orig-fn args)))
(advice-add #'evil-indent :around #'+evil*static-reindent)
;; monkey patch `evil-ex-replace-special-filenames' to add more ex
;; substitution flags to evil-mode
(advice-add #'evil-ex-replace-special-filenames
:override #'+evil*ex-replace-special-filenames)
;; These arg types will highlight matches in the current buffer
(evil-ex-define-argument-type buffer-match :runner +evil-ex-buffer-match)
(evil-ex-define-argument-type global-match :runner +evil-ex-global-match)
;; By default :g[lobal] doesn't highlight matches in the current buffer. I've
;; got to write my own argument type and interactive code to get it to do so.
(evil-ex-define-argument-type global-delim-match :runner +evil-ex-global-delim-match)
(dolist (sym '(evil-ex-global evil-ex-global-inverted))
(evil-set-command-property sym :ex-arg 'global-delim-match))
2017-05-15 13:52:22 +02:00
;; Other commands can make use of this
2017-05-15 13:52:22 +02:00
(evil-define-interactive-code "<//>"
:ex-arg buffer-match (list (when (evil-ex-p) evil-ex-argument)))
(evil-define-interactive-code "<//g>"
:ex-arg global-match (list (when (evil-ex-p) evil-ex-argument)))
(evil-set-command-properties
'+evil:align :move-point t :ex-arg 'buffer-match :ex-bang t :evil-mc t :keep-visual t :suppress-operator t)
(evil-set-command-properties
'+evil:mc :move-point nil :ex-arg 'global-match :ex-bang t :evil-mc t)
;; Move to new split -- setting `evil-split-window-below' &
;; `evil-vsplit-window-right' to non-nil mimics this, but that doesn't update
;; window history. That means when you delete a new split, Emacs leaves you on
;; the 2nd to last window on the history stack, which is jarring.
(defun +evil*window-follow (&rest _) (evil-window-down 1))
(defun +evil*window-vfollow (&rest _) (evil-window-right 1))
(advice-add #'evil-window-split :after #'+evil*window-follow)
(advice-add #'evil-window-vsplit :after #'+evil*window-vfollow))
2017-01-31 19:50:02 -05:00
;;
;; Plugins
;;
(def-package! evil-commentary
2017-01-31 19:50:02 -05:00
:commands (evil-commentary evil-commentary-yank evil-commentary-line)
2017-02-02 04:35:54 -05:00
:config (evil-commentary-mode 1))
2017-01-31 19:50:02 -05:00
(def-package! evil-easymotion
:after evil-snipe
2017-01-31 19:50:02 -05:00
:config
(defvar +evil--snipe-repeat-fn
(evilem-create #'evil-snipe-repeat
:bind ((evil-snipe-scope 'whole-buffer)
(evil-snipe-enable-highlight)
(evil-snipe-enable-incremental-highlight)))))
2017-01-31 19:50:02 -05:00
(def-package! evil-embrace
2017-01-31 19:50:02 -05:00
:after evil-surround
:config
(setq evil-embrace-show-help-p nil)
(evil-embrace-enable-evil-surround-integration)
(defun +evil--embrace-get-pair (char)
2017-02-19 18:14:46 -05:00
(if-let (pair (cdr-safe (assoc (string-to-char char) evil-surround-pairs-alist)))
pair
(if-let (pair (assoc-default char embrace--pairs-list))
(if-let (real-pair (and (functionp (embrace-pair-struct-read-function pair))
(funcall (embrace-pair-struct-read-function pair))))
real-pair
(cons (embrace-pair-struct-left pair) (embrace-pair-struct-right pair)))
(cons char char))))
2017-01-31 19:50:02 -05:00
(defun +evil--embrace-escaped ()
"Backslash-escaped surround character support for embrace."
(let ((char (read-char "\\")))
(if (eq char 27)
(cons "" "")
(let ((pair (+evil--embrace-get-pair (string char)))
(text (if (sp-point-in-string) "\\\\%s" "\\%s")))
(cons (format text (car pair))
(format text (cdr pair)))))))
(defun +evil--embrace-latex ()
"LaTeX command support for embrace."
(cons (format "\\%s{" (read-string "\\")) "}"))
(defun +evil--embrace-elisp-fn ()
"Elisp function support for embrace."
(cons (format "(%s " (or (read-string "(") "")) ")"))
;; Add escaped-sequence support to embrace
(push (cons ?\\ (make-embrace-pair-struct
:key ?\\
:read-function #'+evil--embrace-escaped
2017-01-31 19:50:02 -05:00
:left-regexp "\\[[{(]"
:right-regexp "\\[]})]"))
(default-value 'embrace--pairs-list))
;; Add extra pairs
(add-hook 'LaTeX-mode-hook #'embrace-LaTeX-mode-hook)
(add-hook 'org-mode-hook #'embrace-org-mode-hook)
(add-hook! emacs-lisp-mode
2017-01-31 19:50:02 -05:00
(embrace-add-pair ?\` "`" "'"))
(add-hook! (emacs-lisp-mode lisp-mode)
(embrace-add-pair-regexp ?f "([^ ]+ " ")" #'+evil--embrace-elisp-fn))
(add-hook! (org-mode LaTeX-mode)
(embrace-add-pair-regexp ?l "\\[a-z]+{" "}" #'+evil--embrace-latex)))
2017-01-31 19:50:02 -05:00
(def-package! evil-escape
:commands evil-escape-mode
2017-01-31 19:50:02 -05:00
:init
(setq evil-escape-excluded-states '(normal visual multiedit emacs motion)
2017-05-16 19:21:33 +02:00
evil-escape-excluded-major-modes '(neotree-mode)
evil-escape-key-sequence "jk"
evil-escape-delay 0.25)
(add-hook 'doom-post-init-hook #'evil-escape-mode)
2017-01-31 19:50:02 -05:00
:config
;; no `evil-escape' in minibuffer
2017-06-11 00:59:02 +02:00
(cl-pushnew #'minibufferp evil-escape-inhibit-functions)
(map! :irvo "C-g" #'evil-escape))
2017-01-31 19:50:02 -05:00
(def-package! evil-exchange
2017-01-31 19:50:02 -05:00
:commands evil-exchange
:config
(defun +evil|escape-exchange ()
2017-05-25 12:22:05 +02:00
(when evil-exchange--overlays
(evil-exchange-cancel)
t))
(add-hook '+evil-esc-hook #'+evil|escape-exchange))
2017-01-31 19:50:02 -05:00
(def-package! evil-matchit
2017-01-31 19:50:02 -05:00
:commands (evilmi-jump-items evilmi-text-object global-evil-matchit-mode)
:config (global-evil-matchit-mode 1)
:init
2017-06-04 02:21:52 +02:00
(map! [remap evil-jump-item] #'evilmi-jump-items
:textobj "%" #'evilmi-text-object #'evilmi-text-object)
:config
(defun +evil|simple-matchit ()
"A hook to force evil-matchit to favor simple bracket jumping. Helpful when
the new algorithm is confusing, like in python or ruby."
(setq-local evilmi-always-simple-jump t))
(add-hook 'python-mode-hook #'+evil|simple-matchit))
2017-01-31 19:50:02 -05:00
(def-package! evil-multiedit
:commands (evil-multiedit-match-all
evil-multiedit-match-and-next
evil-multiedit-match-and-prev
evil-multiedit-match-symbol-and-next
evil-multiedit-match-symbol-and-prev
evil-multiedit-toggle-or-restrict-region
evil-multiedit-next
evil-multiedit-prev
evil-multiedit-abort
evil-multiedit-ex-match))
(def-package! evil-mc
2017-05-26 23:08:53 +02:00
:commands (evil-mc-make-cursor-here evil-mc-make-all-cursors
evil-mc-undo-all-cursors evil-mc-pause-cursors
evil-mc-resume-cursors evil-mc-make-and-goto-first-cursor
evil-mc-make-and-goto-last-cursor evil-mc-make-cursor-here
evil-mc-make-cursor-move-next-line
evil-mc-make-cursor-move-prev-line
evil-mc-make-cursor-at-pos
evil-mc-make-and-goto-next-cursor evil-mc-skip-and-goto-next-cursor
evil-mc-make-and-goto-prev-cursor evil-mc-skip-and-goto-prev-cursor
evil-mc-make-and-goto-next-match evil-mc-skip-and-goto-next-match
evil-mc-skip-and-goto-next-match evil-mc-make-and-goto-prev-match
evil-mc-skip-and-goto-prev-match)
:init
(defvar evil-mc-key-map (make-sparse-keymap))
:config
2017-05-26 23:08:53 +02:00
(global-evil-mc-mode +1)
;; Add custom commands to whitelisted commands
(dolist (fn '(doom/deflate-space-maybe doom/inflate-space-maybe
doom/backward-to-bol-or-indent doom/forward-to-last-non-comment-or-eol
doom/backward-kill-to-bol-and-indent doom/newline-and-indent))
2017-05-26 23:08:53 +02:00
(push (cons fn '((:default . evil-mc-execute-default-call))) evil-mc-custom-known-commands))
(defun +evil|escape-multiple-cursors ()
2017-05-26 23:08:53 +02:00
"Clear evil-mc cursors and restore state."
(when (evil-mc-has-cursors-p)
2017-05-25 12:22:05 +02:00
(evil-mc-undo-all-cursors)
2017-05-26 23:08:53 +02:00
(evil-mc-resume-cursors)
2017-05-25 12:22:05 +02:00
t))
(add-hook '+evil-esc-hook #'+evil|escape-multiple-cursors)
2017-05-26 23:08:53 +02:00
;; disable evil-escape in evil-mc; causes unwanted text on invocation
(push 'evil-escape-mode evil-mc-incompatible-minor-modes))
2017-01-31 19:50:02 -05:00
(def-package! evil-snipe :demand t
2017-01-31 19:50:02 -05:00
:init
(setq evil-snipe-smart-case t
evil-snipe-scope 'line
evil-snipe-repeat-scope 'visible
2017-02-19 18:14:46 -05:00
evil-snipe-override-evil-repeat-keys nil
2017-01-31 19:50:02 -05:00
evil-snipe-char-fold t
evil-snipe-aliases '((?\[ "[[{(]")
(?\] "[]})]")
(?\; "[;:]")))
:config
(add-hook 'doom-post-init-hook #'evil-snipe-override-mode))
2017-01-31 19:50:02 -05:00
(def-package! evil-surround
2017-01-31 19:50:02 -05:00
:commands (global-evil-surround-mode
evil-surround-edit
evil-Surround-edit
evil-surround-region)
:config (global-evil-surround-mode 1))
(def-package! evil-vimish-fold
:commands evil-vimish-fold-mode
2017-05-15 20:20:06 +02:00
:init
(setq vimish-fold-dir (concat doom-cache-dir "vimish-fold/")
vimish-fold-indication-mode 'right-fringe)
(add-hook 'doom-post-init-hook #'evil-vimish-fold-mode t))
2017-05-15 20:20:06 +02:00
;; Without `evil-visualstar', * and # grab the word at point and search, no
;; matter what mode you're in. I want to be able to visually select a region and
;; search for other occurrences of it.
(def-package! evil-visualstar
2017-01-31 19:50:02 -05:00
:commands (global-evil-visualstar-mode
evil-visualstar/begin-search
evil-visualstar/begin-search-forward
evil-visualstar/begin-search-backward)
:init
(map! :v "*" #'evil-visualstar/begin-search-forward
:v "#" #'evil-visualstar/begin-search-backward)
:config
(global-evil-visualstar-mode 1))
2017-01-31 19:50:02 -05:00
2017-06-05 00:47:56 +02:00
;;
;; Text object plugins
;;
(def-package! evil-args
:commands (evil-inner-arg evil-outer-arg
evil-forward-arg evil-backward-arg
evil-jump-out-args))
(def-package! evil-indent-plus
:commands (evil-indent-plus-i-indent
evil-indent-plus-a-indent
evil-indent-plus-i-indent-up
evil-indent-plus-a-indent-up
evil-indent-plus-i-indent-up-down
evil-indent-plus-a-indent-up-down))
(def-package! evil-textobj-anyblock
:commands (evil-textobj-anyblock-inner-block evil-textobj-anyblock-a-block))