[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
[nongnu] elpa/multiple-cursors 7a6eb0d 145/434: Never execute keyboard m
From: |
ELPA Syncer |
Subject: |
[nongnu] elpa/multiple-cursors 7a6eb0d 145/434: Never execute keyboard macros for fake cursors. |
Date: |
Sat, 7 Aug 2021 09:20:14 -0400 (EDT) |
branch: elpa/multiple-cursors
commit 7a6eb0df90002f4cddd32993c00333eb096adbfb
Author: Magnar Sveen <magnars@gmail.com>
Commit: Magnar Sveen <magnars@gmail.com>
Never execute keyboard macros for fake cursors.
- The real cursor will execute the keyboard macro, resulting in new
commands in the command loop, and the fake cursors can pick up on
those instead.
Fixes #18
---
features/multiple-cursors-core.feature | 10 +
features/support/env.el | 3 +
multiple-cursors-core.el | 7 +
util/vendor/wrap-region.el | 329 +++++++++++++++++++++++++++++++++
4 files changed, 349 insertions(+)
diff --git a/features/multiple-cursors-core.feature
b/features/multiple-cursors-core.feature
index a889b9c..46d14bf 100644
--- a/features/multiple-cursors-core.feature
+++ b/features/multiple-cursors-core.feature
@@ -119,6 +119,16 @@ Feature: Multiple cursors core
And I type "!"
Then I should see "This ! contains the word ! twice"
+ Scenario: wrap-region (function turns to keyboard macros)
+ Given I turn on wrap-region-mode
+ And I insert "This text contains the word text twice"
+ And I go to the front of the word "text"
+ And I press "C-M-SPC"
+ And I press "C->"
+ And I press "C-g"
+ And I type "("
+ Then I should see "This (text contains the word (text twice"
+
Scenario: Bound keyboard macros
Given I have bound C-! to a keyboard macro that insert "_"
And I have cursors at "text" in "This text contains the word text twice"
diff --git a/features/support/env.el b/features/support/env.el
index a55d319..33fa6e7 100644
--- a/features/support/env.el
+++ b/features/support/env.el
@@ -7,10 +7,12 @@
(add-to-list 'load-path multiple-cursors-root-path)
(add-to-list 'load-path multiple-cursors-util-path)
(add-to-list 'load-path (expand-file-name "espuds" multiple-cursors-util-path))
+(add-to-list 'load-path (expand-file-name "vendor" multiple-cursors-util-path))
(require 'multiple-cursors)
(require 'espuds)
(require 'ert)
+(require 'wrap-region)
(defun mc/save-lists ()) ;; redefine to do nothing when running tests
@@ -31,6 +33,7 @@
(cua-mode 0)
(delete-selection-mode 0)
(subword-mode 0)
+ (wrap-region-mode 0)
(setq set-mark-default-inactive nil)
(deactivate-mark))
diff --git a/multiple-cursors-core.el b/multiple-cursors-core.el
index d38285b..6312db5 100644
--- a/multiple-cursors-core.el
+++ b/multiple-cursors-core.el
@@ -297,6 +297,13 @@ not be recognized through the command-remapping lookup."
(message "[mc] problem in `mc/execute-this-command-for-all-cursors': %s"
(error-message-string error)))))
+;; execute-kbd-macro should never be run for fake cursors. The real cursor will
+;; execute the keyboard macro, resulting in new commands in the command loop,
+;; and the fake cursors can pick up on those instead.
+(defadvice execute-kbd-macro (around skip-fake-cursors activate)
+ (unless mc--executing-command-for-fake-cursor
+ ad-do-it))
+
(defun mc/execute-this-command-for-all-cursors-1 ()
"Used with post-command-hook to execute supported commands for all cursors.
diff --git a/util/vendor/wrap-region.el b/util/vendor/wrap-region.el
new file mode 100644
index 0000000..91bff51
--- /dev/null
+++ b/util/vendor/wrap-region.el
@@ -0,0 +1,329 @@
+;;; wrap-region.el --- Wrap text with punctation or tag
+
+;; Copyright (C) 2008-2012 Johan Andersson
+
+;; Author: Johan Andersson <johan.rejeep@gmail.com>
+;; Maintainer: Johan Andersson <johan.rejeep@gmail.com>
+;; Version: 0.6.0
+;; Keywords: speed, convenience
+;; URL: http://github.com/rejeep/wrap-region
+
+;; This file is NOT part of GNU Emacs.
+
+
+;;; License:
+
+;; 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, 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 GNU Emacs; see the file COPYING. If not, write to the
+;; Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+;; Boston, MA 02110-1301, USA.
+
+
+;;; Commentary:
+
+;; wrap-region is a minor mode that wraps a region with
+;; punctuations. For tagged markup modes, such as HTML and XML, it
+;; wraps with tags.
+;;
+;; To use wrap-region, make sure that this file is in Emacs load-path:
+;; (add-to-list 'load-path "/path/to/directory/or/file")
+;;
+;; Then require wrap-region:
+;; (require 'wrap-region)
+
+;; To start wrap-region:
+;; (wrap-region-mode t) or M-x wrap-region-mode
+;;
+;; If you only want wrap-region active in some mode, use hooks:
+;; (add-hook 'ruby-mode-hook 'wrap-region-mode)
+;;
+;; Or if you want to activate it in all buffers, use the global mode:
+;; (wrap-region-global-mode t)
+
+;; To wrap a region, select that region and hit one of the punctuation
+;; keys. In "tag-modes"" (see `wrap-region-tag-active-modes'), "<" is
+;; replaced and wraps the region with a tag. To activate this behavior
+;; in a mode that is not default:
+;; (add-to-list 'wrap-region-tag-active-modes 'some-tag-mode)
+;;
+;; `wrap-region-table' contains the default punctuations
+;; that wraps. You can add and remove new wrappers by using the
+;; functions `wrap-region-add-wrapper' and
+;; `wrap-region-remove-wrapper' respectively.
+;; (wrap-region-add-wrapper "`" "'") ; hit ` then region ->
`region'
+;; (wrap-region-add-wrapper "/*" "*/" "/") ; hit / then region ->
/*region*/
+;; (wrap-region-add-wrapper "$" "$" nil 'latex-mode) ; hit $ then region ->
$region$ in latex-mode
+;; (wrap-region-remove-wrapper "(")
+;; (wrap-region-remove-wrapper "$" 'latex-mode)
+;;
+;; Some modes may have conflicting key bindings with wrap-region. To
+;; avoid conflicts, the list `wrap-region-except-modes' contains names
+;; of modes where wrap-region should not be activated (note, only in
+;; the global mode). You can add new modes like this:
+;; (add-to-list 'wrap-region-except-modes 'conflicting-mode)
+
+
+;;; Code:
+
+(require 'edmacro)
+(eval-when-compile
+ (require 'cl))
+
+
+(defstruct wrap-region-wrapper key left right modes)
+
+(defvar wrap-region-mode-map (make-sparse-keymap)
+ "Keymap for `wrap-region-mode'.")
+
+(defvar wrap-region-table (make-hash-table :test 'equal)
+ "Table with wrapper pairs.")
+
+(defvar wrap-region-tag-active-modes '(html-mode sgml-mode rhtml-mode
nxml-mode nxhtml-mode)
+ "List of modes that uses tags.")
+
+(defvar wrap-region-except-modes '(calc-mode dired-mode)
+ "A list of modes in which `wrap-region-mode' should not be activated.")
+
+(defvar wrap-region-hook nil
+ "Called when `wrap-region-mode' is turned on.")
+
+(defvar wrap-region-before-wrap-hook nil
+ "Called before wrapping.")
+
+(defvar wrap-region-after-wrap-hook nil
+ "Called after wrapping.")
+
+(defvar wrap-region-only-with-negative-prefix nil
+ "Only wrap if the trigger key is prefixed with a negative value.")
+
+(defvar wrap-region-keep-mark nil
+ "Keep the wrapped region active.")
+
+(defun wrap-region-trigger (arg key)
+ "Called when trigger key is pressed."
+ (let* ((wrapper (wrap-region-find key)))
+ (if (and wrapper
+ (region-active-p)
+ (if wrap-region-only-with-negative-prefix (< arg 0) t))
+ (if (wrap-region-insert-tag-p key)
+ (wrap-region-with-tag)
+ (wrap-region-with-punctuations
+ (wrap-region-wrapper-left wrapper)
+ (wrap-region-wrapper-right wrapper)))
+ (wrap-region-fallback key))))
+
+(defun wrap-region-find (key)
+ "Find first wrapper with trigger KEY that should be active in MAJOR-MODE."
+ (let ((wrappers (gethash key wrap-region-table)))
+ (or
+ (find-if
+ (lambda (wrapper)
+ (member major-mode (wrap-region-wrapper-modes wrapper)))
+ wrappers)
+ (find-if
+ (lambda (wrapper)
+ (not (wrap-region-wrapper-modes wrapper)))
+ wrappers))))
+
+(defun wrap-region-insert-tag-p (key)
+ "Check if tag should be inserted or not."
+ (and
+ (equal key "<")
+ (member major-mode wrap-region-tag-active-modes)))
+
+(defun wrap-region-with-tag ()
+ "Wraps region with tag."
+ (let* ((tag (read-string "Enter Tag (with optional attributes): "))
+ (split (split-string tag " "))
+ (tag-name (car split))
+ (left (concat "<" tag ">"))
+ (right (concat "</" tag-name ">")))
+ (wrap-region-with left right)))
+
+(defun wrap-region-with-punctuations (left right)
+ "Wraps region with LEFT and RIGHT punctuations."
+ (wrap-region-with left right))
+
+(defun wrap-region-with (left right)
+ "Wraps region with LEFT and RIGHT."
+ (run-hooks 'wrap-region-before-wrap-hook)
+ (let ((beg (region-beginning))
+ (end (region-end))
+ (pos (point))
+ (deactivate-mark nil))
+ (save-excursion
+ (goto-char beg)
+ (insert left)
+ (goto-char (+ end (length left)))
+ (insert right))
+ (if wrap-region-keep-mark
+ (let* ((beg-p (eq beg pos))
+ (beg* (+ beg (length left)))
+ (end* (+ end (length left)))
+ (pos* (if beg-p beg* end*)))
+ (push-mark (if beg-p end* beg*) nil t)
+ (goto-char (if beg-p beg* end*)))
+ (deactivate-mark)))
+ (run-hooks 'wrap-region-after-wrap-hook))
+
+(defun wrap-region-fallback (key)
+ "Execute function that KEY was bound to before `wrap-region-mode'."
+ (let ((wrap-region-mode nil))
+ (execute-kbd-macro
+ (edmacro-parse-keys key))))
+
+(defun wrap-region-add-wrappers (wrappers)
+ "Add WRAPPERS by calling `wrap-region-add-wrapper' for each one."
+ (mapc
+ (lambda (wrapper)
+ (apply 'wrap-region-add-wrapper wrapper))
+ wrappers))
+
+(defun wrap-region-add-wrapper (left right &optional key mode-or-modes)
+ "Add new LEFT and RIGHT wrapper.
+
+Optional KEY is the trigger key and MODE-OR-MODES is a single
+mode or multiple modes that the wrapper should trigger in."
+ (or key (setq key left))
+ (let ((wrappers (gethash key wrap-region-table))
+ (modes
+ (if mode-or-modes
+ (if (listp mode-or-modes)
+ mode-or-modes
+ (list mode-or-modes)))))
+ (if wrappers
+ (let ((wrapper-exactly-same
+ (find-if
+ (lambda (wrapper)
+ (and
+ (equal (wrap-region-wrapper-key wrapper) key)
+ (equal (wrap-region-wrapper-left wrapper) left)
+ (equal (wrap-region-wrapper-right wrapper) right)))
+ wrappers)))
+ (if wrapper-exactly-same
+ (when (wrap-region-wrapper-modes wrapper-exactly-same)
+ (if modes
+ (setf
+ (wrap-region-wrapper-modes wrapper-exactly-same)
+ (union modes (wrap-region-wrapper-modes
wrapper-exactly-same)))
+ (let ((new-wrapper (make-wrap-region-wrapper :key key :left
left :right right)))
+ (puthash key (cons new-wrapper wrappers)
wrap-region-table))))
+ (let* ((new-wrapper (make-wrap-region-wrapper :key key :left left
:right right :modes modes))
+ (wrapper-same-trigger
+ (find-if
+ (lambda (wrapper)
+ (equal (wrap-region-wrapper-key wrapper) key))
+ wrappers))
+ (wrapper-same-trigger-modes
+ (wrap-region-wrapper-modes wrapper-same-trigger)))
+ (when (and wrapper-same-trigger wrapper-same-trigger-modes)
+ (let ((new-modes (nset-difference (wrap-region-wrapper-modes
wrapper-same-trigger) modes)))
+ (if new-modes
+ (setf (wrap-region-wrapper-modes wrapper-same-trigger)
new-modes)
+ (setq wrappers (delete wrapper-same-trigger wrappers)))))
+ (puthash key (cons new-wrapper wrappers) wrap-region-table))))
+ (let ((new-wrapper (make-wrap-region-wrapper :key key :left left :right
right :modes modes)))
+ (puthash key (list new-wrapper) wrap-region-table))))
+ (wrap-region-define-trigger key))
+
+(defun wrap-region-remove-wrapper (key &optional mode-or-modes)
+ "Remove wrapper with trigger KEY or exclude from MODE-OR-MODES.
+
+If MODE-OR-MODES is not present, all wrappers for KEY are removed."
+ (if mode-or-modes
+ (let ((wrappers (gethash key wrap-region-table))
+ (modes
+ (if mode-or-modes
+ (if (listp mode-or-modes)
+ mode-or-modes
+ (list mode-or-modes)))))
+ (mapc
+ (lambda (mode)
+ (let ((wrapper-including-mode
+ (find-if
+ (lambda (wrapper)
+ (member mode (wrap-region-wrapper-modes wrapper)))
+ wrappers)))
+ (when wrapper-including-mode
+ (let ((new-modes (delete mode (wrap-region-wrapper-modes
wrapper-including-mode))))
+ (if new-modes
+ (setf (wrap-region-wrapper-modes wrapper-including-mode)
new-modes)
+ (puthash key (delete wrapper-including-mode wrappers)
wrap-region-table))))))
+ modes))
+ (wrap-region-destroy-wrapper key)))
+
+(defun wrap-region-destroy-wrapper (key)
+ "Remove the wrapper bound to KEY, no questions asked."
+ (remhash key wrap-region-table)
+ (wrap-region-unset-key key))
+
+(defun wrap-region-define-wrappers ()
+ "Defines defaults wrappers."
+ (mapc
+ (lambda (pair)
+ (apply 'wrap-region-add-wrapper pair))
+ '(("\"" "\"")
+ ("'" "'")
+ ("(" ")")
+ ("{" "}")
+ ("[" "]")
+ ("<" ">"))))
+
+(defun wrap-region-define-trigger (key)
+ "Defines KEY as wrapper."
+ (wrap-region-define-key
+ key
+ `(lambda (arg)
+ (interactive "p")
+ (wrap-region-trigger arg ,key))))
+
+(defun wrap-region-unset-key (key)
+ "Remove KEY from `wrap-region-mode-map'."
+ (wrap-region-define-key key))
+
+(defun wrap-region-define-key (key &optional fn)
+ "Binds KEY to FN in `wrap-region-mode-map'."
+ (define-key wrap-region-mode-map (read-kbd-macro key) fn))
+
+
+;;;###autoload
+(define-minor-mode wrap-region-mode
+ "Wrap region with stuff."
+ :init-value nil
+ :lighter " wr"
+ :keymap wrap-region-mode-map
+ (when wrap-region-mode
+ (wrap-region-define-wrappers)
+ (run-hooks 'wrap-region-hook)))
+
+;;;###autoload
+(defun turn-on-wrap-region-mode ()
+ "Turn on `wrap-region-mode'"
+ (interactive)
+ (unless (member major-mode wrap-region-except-modes)
+ (wrap-region-mode +1)))
+
+;;;###autoload
+(defun turn-off-wrap-region-mode ()
+ "Turn off `wrap-region-mode'."
+ (interactive)
+ (wrap-region-mode -1))
+
+;;;###autoload
+(define-globalized-minor-mode wrap-region-global-mode
+ wrap-region-mode
+ turn-on-wrap-region-mode)
+
+(provide 'wrap-region)
+
+;;; wrap-region.el ends here
- [nongnu] elpa/multiple-cursors 53dd119 032/434: Add docstring., (continued)
- [nongnu] elpa/multiple-cursors 53dd119 032/434: Add docstring., ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors c2bb81f 042/434: Dont forget to require rectangular-region-mode, ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors ba11b12 051/434: Make interprogram-paste work for all cursors., ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors 8c8d52f 049/434: Run pre-command-hooks for each cursor, ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors c1fba2c 111/434: Add travis badge, ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors f73569f 125/434: Update documentation., ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors ce777dc 134/434: Add some mouse-commands to the run-once list., ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors 879ddb0 150/434: Add Takafumi Arakaki to list of contributors, ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors 64ffd81 142/434: Add quoted-insert to run-for-all list., ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors 00f9055 146/434: Add mc/dump-list to make mc/save-lists diff-friendly, ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors 7a6eb0d 145/434: Never execute keyboard macros for fake cursors.,
ELPA Syncer <=
- [nongnu] elpa/multiple-cursors 82a1fe3 209/434: Add mc/sort-regions and mc/reverse-regions, ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors aa9a1ec 220/434: Add failing scenarios for mark-all-symbols/words, ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors e7a5fe6 213/434: Release 1.2.1, ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors 6b09b9c 172/434: Add some more basic commands to the run-all and run-once lists., ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors 7382b46 170/434: Merge pull request #30 from gvol/master, ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors 5f7c92b 173/434: Add more backward-delete commands to run-for-all, ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors 3f3cdc6 207/434: Merge pull request #56 from paradoxxxzero/transient-mark-mode, ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors 7d79b98 182/434: Autoload new functions., ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors 0b9aae1 188/434: Merge pull request #40 from tkf/fix-dump-list, ELPA Syncer, 2021/08/07
- [nongnu] elpa/multiple-cursors 56839df 208/434: Update docs with mc/edit-*-lines, ELPA Syncer, 2021/08/07