From 46b038c60afe54b8795e77ef239f8487ba418d3d Mon Sep 17 00:00:00 2001 From: Stephane Zermatten Date: Sun, 29 Dec 2024 19:10:33 +0200 Subject: [PATCH] feat: Detect ipython, use vertical movements. Before this change, MisTTY always used only horizontal movement, no matter mistty-vertical-multiline-prompt-regexps. With this change, mistty-vertical-multiline-prompt-regexps is taken into account to enable or not vertical moves. The default value of this option detects the default ipython prompts. --- mistty.el | 19 +++++- test/mistty-test.el | 145 ++++++++++++++++++++++---------------------- 2 files changed, 89 insertions(+), 75 deletions(-) diff --git a/mistty.el b/mistty.el index 8cf7045..5c2ca37 100644 --- a/mistty.el +++ b/mistty.el @@ -186,7 +186,7 @@ Set to 0 to disable truncation." :type 'natnum) (defcustom mistty-move-vertically-regexps - '("^In \\[0-9+\\]: " ; ipython + '("^In \\[[0-9]+\\]: " ; ipython ) "Regexp that signals availability of vertical moves. @@ -1570,6 +1570,11 @@ Also updates prompt and point." (mistty--set-sync-mark-from-end prompt-beg) (setq mistty--has-active-prompt (> cursor prompt-beg))))) + (let ((v (and on-prompt (mistty--can-move-vertically-p)))) + (unless (eq v mistty--can-move-vertically) + (mistty-log "Can move vertically: %s" v) + (setq mistty--can-move-vertically v))) + ;; Turn mistty-forbid-edit on or off (let ((forbid-edit (mistty--match-forbid-edit-regexp-p))) (cond @@ -1684,6 +1689,18 @@ cursor to be considered." (setq match t)))) match))) +(defun mistty--can-move-vertically-p () + "Check whether vertical moves are allowed, return either nil or t. + +Does not update `mistty--can-move-vertically'." + (save-excursion + (goto-char mistty-sync-marker) + (catch 'mistty-end-loop + (dolist (regexp mistty-move-vertically-regexps) + (when (search-forward-regexp regexp (pos-eol) 'noerror) + (throw 'mistty-end-loop t))) + nil))) + (defun mistty--sync-buffer (source-buffer &optional quick) "Copy the sync region of SOURCE-BUFFER to the current buffer. diff --git a/test/mistty-test.el b/test/mistty-test.el index 10c159d..7180bd2 100644 --- a/test/mistty-test.el +++ b/test/mistty-test.el @@ -4435,95 +4435,92 @@ (ert-deftest mistty-test-ipython-reconcile-in-another-line () (mistty-with-test-buffer (:shell ipython) - (let ((mistty--can-move-vertically t)) - (mistty--send-string mistty-proc "for i in (1, 2, 3):\nif i > 2:\nprint(i)") - (mistty-wait-for-output :test (lambda () (save-excursion - (goto-char (point-min)) - (and (search-forward "...:" nil 'noerror) - (search-forward "...:" nil 'noerror))))) - (let ((start (save-excursion - (goto-char (point-min)) - (search-forward "In [") - (match-beginning 0)))) + (mistty--send-string mistty-proc "for i in (1, 2, 3):\nif i > 2:\nprint(i)") + (mistty-wait-for-output :test (lambda () (save-excursion + (goto-char (point-min)) + (and (search-forward "...:" nil 'noerror) + (search-forward "...:" nil 'noerror))))) + (let ((start (save-excursion + (goto-char (point-min)) + (search-forward "In [") + (match-beginning 0)))) - (should (equal (concat "In [1]: for i in (1, 2, 3):\n" - " ...: if i > 2:\n" - " ...: print(i)") - (mistty-test-content :start start))) + (should (equal (concat "In [1]: for i in (1, 2, 3):\n" + " ...: if i > 2:\n" + " ...: print(i)") + (mistty-test-content :start start))) - (mistty-run-command - (goto-char (point-min)) - (search-forward "(1, 2, 3)") - (replace-match "(10, 11)" nil t) - (goto-char (point-min))) + (mistty-run-command + (goto-char (point-min)) + (search-forward "(1, 2, 3)") + (replace-match "(10, 11)" nil t) + (goto-char (point-min))) - (should (equal (concat "In [1]: for i in (10, 11):\n" - " ...: if i > 2:\n" - " ...: print(i)") - (mistty-test-content :start start))))))) + (should (equal (concat "In [1]: for i in (10, 11):\n" + " ...: if i > 2:\n" + " ...: print(i)") + (mistty-test-content :start start)))))) (ert-deftest mistty-test-ipython-reconcile-multiline-delete () (mistty-with-test-buffer (:shell ipython) - (let ((mistty--can-move-vertically t)) - (mistty--send-string mistty-proc "for i in (1, 2, 3):\nif i > 2:\nprint(i)") - (mistty-wait-for-output :test (lambda () (save-excursion - (goto-char (point-min)) - (and (search-forward "...:" nil 'noerror) - (search-forward "...:" nil 'noerror))))) - (let ((start (save-excursion - (goto-char (point-min)) - (search-forward "In [") - (match-beginning 0)))) - - (should (equal (concat "In [1]: for i in (1, 2, 3):\n" - " ...: if i > 2:\n" - " ...: print(i)") - (mistty-test-content :start start))) + (mistty--send-string mistty-proc "for i in (1, 2, 3):\nif i > 2:\nprint(i)") + (mistty-wait-for-output :test (lambda () (save-excursion + (goto-char (point-min)) + (and (search-forward "...:" nil 'noerror) + (search-forward "...:" nil 'noerror))))) + (let ((start (save-excursion + (goto-char (point-min)) + (search-forward "In [") + (match-beginning 0)))) + + (should (equal (concat "In [1]: for i in (1, 2, 3):\n" + " ...: if i > 2:\n" + " ...: print(i)") + (mistty-test-content :start start))) - (mistty-run-command - (goto-char (point-min)) - (search-forward-regexp "if i > 2\\(.\\|\n\\)*print(i)") - (replace-match "total += i" nil t)) - (goto-char (point-min)) + (mistty-run-command + (goto-char (point-min)) + (search-forward-regexp "if i > 2\\(.\\|\n\\)*print(i)") + (replace-match "total += i" nil t)) + (goto-char (point-min)) - (should (equal (concat "In [1]: for i in (1, 2, 3):\n" - " ...: total += i") - (mistty-test-content :start start))))))) + (should (equal (concat "In [1]: for i in (1, 2, 3):\n" + " ...: total += i") + (mistty-test-content :start start)))))) (ert-deftest mistty-test-ipython-move-cursor () (mistty-with-test-buffer (:shell ipython) - (let ((mistty--can-move-vertically t)) - (mistty--send-string mistty-proc "for i in (1, 2, 3):\nif i > 2:\nprint(i)") - (mistty-wait-for-output :test (lambda () (save-excursion - (goto-char (point-min)) - (and (search-forward "...:" nil 'noerror) - (search-forward "...:" nil 'noerror))))) - (let ((start (save-excursion - (goto-char (point-min)) - (search-forward "In [") - (match-beginning 0)))) + (mistty--send-string mistty-proc "for i in (1, 2, 3):\nif i > 2:\nprint(i)") + (mistty-wait-for-output :test (lambda () (save-excursion + (goto-char (point-min)) + (and (search-forward "...:" nil 'noerror) + (search-forward "...:" nil 'noerror))))) + (let ((start (save-excursion + (goto-char (point-min)) + (search-forward "In [") + (match-beginning 0)))) - (should (equal (concat "In [1]: for i in (1, 2, 3):\n" - " ...: if i > 2:\n" - " ...: print(i)<>") - (mistty-test-content :start start :show (point)))) + (should (equal (concat "In [1]: for i in (1, 2, 3):\n" + " ...: if i > 2:\n" + " ...: print(i)<>") + (mistty-test-content :start start :show (point)))) - (mistty-run-command - (goto-char (point-min)) - (mistty-test-goto "(1, 2, 3)")) + (mistty-run-command + (goto-char (point-min)) + (mistty-test-goto "(1, 2, 3)")) - (should (equal (concat "In [1]: for i in <>(1, 2, 3):\n" - " ...: if i > 2:\n" - " ...: print(i)") - (mistty-test-content :start start :show (point)))) + (should (equal (concat "In [1]: for i in <>(1, 2, 3):\n" + " ...: if i > 2:\n" + " ...: print(i)") + (mistty-test-content :start start :show (point)))) - (mistty-run-command - (goto-char (point-min)) - (mistty-test-goto "if")) + (mistty-run-command + (goto-char (point-min)) + (mistty-test-goto "if")) - (should (equal (concat "In [1]: for i in (1, 2, 3):\n" - " ...: <>if i > 2:\n" - " ...: print(i)") - (mistty-test-content :start start :show (point)))))))) + (should (equal (concat "In [1]: for i in (1, 2, 3):\n" + " ...: <>if i > 2:\n" + " ...: print(i)") + (mistty-test-content :start start :show (point)))))))