aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--modules/term-config.el91
-rw-r--r--tests/test-term-tmux-history.el81
2 files changed, 162 insertions, 10 deletions
diff --git a/modules/term-config.el b/modules/term-config.el
index c1c28911d..d8e9c05d0 100644
--- a/modules/term-config.el
+++ b/modules/term-config.el
@@ -206,6 +206,63 @@ start of the line for the same column-0 reason."
(ghostel-copy-mode)
(beginning-of-line)))
+;; --------------------------- directional copy mode ---------------------------
+;;
+;; A modified arrow both enters copy-mode and carries its own movement, so one
+;; stroke scrolls back in the intended direction. C-<arrow> and M-<arrow> bind
+;; to these in ghostel buffers (and join `ghostel-keymap-exceptions' so they
+;; reach Emacs instead of the pty).
+
+(defconst cj/--term-copy-mode-tmux-arrows
+ '((up . "\e[A") (down . "\e[B") (left . "\e[D") (right . "\e[C"))
+ "Per-direction arrow escape sequence written into tmux copy-mode.
+After `cj/term-copy-mode-dwim' lands in tmux's copy-mode, writing the matching
+sequence into the pty moves the copy cursor one step; tmux binds the cursor
+keys in both its emacs and vi copy-mode tables.")
+
+(defun cj/--term-copy-mode-move-step (direction in-tmux)
+ "Take one copy-mode movement step in DIRECTION.
+DIRECTION is one of `up', `down', `left', `right'. When IN-TMUX is non-nil,
+write the matching arrow escape sequence into the pty so tmux's copy-mode
+cursor follows it; otherwise move point in the `ghostel-copy-mode' buffer.
+An unknown DIRECTION is a no-op."
+ (if in-tmux
+ (let ((seq (cdr (assq direction cj/--term-copy-mode-tmux-arrows))))
+ (when seq (ghostel-send-string seq)))
+ (pcase direction
+ ('up (forward-line -1))
+ ('down (forward-line 1))
+ ('left (backward-char))
+ ('right (forward-char)))))
+
+(defun cj/term-copy-mode-move (direction)
+ "Enter copy-mode (see `cj/term-copy-mode-dwim') then step one DIRECTION.
+DIRECTION is one of `up', `down', `left', `right'. The in-tmux engine is
+resolved once up front so the entry and the step pick the same surface."
+ (let ((in-tmux (cj/term--in-tmux-p)))
+ (cj/term-copy-mode-dwim)
+ (cj/--term-copy-mode-move-step direction in-tmux)))
+
+(defun cj/term-copy-mode-up ()
+ "Enter the terminal's copy-mode and move up one step."
+ (interactive)
+ (cj/term-copy-mode-move 'up))
+
+(defun cj/term-copy-mode-down ()
+ "Enter the terminal's copy-mode and move down one step."
+ (interactive)
+ (cj/term-copy-mode-move 'down))
+
+(defun cj/term-copy-mode-left ()
+ "Enter the terminal's copy-mode and move left one step."
+ (interactive)
+ (cj/term-copy-mode-move 'left))
+
+(defun cj/term-copy-mode-right ()
+ "Enter the terminal's copy-mode and move right one step."
+ (interactive)
+ (cj/term-copy-mode-move 'right))
+
;; ----------------------------- ghostel package -------------------------------
(defun cj/turn-off-chrome-for-term ()
@@ -245,16 +302,20 @@ run its own project-named tmux session instead of a bare, auto-named one.
;; `add-to-list' alone updates the list but not the already-built map, so the
;; rebuild is what actually lets the key through to `ghostel-mode-map' / the
;; global map. C-; and F12 are the prefix + toggle; the modified arrows are
- ;; windmove (S-arrows, focus) and buffer-move (C-M-arrows, swap), which the
- ;; ai-term workflow expects to work from inside an agent buffer. F8 and F10
- ;; are global bindings (org agenda, music-playlist toggle) that reach Emacs by
- ;; falling through to the global map once the semi-char map stops forwarding
- ;; them. (Server shutdown moved off C-F10 to C-x C, which is deliberately
- ;; left forwarding to the terminal program inside an agent buffer.)
+ ;; windmove (S-arrows, focus), buffer-move (C-M-arrows, swap), and copy-mode
+ ;; entry (C-arrows and M-arrows, each entering copy-mode and carrying its
+ ;; direction via `cj/term-copy-mode-up' & friends), which the ai-term workflow
+ ;; expects to work from inside an agent buffer. F8 and F10 are global
+ ;; bindings (org agenda, music-playlist toggle) that reach Emacs by falling
+ ;; through to the global map once the semi-char map stops forwarding them.
+ ;; (Server shutdown moved off C-F10 to C-x C, which is deliberately left
+ ;; forwarding to the terminal program inside an agent buffer.)
(with-eval-after-load 'ghostel
(dolist (key '("C-;" "<f8>" "<f12>" "<f10>"
"S-<up>" "S-<down>" "S-<left>" "S-<right>"
- "C-M-<up>" "C-M-<down>" "C-M-<left>" "C-M-<right>"))
+ "C-M-<up>" "C-M-<down>" "C-M-<left>" "C-M-<right>"
+ "C-<up>" "C-<down>" "C-<left>" "C-<right>"
+ "M-<up>" "M-<down>" "M-<left>" "M-<right>"))
(add-to-list 'ghostel-keymap-exceptions key))
(ghostel--rebuild-semi-char-keymap))
:hook
@@ -452,12 +513,22 @@ Forwarding NUL makes C-Space behave like a terminal key."
(defun cj/term-install-keys ()
"Make `C-;' resolve as the personal keymap inside ghostel buffers, bind the
-F12 toggle, and forward C-SPC so it reaches the terminal (see
-`cj/term-send-C-SPC')."
+F12 toggle, forward C-SPC so it reaches the terminal (see
+`cj/term-send-C-SPC'), and bind C-/M-<arrow> to enter copy-mode and step in
+that direction."
(when (boundp 'ghostel-mode-map)
(keymap-set ghostel-mode-map "C-;" cj/custom-keymap)
(keymap-set ghostel-mode-map "<f12>" #'cj/term-toggle)
- (keymap-set ghostel-mode-map "C-SPC" #'cj/term-send-C-SPC)))
+ (keymap-set ghostel-mode-map "C-SPC" #'cj/term-send-C-SPC)
+ (dolist (binding '(("C-<up>" . cj/term-copy-mode-up)
+ ("M-<up>" . cj/term-copy-mode-up)
+ ("C-<down>" . cj/term-copy-mode-down)
+ ("M-<down>" . cj/term-copy-mode-down)
+ ("C-<left>" . cj/term-copy-mode-left)
+ ("M-<left>" . cj/term-copy-mode-left)
+ ("C-<right>" . cj/term-copy-mode-right)
+ ("M-<right>" . cj/term-copy-mode-right)))
+ (keymap-set ghostel-mode-map (car binding) (cdr binding)))))
(cj/term-install-keys)
(with-eval-after-load 'ghostel
diff --git a/tests/test-term-tmux-history.el b/tests/test-term-tmux-history.el
index 4ad7fb79d..89b4875eb 100644
--- a/tests/test-term-tmux-history.el
+++ b/tests/test-term-tmux-history.el
@@ -355,5 +355,86 @@ Emacs region gets stuck in the ghostel buffer and tmux copy-mode's
begin-selection never starts."
(should (eq (keymap-lookup ghostel-mode-map "C-SPC") #'cj/term-send-C-SPC)))
+;; --------------------------- directional copy mode ---------------------------
+
+(ert-deftest test-term-copy-mode-step-tmux-sends-arrow ()
+ "Normal: in tmux, each direction writes its arrow escape sequence to the pty."
+ (dolist (case '((up . "\e[A") (down . "\e[B") (left . "\e[D") (right . "\e[C")))
+ (let ((sent nil))
+ (cl-letf (((symbol-function 'ghostel-send-string)
+ (lambda (s) (push s sent))))
+ (cj/--term-copy-mode-move-step (car case) t)
+ (should (equal sent (list (cdr case))))))))
+
+(ert-deftest test-term-copy-mode-step-nontmux-moves-point ()
+ "Normal: without tmux, up/down move by line and left/right move by char."
+ (with-temp-buffer
+ (insert "abc\ndef\nghi\n")
+ ;; up/down: land on line 2, step around.
+ (goto-char (point-min))
+ (forward-line 1)
+ (should (= (line-number-at-pos) 2))
+ (cj/--term-copy-mode-move-step 'up nil)
+ (should (= (line-number-at-pos) 1))
+ (cj/--term-copy-mode-move-step 'down nil)
+ (should (= (line-number-at-pos) 2))
+ ;; left/right: move by a single character.
+ (goto-char 2)
+ (cj/--term-copy-mode-move-step 'right nil)
+ (should (= (point) 3))
+ (cj/--term-copy-mode-move-step 'left nil)
+ (should (= (point) 2))))
+
+(ert-deftest test-term-copy-mode-step-unknown-direction-no-op ()
+ "Boundary: an unknown direction sends nothing in the tmux branch."
+ (let ((sent nil))
+ (cl-letf (((symbol-function 'ghostel-send-string)
+ (lambda (s) (push s sent))))
+ (cj/--term-copy-mode-move-step 'sideways t)
+ (should-not sent))))
+
+(ert-deftest test-term-copy-mode-up-command-enters-then-steps ()
+ "Normal: the bound command enters tmux copy-mode, then sends the arrow,
+so a single modified arrow both enters copy-mode and carries its direction."
+ (let ((agent (cj/test--make-fake-ghostel-buffer "agent [emacs.d]"))
+ (sent nil))
+ (unwind-protect
+ (with-current-buffer agent
+ (cl-letf (((symbol-function 'get-buffer-process)
+ (lambda (_buffer) 'fake-process))
+ ((symbol-function 'process-tty-name)
+ (lambda (_process &rest _) "/dev/pts/8"))
+ ((symbol-function 'ghostel-send-string)
+ (lambda (s) (push s sent))))
+ (test-term-tmux-history--with-tmux-mock
+ '((("list-clients" "-F" "#{client_tty}\t#{pane_id}") 0
+ "/dev/pts/8\t%8\n"))
+ (cj/term-copy-mode-up)
+ (should (equal (reverse sent) '("\C-b[\C-a" "\e[A"))))))
+ (when (buffer-live-p agent)
+ (kill-buffer agent)))))
+
+(ert-deftest test-term-copy-mode-arrows-bound-in-ghostel-map ()
+ "Normal: C-<arrow> and M-<arrow> both reach the directional copy-mode commands."
+ (dolist (case '(("C-<up>" . cj/term-copy-mode-up)
+ ("M-<up>" . cj/term-copy-mode-up)
+ ("C-<down>" . cj/term-copy-mode-down)
+ ("M-<down>" . cj/term-copy-mode-down)
+ ("C-<left>" . cj/term-copy-mode-left)
+ ("M-<left>" . cj/term-copy-mode-left)
+ ("C-<right>" . cj/term-copy-mode-right)
+ ("M-<right>" . cj/term-copy-mode-right)))
+ (should (eq (keymap-lookup ghostel-mode-map (car case)) (cdr case)))))
+
+(ert-deftest test-term-copy-mode-arrows-in-keymap-exceptions ()
+ "Regression: C-/M-<arrow> are in `ghostel-keymap-exceptions' and the rebuilt
+semi-char map no longer forwards them to the pty, so they reach Emacs and
+trigger copy-mode entry from inside a ghostel buffer."
+ (dolist (key '("C-<up>" "C-<down>" "C-<left>" "C-<right>"
+ "M-<up>" "M-<down>" "M-<left>" "M-<right>"))
+ (should (member key ghostel-keymap-exceptions)))
+ (should-not (eq (keymap-lookup ghostel-semi-char-mode-map "C-<up>")
+ 'ghostel--send-event)))
+
(provide 'test-term-tmux-history)
;;; test-term-tmux-history.el ends here