Line editing in Bash
You probably know the GNU Bash. Many people would say that it does not have many features or it is even user-unfriendly. I've already tried other POSIX-like shells, but I still stick with my favourite GNU Bash. (I think that fish is quite a good shell, but lacks flexibility of configuration that Bash does have.)
Bash uses the readline library to provide the line-editing features. Readline provides editing of the text you type in, history etc. It has many features, but a lot of them is hidden behind obscure keyboard shortcuts or configuration options.
This page tries to provide an overview of features provided by the readline, especially useful in the Bash. I attempted to order the sections of this page from the most basic operations to more advanced and obscure features.
The listing of the shortcuts is not complete. Refer to readline and Bash manuals for complete lists of the shortcuts.
Few notes
Keyboard shortcuts
We will use three modifier keys: shift, control and alt (also called Meta). I will follow Emacs conventions of keyboard shortcut notation, since this notation is also used by the readline docs and the keyboard shortcuts are significantly influenced by the Emacs. Do not expect CUA-like keyboard shortcuts you know from Windows or many other contemporary desktop environments. The keyboard shortcuts were designed before any common convention was formed.
Notation of keyboard shortcuts I will use:
Shortcut | Description |
---|---|
a
|
A single key “a”, without any modifier keys or Caps Lock. |
C-a or ^A
|
Control-a (case-insensitive) |
M-A
|
Meta-a (Alt-a) |
A
|
Uppercase “A” (Shift-a) |
C-A
|
Control-Shift-A (assuming disabled Caps Lock) |
[
|
That key that sends character “[” on your keyboard layout |
^[
|
Escape |
C-a a
|
Control-a followed by letter “a” |
Also note that the keyboard shortcuts are processed on level of text, not keycodes or even scancodes. This means that your local keyboard mapping and Caps Lock state is applied before the Readline processes the keyboard shortcuts. Also, Control performs bitwise and operation with number 0x1f on the character code, therefore all shortcuts with Control are case-insensitive.
The Meta (Alt) key is usually implemented by prefixing all individual key codes with escape character (^[
, 0x1b). When you do not have Alt key or your terminal emulator is broken, you can prefix each of the shortcuts manually with a press of the Escape key. This is useful when using terminal emulators on Android devices, where support of the Alt key is frequently broken or half-functioning.
All keys that do not have a place in the ASCII character set and do not produce a text characters produce some terminal escape sequences. I will call such keys keypad keys, for historical reasons – you can frequently find this terminology in documentation of terminal-based software.
Emacs key mnemonics
Many of keyboard shortcuts inherited from Emacs can be remembered by a letter (usually the initial one) in word describing action performed by the key. I will try to emphasize this by underlining some letters in the description of the keys.
Adding Meta (Alt) to the shortcuts usually means operation on words instead of characters.
Flow control characters
By default, you will be unable to use any shortcuts that contain ^Q
or ^S
, because these work as a flow control characters. Pressing ^S
stops the terminal output (but your input is still processed!). Press ^Q
to restore the terminal output.
Line-editing features
Cursor movement
Group | Keypad key | … that sends sequence | Shortcut | Action |
---|---|---|---|---|
History navigation | ↑
|
^[A
|
C-p
|
Previous item in the history |
↓
|
^[B
|
C-n
|
Next item in the history | |
C-r
|
Reverse iterative search in history | |||
Character movement | ←
|
^[D
|
C-b
|
Back, one character |
→
|
^[C
|
C-f
|
Forward, one character | |
Word movement | M-←
|
^[[1;3D
|
M-b
|
Back, one word |
M-→
|
^[[1;3C
|
M-f
|
Forward, one word | |
Line movement | Home
|
^[[H
|
C-a
|
Move to the line start |
End
|
^[[F
|
C-e
|
Move to the line end |
Press C-r
to enter iterative search mode. Type a substring of a command you want to edit. Press ^R
again to search for the previous item. Exit by pressing any key that does not type text.
Note that the history editing edits the history in-place. When you move to another history item and return to the item that you have edited, the edits will remain there.
In theory, C-s
will perform a forward iterative search. That means that you will move to later items in the history. But since the terminal is usually configured to interpret ^S
as a flow control character, it will stop the terminal output and the shell could appear “broken” in eyes of novice users.
Arbitrary character insertion
Press ^V
followed by any key. The key code will be inserted literally, without interpreting it as a shortcut. You can use this to insert control characters, such as Escape.
Example
Try running this command, but instead of each ^[
press C-v Escape
(and keep the other [
s without modification):
echo "^[[31mUwU^[[0m"
After executing this, a red “UwU” should appear in your terminal. See also Terminal escape sequences.
History replay
When you move to some command in your history and press C-o
, the command will get executed and when it ends, the Bash will pre-enter the next command from the history. When you reach the original end of the history, the first command will be pre-entered in the prompt again.
Example
Run this first:
echo hello echo people echo there
Then press ↑
or C-p
three times.
After that, press C-o
repeatedly. These three commands will be executed in the same order, one command at each press of C-o
.
Real world example
This is useful when you want to build and run your program. The commands executed in the “loop” could be these two:
make -j4 ./my-program
Alternative keys to run a command
Pressing C-m
sends Return (\r
in C parlance) and C-j
sends Line Feed (\n
). Both of them can be used to run the command you have typed so far.
This is useful when the Return (Enter) key is too far to be reached easily. ☺
Stashing commands
I sometimes type some command, but I realize that I need to run another command before it. Pressing M-#
(Alt-Shift-3
on common US English layout) places a “#” at the start of the command and stores it in the history.
Insert last word of the previous command
Shortcut M-.
(Alt with a dot) inserts the last word from the previous command. Pressing it repeatedly iterates over the previous commands.
Example
mkdir foo cd (press M-. here)
Repetition and numeric arguments
Hold down Alt and type a number. The next non-numeric key or shortcut will be repeated that number of times.
Holding Alt only before the first number is enough. You can also press Escape and then type a number.
Note: This is also used to pass a numeric arguments to some commands.
Killing and yanking (i.e. clipboard in Bash)
Bash (in reality the Readline library) as built-in features for something similar to clipboard on modern desktop systems. Similarly to Emacs, the cut and delete operations are conflated into one (“kill”) operation. Place the cursor at a convenient point and try one of these:
Shortcut | Action |
---|---|
C-y
|
Yank (“paste”) |
C-k
|
Kill (“cut”) everything after the cursor |
C-u
|
Kill everything before the cursor |
M-d
|
Delete (kill) a single word to the right from the cursor |
C-M-y
|
Yank the second word from the last command in the history
(With numerical argument, it yanks the n-th word, where the command name is word zero.) |