Vim like no one else
Nothing beats the dissatisfaction of not being able to get faster at something you enjoy doing. Knowing very well that increasing your pace will drastically improve your competence. That was me before I experienced the magic of using Vim motions. And now, almost a month later, I can’t live without it. The journey of becoming a Vim wizard comes at a cost, apparently.
If you’re not already aware, Vim is a text-editor that rests on “the keyboard is all you need” philosophy. It employs an ingenious technique (referred to as motions) which allows users to move around and edit a block of text without the need of a mouse.
It is not compulsory for you to use Vim to learn Vim motions. You can use any editor of your choice that provides support for a Vim plugin. If you’re like me and love using VSCode, you’d need to install the Vim extension.
Before we get started, I feel compelled to emphasize the importance of touch typing in this context. It is an invaluable skill for people like me who spend hours fiddling with computers on a daily basis. While touch typing allows you to type at the speed you think, Vim empowers you to edit at the same speed. I do not intend this to be a Vim tutorial but rather a guide to understand the intuition behind Vim motions. Hence, I will not be going over commands or features that do not contribute towards our objective. And yes, that also includes exiting the editor.
Vim may not be productive for you if you do not touch type. A strong muscle memory is crucial for building up speed and effectiveness.
Modes
Unlike other editors, Vim has modes, which are perspectives that segregate the actions of editing and text input, with the ability to switch between them as needed. The most important ones are -
-
Normal Mode
This is the default mode in Vim, which enables the user to navigate through the document, primarily by using the home row keys of
j
,k
h
andl
for moving down, up, left and right respectively. Other keys such asw
(alsoe
) orb
for moving forward and backwards by a word,{
and}
for moving up and down empty lines, etc. make it super easy to jump from one line to another (These are called movements and are discussed later). This mode is special since other editing modes can be entered only through the Normal Mode. TheEsc
key is reserved by default to enter Normal Mode from other modes. -
Insert Mode
The Insert Mode is where Vim behaves like any other text editor by allowing us to input and manipulate text. We can enter it by pressing
i
(insert to the left of current position) ora
(insert to the right of current position). -
Visual Mode
To remove the need of selecting a block of text by holding down the mouse, Vim has Visual Mode. It essentially has two important sub-modes, the Character Visual Mode (
v
) and the Line Visual Mode (V
). The former allows the flexibility to select every character starting from the current cursor position while the latter selects each line individually. Theo
key allows our cursor to move from one end of the selected block to the other. -
Command Mode
This mode is reserved for entering commands in the Vim editor’s command line. It can be entered by pressing
:
or/
to perform certain tasks like saving a file (:w
) or performing search (/word
) and replace (:%s/old/new
) operations. Vim boasts plenty of other commands which are not very useful if you’re not working on the Vim editor itself.
Additional modes like Replace or Terminal also exist, but they’ve been of little practical use to me as far as Vim motions are concerned.
The Grammar
Motions started to make way more sense once I realized they could be deconstructed as lexical items in a human language. Vim is based on a meticulous system of nouns, verbs and movements that combine together to create motions, much like sentences. Let’s explore this further to grasp their incredible power.
Movements
Movements, in simple terms, are the keys that make the cursor move in Normal Mode. These include the
home-row keys (j
, k
, h
, l
), word-movement keys (w
, e
, b
), line-movement keys -
0
- move to start of line$
- move to the end of linegg
- move to the first line of fileG
- move to the last line of file(
- move backwards to the start of each sentence)
- move forwards to the start of each sentence%
- move to the opposite matching brace
and 4 special movement keys, which I like to call c-moves -
t
- (till character)f
- (to character)T
- (till character backwards)F
- (to character backwards)
These c-moves require a second argument as their target character.
(c-move)(character)
For example, ti
moves the cursor forward to the position before the first occurrence of the character
i
in the current line. Similarly, Fg
moves the cursor backwards to the position of the first occurrence of the character
g
in the current line.
Nouns and Verbs
Nouns are targets (words, paragraphs, etc.) against which we perform a particular action (verb), like copying/deleting or selecting text. They include -
w
- (word)p
- (paragraph)s
- (sentence)(
- (also b for parentheses){
- (also B for curly braces)"
- (double quotes)[
- (square brackets)`
- (back ticks)t
- (HTML tags)
with the verbs being -
d
- (delete)c
- (change)y
- (yank or copy)v
- (select)
The delete and change verbs have only one distinction. Delete removes a noun and preserves the Normal mode while change removes a Noun and enters Insert Mode.
Motions
We can combine a verb and a movement to produce a motion.
For example, dw
(delete a word when cursor is in the beginning) becomes a verb + movement motion.
It is important to note that in this case, w
is the word-movement and not the word-noun.
We can substitute any verb with a movement to carry out an action. We can also include a number called count
after the verb to specify the length of characters or lines that will be modified by the action. For example,
d4j
deletes 4 lines down from the current cursor position, since j
is the downwards
movement key.
(verb){count}(movement)
C-moves help create more precise motions. They can be implemented in the following structure.
(verb){count}(c-move)(character)
dt.
- delete everything from current cursor position till the first occurrence of a.
vfo
- select everything from current cursor position to the first occurrence of ao
yF-
- copy/yank everything backwards from current cursor position to the first occurrence of a-
d3fy
- delete everything from current cursor position till the first 3 occurrences ofy
Nouns do not produce motions with verbs. This is because nouns are a little special and require the addition of, what I like to call a wrap-specifier, to help define our motion accurately.
Compounding motions using wrap-specifiers
Good things come in pairs, so do wrap-specifiers.
i
- (inside)a
- (around)
The i
specifier is used to instruct the verb to perform the motion inside the noun, while
a
is used to perform the motion around it. It usually follows the given structure.
(verb){count}(specifier)(noun)
Here are a few examples.
daw
- deletes the current word from any cursor position along with the adjacent space, hence around.ciw
- changes (deletes and inserts) the current word from any cursor position without removing the adjacent space, hence inside.dab
- deletes around a pair of parentheses (b also means ()), resulting in the removal of everything inside the braces along with the braces itself.yip
- yanks/copies inside the current paragraph, resulting in a block of text without any surrounding line-breaks.v3as
- selects around three sentences to the right including the current sentence and a trailing space.
That’s pretty much all we need to know to construct complex motions of our own.
Macros
Macros are a lifesaver, when implemented correctly. They consist of a pre-recorded set of keystrokes that can be used to perform repetitive tasks. Like appending a specific phrase on every line or inserting line-breaks after a definite interval of characters. It works by mapping a key to the recorded motion which can be called later and reused throughout the document.
To start recording a macro (in Normal Mode), we press the q
key followed by any key of our choice (say e
) as
the key we want to map our macro to. Once we execute that, we will notice
Recording @e
on the command line of our editor. From this point onwards, every keystroke we make will be recorded as a
macro-move till we press the q
key again to end the recording. To utilize or call this macro, we need to
press the @
key followed by the key we mapped our macro to, in our case e
. This will modify the document
from our current cursor position in the order of the recorded keystrokes.
Tricks
Here are my most used motions, in no particular order.
-
There are two commands that are indispensable.
daw
andciw
. They both do the same job, which is deleting a word, but in slightly different ways. -
I
andA
. They move your cursor to the beginning and end of the current line respectively. Quick and extremely useful. -
If you need to jump to a word situated a few words (or lines) after your cursor, you can
/word
and hit enter to teleport to the first match. If there are multiple matches, pressingn
will cycle between them. The equivalent for jumping to word that lies before the cursor is?word
. -
cc
to delete and insert in the current line. -
xp
quickly swaps the current character with the next character. Great for turningblaot
tobloat
without leaving normal mode. -
bi
to insert in front of the current word andea
to insert behind it. Priceless moves. -
To replace a character without having to go to insert mode, press
r
over the character you want to replace followed by the new character. -
Now this is exclusively for a programming scenario. If you’re on a variable and need to find the line where it was declared, press
gd
over the word and your cursor will instantly jump to the desired line. -
When the cursor is on an opening brace,
V%
(alsoy%
,d%
) will select the entire block. I use it frequently for selecting an entire function. -
Marking is easily the coolest feature in Vim. It works a bit like macros by mapping a character to a particular line which you can call later and jump to. Let me explain this better. Say you find yourself visiting line 481 quite frequently. Instead of having to type
:481
on the command line to jump each time, you can instead mark the line using the commandm(character)
against a character of your choice, saya
. Now whenever you need to jump to line 481, all you need to do is'(character)
, or in our case'a
from anywhere on the document. Isn’t that stupidly simple?
I compiled a non-exhaustive list of useful Vim commands in this Github Repository. Also, here’s an awesome twitter page for you.
You can check them out if you like :)
Conclusion
Vim motions have been unreasonably impactful in the way I write code now. The only downside of Vim-ing hard is it induces a subtle restlessness in your fingers once you start getting the hang of it. Something that doesn’t wear off easily. At least, not when you’re consistently typing over a hundred words every day. Perhaps slowing down will save you the dread of losing your phalanges over time, or maybe it’s just me. I haven’t a clue. But nevertheless, I’ve loved very minute I’ve Vim-med and there’s just no going back. Remember, it demands a good deal of practice before you find yourself whizzing through your code-base like a golden snitch. So get your knuckles cracking and Vim on!
Finished in my dorm room (8052), after a week of feverish anticipation for nothing