evil-collection/evil-collection-term.el

161 lines
5.9 KiB
EmacsLisp

;;; evil-collection-term.el --- Evil bindings for term and ansi-term -*- lexical-binding: t -*-
;; Copyright (C) 2017 Pierre Neidhardt
;; Author: Pierre Neidhardt <mail@ambrevar.xyz>
;; Maintainer: James Nguyen <james@jojojames.com>
;; Pierre Neidhardt <mail@ambrevar.xyz>
;; URL: https://github.com/emacs-evil/evil-collection
;; Version: 0.0.1
;; Package-Requires: ((emacs "25.1"))
;; Keywords: evil, term, tools
;; This file 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, or (at your
;; option) any later version.
;;
;; This file 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.
;;
;; For a full copy of the GNU General Public License
;; see <http://www.gnu.org/licenses/>.
;;; Commentary:
;; Evil integration for `term' and `ansi-term'.
;; This should also work for other terminal emulators such as `multi-term'.
;;
;; Switching to normal state will automatically switch to line mode.
;; Conversely, switching to insert state will automatically switch to char mode.
;;; Code:
(require 'evil-collection)
(require 'term)
(defcustom evil-collection-term-sync-state-and-mode-p t
"Synchronize insert/normal state with char/line-mode respectively.
When non-nil, going to normal state will automatically switch to
line-mode. Conversely, going to insert state on the last
commandline will automatically switch to char-mode.
Warning: This feature is experimental."
:group 'evil-collection-term
:type 'boolean)
(defcustom evil-collection-term-sync-state-function
'evil-collection-term-switch-to-char-mode-on-insert
"Function used when synchronizing insert/normal state with char/line-mode.
This is only used if `evil-collection-term-sync-state-and-mode-p' is true."
:group 'evil-collection-term
:type 'function)
;; TODO: Rebinding ESC has the drawback that programs like vi cannot use it anymore.
;; Workaround: switch to Emacs state and double-press ESC.
;; Otherwise leave ESC to "C-c C-j".
;; Or bind char-mode ESC to "C-c C-x"?
;; TODO: Add support for normal-state editing.
(defconst evil-collection-term-maps '(term-raw-map
term-mode-map))
(defun evil-collection-term-escape-stay ()
"Go back to normal state but don't move cursor backwards.
Moving cursor backwards is the default Vim behavior but
it is not appropriate in some cases like terminals."
(setq-local evil-move-cursor-back nil))
(defun evil-collection-term-char-mode-insert ()
"Switch to `term-char-mode' and enter insert state."
(interactive)
(term-char-mode)
(evil-insert-state))
(defun evil-collection-term-char-mode-entry-function ()
"Maybe switch to `term-char-mode' on insert state."
(when (get-buffer-process (current-buffer))
(let (last-prompt)
(save-excursion
(goto-char (point-max))
(when (= (line-beginning-position) (line-end-position))
(ignore-errors (backward-char)))
(setq last-prompt (max (term-bol nil) (line-beginning-position))))
(when (>= (point) last-prompt)
(term-char-mode)))))
(defun evil-collection-term-switch-to-char-mode-on-insert ()
"Switch to `term-char-mode' on insert state."
(when (get-buffer-process (current-buffer))
(term-char-mode)))
(defun evil-collection-term-sync-state-and-mode ()
"Sync `term-char-mode' and `term-line-mode' with insert and normal state."
(add-hook 'evil-insert-state-entry-hook
evil-collection-term-sync-state-function nil t)
(add-hook 'evil-insert-state-exit-hook 'term-line-mode nil t))
(defun evil-collection-term-send-tab ()
"Send tab in term mode."
(interactive)
(term-send-raw-string "\t"))
;;;###autoload
(defun evil-collection-term-setup ()
"Set up `evil' bindings for `term'."
(evil-set-initial-state 'term-mode 'insert)
(if evil-collection-term-sync-state-and-mode-p
(add-hook 'term-mode-hook 'evil-collection-term-sync-state-and-mode)
(remove-hook 'term-mode-hook 'evil-collection-term-sync-state-and-mode))
(add-hook 'term-mode-hook 'evil-collection-term-escape-stay)
;; Evil has some "C-" bindings in insert state that shadow regular terminal bindings.
;; Don't raw-send "C-c" (prefix key) nor "C-h" (help prefix).
(evil-collection-define-key 'insert 'term-raw-map
(kbd "C-a") 'term-send-raw
(kbd "C-b") 'term-send-raw ; Should not be necessary.
(kbd "C-d") 'term-send-raw
(kbd "C-e") 'term-send-raw
(kbd "C-f") 'term-send-raw ; Should not be necessary.
(kbd "C-k") 'term-send-raw
(kbd "C-l") 'term-send-raw ; Should not be necessary.
(kbd "C-n") 'term-send-raw
(kbd "C-o") 'term-send-raw
(kbd "C-p") 'term-send-raw
(kbd "C-q") 'term-send-raw ; Should not be necessary.
(kbd "C-r") 'term-send-raw
(kbd "C-s") 'term-send-raw ; Should not be necessary.
(kbd "C-t") 'term-send-raw
(kbd "C-u") 'term-send-raw ; Should not be necessary.
(kbd "C-v") 'term-send-raw ; Should not be necessary.
(kbd "C-w") 'term-send-raw
(kbd "C-y") 'term-send-raw
(kbd "C-z") 'term-send-raw
(kbd "<tab>") 'evil-collection-term-send-tab ; Should not be necessary.
(kbd "C-c C-d") 'term-send-eof
(kbd "C-c C-z") 'term-stop-subjob)
(evil-collection-define-key 'normal 'term-mode-map
(kbd "C-c C-k") 'evil-collection-term-char-mode-insert
(kbd "<return>") 'term-send-input
(kbd "p") 'term-paste
;; motion
"[[" 'term-previous-prompt
"]]" 'term-next-prompt
(kbd "C-k") 'term-previous-prompt
(kbd "C-j") 'term-next-prompt
"gk" 'term-previous-prompt
"gj" 'term-next-prompt
;; "0" 'term-bol ; "0" is meant to really go at the beginning of line.
"^" 'term-bol
"$" 'term-show-maximum-output))
(provide 'evil-collection-term)
;;; evil-collection-term.el ends here