2015-11-19 05:24:27 +00:00
|
|
|
;;; evil-evilified-state.el --- A minimalistic evil state
|
2015-08-12 05:31:44 +00:00
|
|
|
|
|
|
|
;; Copyright (C) 2014, 2015 syl20bnr
|
2015-06-26 03:49:53 +00:00
|
|
|
;;
|
|
|
|
;; Author: Sylvain Benner <sylvain.benner@gmail.com>
|
2015-08-12 05:31:44 +00:00
|
|
|
;; Keywords: convenience editing evil spacemacs
|
|
|
|
;; Created: 22 Mar 2015
|
|
|
|
;; Version: 1.0
|
2015-11-05 13:09:00 +00:00
|
|
|
;; Package-Requires: ((evil "1.0.9"))
|
2015-08-12 05:31:44 +00:00
|
|
|
|
2015-06-26 03:49:53 +00:00
|
|
|
;; This file is not part of GNU Emacs.
|
2015-08-12 05:31:44 +00:00
|
|
|
|
|
|
|
;; This program is free software; you can redistribute it and/or modify
|
|
|
|
;; it under the terms of the GNU General Public License as published by
|
|
|
|
;; the Free Software Foundation, either version 3 of the License, or
|
|
|
|
;; (at your option) any later version.
|
|
|
|
|
|
|
|
;; This program is distributed in the hope that it will be useful,
|
|
|
|
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
;; GNU General Public License for more details.
|
|
|
|
|
|
|
|
;; You should have received a copy of the GNU General Public License
|
|
|
|
;; along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
;;; Commentary:
|
|
|
|
|
|
|
|
;; Define a `evilified' evil state inheriting from `emacs' state and
|
|
|
|
;; setting a minimalist list of Vim key bindings (like navigation, search, ...)
|
|
|
|
|
2015-11-19 05:24:27 +00:00
|
|
|
;; The shadowed original mode key bindings are automatically reassigned
|
|
|
|
;; following a set of rules:
|
|
|
|
|
|
|
|
;;
|
|
|
|
|
|
|
|
|
2015-08-12 05:31:44 +00:00
|
|
|
;;; Code:
|
|
|
|
|
|
|
|
(require 'evil)
|
|
|
|
|
2015-11-05 13:09:00 +00:00
|
|
|
(defvar evilified-state--modes nil
|
2015-08-12 05:31:44 +00:00
|
|
|
"List of all evilified modes.")
|
|
|
|
|
2015-11-05 13:09:00 +00:00
|
|
|
(defvar evilified-state--visual-state-map evil-visual-state-map
|
2015-08-12 05:31:44 +00:00
|
|
|
"Evil visual state map backup.")
|
|
|
|
|
2015-11-05 13:09:00 +00:00
|
|
|
(defvar evilified-state--evil-surround nil
|
2015-08-12 05:31:44 +00:00
|
|
|
"Evil surround mode variable backup.")
|
2015-11-05 13:09:00 +00:00
|
|
|
(make-variable-buffer-local 'evilified-state--evil-surround)
|
2015-08-12 05:31:44 +00:00
|
|
|
|
|
|
|
(evil-define-state evilified
|
|
|
|
"Evilified state.
|
|
|
|
Hybrid `emacs state' with carrefully selected Vim key bindings.
|
|
|
|
See spacemacs conventions for more info."
|
|
|
|
:tag " <Ev> "
|
|
|
|
:enable (emacs)
|
|
|
|
:message "-- EVILIFIED BUFFER --"
|
|
|
|
:cursor box)
|
|
|
|
|
2015-11-05 13:09:00 +00:00
|
|
|
(add-hook 'evil-evilified-state-entry-hook 'evilified-state--evilified-state-on-entry)
|
|
|
|
(add-hook 'evil-evilified-state-exit-hook 'evilified-state--evilified-state-on-exit)
|
2015-09-26 01:29:48 +00:00
|
|
|
|
2015-11-05 13:09:00 +00:00
|
|
|
(add-hook 'evil-visual-state-entry-hook 'evilified-state--visual-state-on-entry)
|
|
|
|
(add-hook 'evil-visual-state-exit-hook 'evilified-state--visual-state-on-exit)
|
2015-09-26 01:46:01 +00:00
|
|
|
|
2015-11-05 13:09:00 +00:00
|
|
|
(defun evilified-state--pre-command-hook ()
|
2015-09-26 01:29:48 +00:00
|
|
|
(let ((map (get-char-property (point) 'keymap)))
|
2015-11-30 16:28:28 +00:00
|
|
|
(when (and (not (window-minibuffer-active-p))
|
|
|
|
map (assq 'evilified-state map))
|
2015-09-26 01:29:48 +00:00
|
|
|
(let* ((submap (cdr (assq 'evilified-state map)))
|
|
|
|
(command (when (and submap (eq 1 (length (this-command-keys))))
|
|
|
|
(lookup-key submap (this-command-keys)))))
|
|
|
|
(when command
|
|
|
|
(setq this-command command))))))
|
2015-08-12 05:31:44 +00:00
|
|
|
|
2015-11-05 13:09:00 +00:00
|
|
|
(defun evilified-state--evilified-state-on-entry ()
|
2015-08-12 05:31:44 +00:00
|
|
|
"Setup evilified state."
|
2015-11-05 13:09:00 +00:00
|
|
|
(add-hook 'pre-command-hook 'evilified-state--pre-command-hook nil 'local)
|
2015-09-10 03:33:37 +00:00
|
|
|
(when (bound-and-true-p evil-surround-mode)
|
|
|
|
(make-local-variable 'evil-surround-mode)
|
2015-08-12 05:31:44 +00:00
|
|
|
(evil-surround-mode -1))
|
2015-09-10 03:33:37 +00:00
|
|
|
(setq-local evil-normal-state-map (cons 'keymap nil))
|
|
|
|
(setq-local evil-visual-state-map (cons 'keymap (list (cons ?y 'evil-yank)))))
|
2015-08-12 05:31:44 +00:00
|
|
|
|
2015-11-05 13:09:00 +00:00
|
|
|
(defun evilified-state--evilified-state-on-exit ()
|
2015-09-26 01:55:37 +00:00
|
|
|
"Clean evilified state"
|
2015-11-05 13:09:00 +00:00
|
|
|
(remove-hook 'pre-command-hook 'evilified-state--pre-command-hook 'local))
|
2015-09-26 01:55:37 +00:00
|
|
|
|
2015-11-05 13:09:00 +00:00
|
|
|
(defun evilified-state--visual-state-on-entry ()
|
2015-09-26 01:55:37 +00:00
|
|
|
"Setup visual state."
|
2015-11-05 13:09:00 +00:00
|
|
|
(add-hook 'pre-command-hook 'evilified-state--pre-command-hook nil 'local))
|
2015-09-26 01:55:37 +00:00
|
|
|
|
2015-11-05 13:09:00 +00:00
|
|
|
(defun evilified-state--visual-state-on-exit ()
|
2015-09-26 01:55:37 +00:00
|
|
|
"Clean visual state"
|
2015-11-05 13:09:00 +00:00
|
|
|
(remove-hook 'pre-command-hook 'evilified-state--pre-command-hook 'local))
|
2015-09-26 01:29:48 +00:00
|
|
|
|
2015-08-12 05:31:44 +00:00
|
|
|
;; default key bindings for all evilified buffers
|
2015-11-18 00:35:32 +00:00
|
|
|
(define-key evil-evilified-state-map (kbd dotspacemacs-leader-key)
|
|
|
|
spacemacs-default-map)
|
2015-08-12 05:31:44 +00:00
|
|
|
(define-key evil-evilified-state-map "/" 'evil-search-forward)
|
|
|
|
(define-key evil-evilified-state-map ":" 'evil-ex)
|
|
|
|
(define-key evil-evilified-state-map "h" 'evil-backward-char)
|
|
|
|
(define-key evil-evilified-state-map "j" 'evil-next-visual-line)
|
|
|
|
(define-key evil-evilified-state-map "k" 'evil-previous-visual-line)
|
|
|
|
(define-key evil-evilified-state-map "l" 'evil-forward-char)
|
|
|
|
(define-key evil-evilified-state-map "n" 'evil-search-next)
|
|
|
|
(define-key evil-evilified-state-map "N" 'evil-search-previous)
|
|
|
|
(define-key evil-evilified-state-map "v" 'evil-visual-char)
|
|
|
|
(define-key evil-evilified-state-map "V" 'evil-visual-line)
|
|
|
|
(define-key evil-evilified-state-map "gg" 'evil-goto-first-line)
|
|
|
|
(define-key evil-evilified-state-map "G" 'evil-goto-line)
|
|
|
|
(define-key evil-evilified-state-map (kbd "C-f") 'evil-scroll-page-down)
|
|
|
|
(define-key evil-evilified-state-map (kbd "C-b") 'evil-scroll-page-up)
|
|
|
|
(define-key evil-evilified-state-map (kbd "C-d") 'evil-scroll-down)
|
|
|
|
(define-key evil-evilified-state-map (kbd "C-u") 'evil-scroll-up)
|
2015-09-23 00:23:01 +00:00
|
|
|
(define-key evil-evilified-state-map (kbd "C-z") 'evil-emacs-state)
|
2015-08-12 05:31:44 +00:00
|
|
|
|
2015-09-26 04:26:16 +00:00
|
|
|
;; old macro
|
2015-11-05 13:09:00 +00:00
|
|
|
;;;###autoload
|
|
|
|
(defmacro evilified-state-evilify (mode map &rest body)
|
2015-09-26 04:26:16 +00:00
|
|
|
"Set `evilified state' as default for MODE.
|
|
|
|
|
|
|
|
BODY is a list of additional key bindings to apply for the given MAP in
|
|
|
|
`evilified state'."
|
|
|
|
(let ((defkey (when body `(evil-define-key 'evilified ,map ,@body))))
|
|
|
|
`(progn (unless ,(null mode)
|
2015-11-05 13:09:00 +00:00
|
|
|
(unless (memq ',mode evilified-state--modes)
|
|
|
|
(push ',mode evilified-state--modes))
|
2015-09-26 04:26:16 +00:00
|
|
|
(unless (or (bound-and-true-p holy-mode)
|
|
|
|
(memq ',mode evil-evilified-state-modes))
|
|
|
|
(delq ',mode evil-emacs-state-modes)
|
|
|
|
(push ',mode evil-evilified-state-modes)))
|
|
|
|
(unless ,(null defkey) (,@defkey)))))
|
2015-11-05 13:09:00 +00:00
|
|
|
(put 'evilified-state-evilify 'lisp-indent-function 'defun)
|
2015-09-26 04:26:16 +00:00
|
|
|
|
|
|
|
;; new macro
|
2015-11-05 13:09:00 +00:00
|
|
|
;;;###autoload
|
|
|
|
(defmacro evilified-state-evilify-map (map &rest props)
|
2015-09-26 04:26:16 +00:00
|
|
|
"Evilify MAP.
|
|
|
|
|
|
|
|
Avaiblabe PROPS:
|
|
|
|
|
|
|
|
`:mode SYMBOL'
|
|
|
|
A mode SYMBOL associated with MAP. Used to add SYMBOL to the list of modes
|
|
|
|
defaulting to `evilified-state'.
|
|
|
|
|
|
|
|
`:evilified-map SYMBOL'
|
|
|
|
A map SYMBOL of an alternate evilified map, if nil then
|
|
|
|
`evil-evilified-state-map' is used.
|
|
|
|
|
|
|
|
`:eval-after-load SYMBOL'
|
|
|
|
If specified the evilification of MAP is deferred to the loading of the feature
|
|
|
|
bound to SYMBOL. May be required for some lazy-loaded maps.
|
|
|
|
|
|
|
|
`:bindings EXPRESSIONS'
|
|
|
|
One or several EXPRESSIONS with the form `KEY FUNCTION':
|
|
|
|
KEY1 FUNCTION1
|
|
|
|
KEY2 FUNCTION2
|
|
|
|
...
|
|
|
|
Each pair KEYn FUNCTIONn is defined in MAP after the evilification of it."
|
|
|
|
(declare (indent 1))
|
|
|
|
(let* ((mode (plist-get props :mode))
|
|
|
|
(evilified-map (plist-get props :evilified-map))
|
|
|
|
(eval-after-load (plist-get props :eval-after-load))
|
2015-11-05 13:09:00 +00:00
|
|
|
(bindings (evilified-state--mplist-get props :bindings))
|
2015-09-26 04:26:16 +00:00
|
|
|
(defkey (when bindings `(evil-define-key 'evilified ,map ,@bindings)))
|
|
|
|
(body
|
|
|
|
`(progn
|
2015-11-05 13:09:00 +00:00
|
|
|
(let ((sorted-map (evilified-state--sort-keymap
|
2015-09-26 04:26:16 +00:00
|
|
|
(or ,evilified-map evil-evilified-state-map)))
|
|
|
|
processed)
|
|
|
|
(mapc (lambda (map-entry)
|
|
|
|
(unless (member (car map-entry) processed)
|
2015-11-05 13:09:00 +00:00
|
|
|
(setq processed (evilified-state--evilify-event
|
2015-09-26 04:26:16 +00:00
|
|
|
,map ',map
|
|
|
|
(or ,evilified-map
|
|
|
|
evil-evilified-state-map)
|
|
|
|
(car map-entry) (cdr map-entry)))))
|
|
|
|
sorted-map))
|
|
|
|
(unless ,(null defkey)
|
|
|
|
(,@defkey))
|
|
|
|
(unless ,(null mode)
|
2015-11-05 13:09:00 +00:00
|
|
|
(evilified-state--configure-default-state ',mode)))))
|
2015-09-26 04:26:16 +00:00
|
|
|
(if (null eval-after-load)
|
|
|
|
`(,@body)
|
2015-10-07 20:33:29 +00:00
|
|
|
`(with-eval-after-load ',eval-after-load ,body))))
|
2015-11-05 13:09:00 +00:00
|
|
|
(put 'evilified-state-evilify-map 'lisp-indent-function 'defun)
|
2015-09-26 04:26:16 +00:00
|
|
|
|
2015-11-05 13:09:00 +00:00
|
|
|
(defun evilified-state--configure-default-state (mode)
|
2015-09-26 04:26:16 +00:00
|
|
|
"Configure default state for the passed mode."
|
2015-11-05 13:09:00 +00:00
|
|
|
(add-to-list 'evilified-state--modes mode)
|
2015-09-26 04:26:16 +00:00
|
|
|
(unless (bound-and-true-p holy-mode)
|
|
|
|
(delq mode evil-emacs-state-modes)
|
|
|
|
(add-to-list 'evil-evilified-state-modes mode)))
|
|
|
|
|
2015-11-05 13:09:00 +00:00
|
|
|
(defun evilified-state--evilify-event (map map-symbol evil-map event evil-value
|
2015-09-26 04:26:16 +00:00
|
|
|
&optional processed pending-funcs)
|
|
|
|
"Evilify EVENT in MAP and return a list of PROCESSED events."
|
|
|
|
(if (and event (or evil-value pending-funcs))
|
|
|
|
(let* ((kbd-event (kbd (single-key-description event)))
|
|
|
|
(map-value (lookup-key map kbd-event))
|
|
|
|
(evil-value (or evil-value
|
|
|
|
(lookup-key evil-map kbd-event)
|
|
|
|
(car (pop pending-funcs)))))
|
|
|
|
(when evil-value
|
|
|
|
(evil-define-key 'evilified map kbd-event evil-value))
|
|
|
|
(when map-value
|
|
|
|
(add-to-list 'pending-funcs (cons map-value event) 'append))
|
|
|
|
(push event processed)
|
2015-11-05 13:09:00 +00:00
|
|
|
(setq processed (evilified-state--evilify-event
|
2015-09-26 04:26:16 +00:00
|
|
|
map map-symbol evil-map
|
2015-11-05 13:09:00 +00:00
|
|
|
(evilified-state--find-new-event event) nil
|
2015-09-26 04:26:16 +00:00
|
|
|
processed pending-funcs)))
|
|
|
|
(when pending-funcs
|
|
|
|
(spacemacs-buffer/warning
|
|
|
|
(concat (format (concat "Auto-evilication could not remap these "
|
|
|
|
"functions in map `%s':\n")
|
|
|
|
map-symbol)
|
|
|
|
(mapconcat (lambda (x)
|
|
|
|
(format " - `%s' originally mapped on `%s'"
|
|
|
|
(car x) (single-key-description (cdr x))))
|
|
|
|
pending-funcs "\n")))))
|
|
|
|
processed)
|
|
|
|
|
2015-11-05 13:09:00 +00:00
|
|
|
(defun evilified-state--find-new-event (event)
|
2015-09-26 04:26:16 +00:00
|
|
|
"Return a new event for the evilified EVENT."
|
|
|
|
(when event
|
|
|
|
(cond
|
2015-09-30 01:35:03 +00:00
|
|
|
((equal event ?\a) nil) ; C-g (cannot remap C-g)
|
|
|
|
((equal event 32) ?') ; space
|
2015-09-30 01:16:25 +00:00
|
|
|
((equal event ?/) ?\\)
|
2015-09-30 01:27:21 +00:00
|
|
|
((equal event ?:) ?|)
|
2015-09-26 04:26:16 +00:00
|
|
|
((and (<= ?a event) (<= event ?z)) (- event 32))
|
2015-09-30 01:35:03 +00:00
|
|
|
((equal event ?G) (+ (expt 2 25) ?\a)) ; G is mapped directly to C-S-g
|
2015-09-26 04:26:16 +00:00
|
|
|
((and (<= ?A event) (<= event ?Z)) (- event 64))
|
|
|
|
((and (<= 1 event) (<= event 26)) (+ (expt 2 25) event)))))
|
|
|
|
|
2015-11-05 13:09:00 +00:00
|
|
|
(defun evilified-state--sort-keymap (map)
|
2015-09-26 04:26:16 +00:00
|
|
|
"Sort MAP following the order: `s' > `S' > `C-s' > `C-S-s'"
|
|
|
|
(let (list)
|
|
|
|
(map-keymap (lambda (a b) (push (cons a b) list)) map)
|
|
|
|
(sort list
|
|
|
|
(lambda (a b)
|
|
|
|
(setq a (car a) b (car b))
|
|
|
|
(if (integerp a)
|
|
|
|
(if (integerp b)
|
|
|
|
(if (and (< a 256) (< b 256))
|
|
|
|
(> a b)
|
|
|
|
(< a b))
|
|
|
|
t)
|
|
|
|
(if (integerp b) nil
|
|
|
|
(string< a b)))))))
|
|
|
|
|
2015-11-05 13:09:00 +00:00
|
|
|
(defun evilified-state--mplist-get (plist prop)
|
|
|
|
"Get the values associated to PROP in PLIST, a modified plist.
|
|
|
|
|
|
|
|
A modified plist is one where keys are keywords and values are
|
|
|
|
all non-keywords elements that follow it.
|
|
|
|
|
|
|
|
If there are multiple properties with the same keyword, only the first property
|
|
|
|
and its values is returned.
|
|
|
|
|
|
|
|
Currently this function infloops when the list is circular."
|
|
|
|
(let ((tail plist)
|
|
|
|
result)
|
|
|
|
(while (and (consp tail) (not (eq prop (car tail))))
|
|
|
|
(pop tail))
|
|
|
|
;; pop the found keyword
|
|
|
|
(pop tail)
|
|
|
|
(while (and (consp tail) (not (keywordp (car tail))))
|
|
|
|
(push (pop tail) result))
|
|
|
|
(nreverse result)))
|
|
|
|
|
2015-11-19 05:24:27 +00:00
|
|
|
(provide 'evil-evilified-state)
|
2015-08-12 05:31:44 +00:00
|
|
|
|
|
|
|
;;; core-evilified-state.el ends here
|