From: Drew Adams <drew.adams@oracle.com>
To: Christopher Dimech <dimech@gmx.com>, Yuri Khan <yuri.v.khan@gmail.com>
Cc: help-gnu-emacs <help-gnu-emacs@gnu.org>
Subject: repeatable keys, prefix keys [was: Emacs keybindings according to file type]
Date: Wed, 23 Sep 2020 10:04:03 -0700 (PDT) [thread overview]
Message-ID: <680e46d5-de32-4776-a258-201aa6fe9e2c@default> (raw)
> For instance one can make it to continue pressing
> C-t l f f f to continue moving line downward.
>
> Any sample code to try out?
Sure:
(defun repeat-command (command)
"Repeat COMMAND."
(require 'repeat) ; Define its vars before we let-bind them.
(let ((repeat-previous-repeated-command command)
(repeat-message-function #'ignore)
(last-repeatable-command 'repeat))
(repeat nil)))
You can use it with nearly any COMMAND. E.g., to make
a repeatable version of command `other-window':
(defun other-window-repeat ()
"Same as `other-window', but repeatable even on a prefix key.
E.g., if bound to `C-x o' then you can use `C-x o o o...' to repeat."
(interactive)
(repeat-command 'other-window))
(global-set-key [remap other-window] 'other-window-repeat)
____
I do this all over the place, where it makes sense.
Keyboard keys are scarce. When Emacs decides to bind
a key by default - a key that wasn't bound by default
before, I often raise these two points:
1. Save naturally repeatable keys for, well, commands
that can be repeated, i.e., commands for which it
makes sense to be able to just hold down a key to
repeat them.
2. Save some keys for prefix keys, as opposed to just
sacrificing them for a single command. A prefix
key essentially gives you a whole keyboard of
possibilities for a single key. Think of all the
mileage we get out of the prefix key `C-x'! Of
course, adding a prefix key makes a key sequence
longer, more complex. Tradeoff.
I tend to define lots of keys for features I write,
and put them, by groups, on their own keymaps, and
then put those keymaps on prefix keys. Even if such a
prefix key appears complicated or slow, the fact of
using a separate keymap means that a user can easily
put it on a different, shorter keymap, or remap it to
a more global keymap.
E.g., in Bookmark+ there are all of these keymaps on
prefix keys by default:
bookmark-map C-x x
bmkp-annotate-map C-x x a
bmkp-set-map C-x x c ("create")
bmkp-tags-map C-x x t
bmkp-jump-map C-x j
bmkp-jump-other-window-map C-x 4 j
And it goes on from there. `C-x j t' is a prefix key
for jumping to bookmarks with particular tags.
`C-x j t *' and `C-x j t +' are prefix keys for
jumping to bookmarks with all (*), or some (+), of a
given set of tags.
`C-x j t % *' and `C-x j t % +' are prefix keys for
jumping to bookmarks with tags, all, or some, of which
match a given regexp. `%' for regexp, `*' for all
(Boolean multiplication), `+' for some (Boolean
addition).
And so on. These bindings are mnemonic, but more
importantly, they're grouped so you can easily move a
group to a different prefix key or a global keymap
instead.
Without using prefix keys, there's no way I could
provide key bindings for so many commands, let alone
do so in an organized way (discoverable, mnemonic).
____
Now imagine that keys aren't reserved by Emacs this
way - repeatable keys for repeatable commands, and
some keys available to be used as prefix keys.
Imagine if Emacs just predefined `C-x' for a single
command (e.g. `cut'). Zillions of keys bound to
keymaps under `C-x' would be sacrificed.
At the very least, when a new key is decided to be
sacrificed by default (vanilla Emacs), it had better
be bound to a repeatable command, not one (such as
`cut') that it makes no sense to repeat. And even a
repeatable key is a sacrifice - consider if `C-x' were
bound to, say, `forward-word'. Repeatable, yes, but
think of all the bindings now under `C-x' that would
be sacrificed.
Keyboard keys are precious - scarce. Too many have
already been sacrificed to default bindings, I think.
Sure, any user or library can redefine any keys. But
once blessed as a default vanilla-Emacs key binding, a
key is, for practical purposes, kinda off limits for a
library.
For example, for years I had `bookmark-map' on prefix
`C-x p'. I had to change that recently to `C-x x'
because Emacs decided to co-opt `C-x p' (& `C-x 4 p')
for keys for library `project.el'. (Luckily `C-x x'
was still available.)
And for years I've had `Do Re Mi' commands on prefix
key `C-x t'. I'll now have to change that to some
other prefix key, as Emacs has decided to co-opt that
prefix for `tab-bar-mode' keys.
That's fine. At least Emacs is doing the right thing
by using prefix keys (and not top-level keys, which
are typically repeatable, but keys under `C-x').
The point is that (1) it's easy to move a keymap from
one prefix key to another, and (2) there need to be
some prefix keys available to move maps to.
Eventually, I imagine that some simple, repeatable
keys that have been assigned default Emacs bindings
for commands that aren't repeatable, or that aren't
super useful, or that don't really need a single-key
binding, will be recycled and put to better use: for
repeatable commands, as prefix keys, or just unbound
by default and left available for libraries.
No, I don't have particular suggestions, and no, it's
not urgent.
But consider `M-!', for example. Sure, `!' is
mnemonic for shell. But `M-!' is repeatable (just
hold it down), and it makes no sense to repeat
`shell-command'. There are other default key bindings
like this - essentially wasted.
Some commands bound to repeatable keys should be
replaced by repeatable versions. I do that for `C-a'
and `C-e', for example, so they're similar to `C-n'
and `C-p' (repeatable). That kind of change is
minimal - it's the least we can do to make things a
little saner. And `repeat-command' lets you do this
kind of thing easily.
Take a look at `C-h b', and see which repeatable keys
are bound to non-repeatable commands. Not too many,
but there are some.
Even `C-w' is "wasted" on a non-repeatable command.
Am I suggesting `C-w' should not be bound by default
to `kill-region'? Not really. That's not urgent, at
least. But hey, keys are limited - the keyboard's a
small planet. ;-)
And `beginning-of-buffer'. That non-repeatable
command's bound by default to two repeatable keys,
`M-<' and `C-home'. Both mnemonic (good). But...
reply other threads:[~2020-09-23 17:04 UTC|newest]
Thread overview: [no followups] expand[flat|nested] mbox.gz Atom feed
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=680e46d5-de32-4776-a258-201aa6fe9e2c@default \
--to=drew.adams@oracle.com \
--cc=dimech@gmx.com \
--cc=help-gnu-emacs@gnu.org \
--cc=yuri.v.khan@gmail.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
Code repositories for project(s) associated with this external index
https://git.savannah.gnu.org/cgit/emacs.git
https://git.savannah.gnu.org/cgit/emacs/org-mode.git
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.