of how to edit the macro. When you are finished editing, type
@kbd{C-c C-c}.
+@findex edmacro-insert-key
+@findex edmacro-set-macro-to-region-lines
+ @code{edmacro-mode}, the major mode used by
+@code{kmacro-edit-macro}, provides commands for more easily editing
+the formatted macro. Use @kbd{C-c C-q} (@code{edmacro-insert-key}) to
+insert the next key sequence that you type into the buffer using the
+correct format, similar to @kbd{C-q} (@code{quoted-insert}). Use
+@kbd{C-c C-r} (@code{edmacro-set-macro-to-region-lines}) to replace
+the macro's text with the text in the region. If the region does not
+begin at the start of a line or if it does not end at the end of a
+line, the region is extended to include complete lines. If the region
+ends at the beginning of a line, that final line is excluded.
+
@findex edit-kbd-macro
@kindex C-x C-k e
You can edit a named keyboard macro or a macro bound to a key by typing
@kbd{M-x @var{name}} or some other key sequence.
@findex kmacro-edit-lossage
+@vindex edmacro-reverse-macro-lines
@kindex C-x C-k l
You can edit the last 300 keystrokes as a macro by typing
-@kbd{C-x C-k l} (@code{kmacro-edit-lossage}).
+@kbd{C-x C-k l} (@code{kmacro-edit-lossage}). By default,
+your most recent keystrokes are listed at the bottom of the buffer.
+To list a macro's key sequences in reverse order, set
+@code{edmacro-reverse-macro-lines} to @code{t}.
@node Keyboard Macro Step-Edit
@section Stepwise Editing a Keyboard Macro
:type 'boolean
:group 'kmacro)
+(defcustom edmacro-reverse-macro-lines nil
+ "Non-nil if `edit-kbd-macro' should show the most recent line of key sequences first.
+
+This is useful when dealing with long lists of key sequences, such as
+from `kmacro-edit-lossage'."
+ :type 'boolean
+ :group 'kmacro
+ :version "30.1")
+
(defvar-keymap edmacro-mode-map
"C-c C-c" #'edmacro-finish-edit
- "C-c C-q" #'edmacro-insert-key)
+ "C-c C-q" #'edmacro-insert-key
+ "C-c C-r" #'edmacro-set-macro-to-region-lines)
(defface edmacro-label
'((default :inherit bold)
:group 'kmacro)
(defvar edmacro-mode-font-lock-keywords
- `((,(rx bol (group (or "Command" "Key" "Macro") ":")) 0 'edmacro-label)
+ `((,(rx bol (group (or "Command" "Key"
+ (seq "Macro" (zero-or-one " (most recent line first)")))
+ ":"))
+ 0 'edmacro-label)
(,(rx bol
(group ";; Keyboard Macro Editor. Press ")
(group (*? nonl))
(let* ((oldbuf (current-buffer))
(mmac (edmacro-fix-menu-commands mac))
(fmt (edmacro-format-keys mmac 1))
- (fmtv (edmacro-format-keys mmac (not prefix)))
+ (fmtv (let ((fmtv (edmacro-format-keys mmac (not prefix))))
+ (if (not edmacro-reverse-macro-lines)
+ fmtv
+ (with-temp-buffer
+ (insert fmtv)
+ (reverse-region (point-min) (point-max))
+ (buffer-string)))))
(buf (get-buffer-create "*Edit Macro*")))
(message "Formatting keyboard macro...done")
(switch-to-buffer buf)
(setq-local font-lock-defaults
'(edmacro-mode-font-lock-keywords nil nil ((?\" . "w"))))
(setq font-lock-multiline nil)
+ ;; Make buffer-local so that the commands still work
+ ;; even if the default value changes.
+ (make-local-variable 'edmacro-reverse-macro-lines)
(erase-buffer)
(insert (substitute-command-keys
(concat
(insert "Key: none\n")))
(when (and mac-counter mac-format)
(insert (format "Counter: %d\nFormat: \"%s\"\n" mac-counter mac-format))))
- (insert "\nMacro:\n\n")
+ (insert (format "\nMacro%s:\n\n" (if edmacro-reverse-macro-lines
+ " (most recent line first)"
+ "")))
(save-excursion
(insert fmtv "\n"))
(recenter '(4))
\f
;;; Commands for *Edit Macro* buffer.
+(defvar edmacro--skip-line-regexp
+ "[ \t]*\\($\\|;;\\|REM[ \t\n]\\)"
+ "A regexp identifying lines that should be ignored.")
+
+(defvar edmacro--command-line-regexp
+ "Command:[ \t]*\\([^ \t\n]*\\)[ \t]*$"
+ "A regexp identifying the line containing the command name.")
+
+(defvar edmacro--key-line-regexp
+ "Key:\\(.*\\)$"
+ "A regexp identifying the line containing the bound key sequence.")
+
+(defvar edmacro--counter-line-regexp
+ "Counter:[ \t]*\\([^ \t\n]*\\)[ \t]*$"
+ "A regexp identifying the line containing the counter value.")
+
+(defvar edmacro--format-line-regexp
+ "Format:[ \t]*\"\\([^\n]*\\)\"[ \t]*$"
+ "A regexp identifying the line containing the counter format.")
+
+(defvar edmacro--macro-lines-regexp
+ (rx "Macro"
+ (zero-or-one " (most recent line first)")
+ ":"
+ (zero-or-more (any " \t\n")))
+ "A regexp identifying the lines that precede the macro's contents.")
+
(defun edmacro-finish-edit ()
(interactive nil edmacro-mode)
(unless (eq major-mode 'edmacro-mode)
(top (point-min)))
(goto-char top)
(let ((case-fold-search nil))
- (while (cond ((looking-at "[ \t]*\\($\\|;;\\|REM[ \t\n]\\)")
+ (while (cond ((looking-at edmacro--skip-line-regexp)
t)
- ((looking-at "Command:[ \t]*\\([^ \t\n]*\\)[ \t]*$")
+ ((looking-at edmacro--command-line-regexp)
(when edmacro-store-hook
(error "\"Command\" line not allowed in this context"))
(let ((str (match-string 1)))
cmd)))
(keyboard-quit))))
t)
- ((looking-at "Key:\\(.*\\)$")
+ ((looking-at edmacro--key-line-regexp)
(when edmacro-store-hook
(error "\"Key\" line not allowed in this context"))
(let ((key (kbd (match-string 1))))
(edmacro-format-keys key 1))))
(keyboard-quit))))))
t)
- ((looking-at "Counter:[ \t]*\\([^ \t\n]*\\)[ \t]*$")
+ ((looking-at edmacro--counter-line-regexp)
(when edmacro-store-hook
(error "\"Counter\" line not allowed in this context"))
(let ((str (match-string 1)))
(unless (equal str "")
(setq mac-counter (string-to-number str))))
t)
- ((looking-at "Format:[ \t]*\"\\([^\n]*\\)\"[ \t]*$")
+ ((looking-at edmacro--format-line-regexp)
(when edmacro-store-hook
(error "\"Format\" line not allowed in this context"))
(let ((str (match-string 1)))
(unless (equal str "")
(setq mac-format str)))
t)
- ((looking-at "Macro:[ \t\n]*")
+ ((looking-at edmacro--macro-lines-regexp)
(goto-char (match-end 0))
nil)
((eobp) nil)
(when (buffer-name obuf)
(set-buffer obuf))
(message "Compiling keyboard macro...")
- (let ((mac (edmacro-parse-keys str)))
+ (let ((mac (edmacro-parse-keys (if edmacro-reverse-macro-lines
+ (with-temp-buffer
+ (insert str)
+ (reverse-region (point-min)
+ (point-max))
+ (buffer-string))
+ str))))
(message "Compiling keyboard macro...done")
(if store-hook
(funcall store-hook mac)
(insert (edmacro-format-keys key t) "\n")
(insert (edmacro-format-keys key) " ")))
+(defun edmacro-set-macro-to-region-lines (beg end)
+ "Set the macro text to lines of text in the buffer between BEG and END.
+
+Interactively, BEG and END are the beginning and end of the
+region. If the region does not begin at the start of a line or
+if it does not end at the end of a line, the region is extended
+to include complete lines. If the region ends at the beginning
+of a line, that final line is excluded."
+ (interactive "*r" edmacro-mode)
+ ;; Use `save-excursion' to restore region if there are any errors.
+ ;; If there are no errors, update the macro text, then go to the
+ ;; beginning of the macro text.
+ (let ((final-position))
+ (save-excursion
+ (goto-char beg)
+ (unless (bolp) (setq beg (pos-bol)))
+ (goto-char end)
+ (unless (or (bolp) (eolp)) (setq end (pos-eol)))
+ (let ((text (buffer-substring beg end)))
+ (goto-char (point-min))
+ (if (not (let ((case-fold-search nil))
+ (re-search-forward edmacro--macro-lines-regexp nil t)))
+ (user-error "\"Macro:\" line not found")
+ (delete-region (match-end 0)
+ (point-max))
+ (goto-char (point-max))
+ (insert text)
+ (setq final-position (match-beginning 0)))))
+ (goto-char final-position)))
+
(defun edmacro-mode ()
"\\<edmacro-mode-map>Keyboard Macro Editing mode. Press \
\\[edmacro-finish-edit] to save and exit.
You can edit these lines to change the places where the new macro
is stored.
+Press \\[edmacro-set-macro-to-region-lines] to replace the text following the \"Macro:\" line
+with the text of the lines overlapping the region of text between
+point and mark. If that region ends at the beginning of a line,
+that final line is excluded.
Format of keyboard macros during editing: