You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
The pencil plugin aspires to make Vim as powerful a tool for writers as it is for coders by focusing narrowly on the handful of tweaks needed to smooth the path to writing prose.
For editing prose-oriented file types such as text, markdown, mail, rst, tex, textile, and asciidoc
Agnostic on soft line wrap versus hard line breaks, supporting both
Auto-detects wrap mode via modeline and sampling
Adjusts navigation key mappings to suit the wrap mode
Creates undo points on common punctuation during Insert mode, including deletion via line <C-U> and word <C-W>
Buffer-scoped configuration (with a few minor exceptions, pencil preserves your global settings)
Support for Vim’s Conceal feature to hide markup defined by Syntax plugins (e.g., _ and * markup for styled text in Markdown)
Support for display of mode indicator (␍ and ⤸, e.g.) in the status line
Pure Vimscript with no dependencies
In addition, when using hard line break mode:
Makes use of Vim’s powerful autoformat while inserting text, except for tables and code blocks where you won’t want it.
NEW Optional key mapping to suspend autoformat for the Insert.
Need spell-check, distraction-free editing, and other features? Vim is about customization. To complete your editing environment, learn to configure Vim and draw upon its rich ecosystem of plugins.
Why use Vim for writing?
With plenty of word processing applications available, including those that specifically cater to writers, why use a modal editor like Vim? Several reasons have been offered:
Your hands can rest in a neutral ‘home’ position, only rarely straying to reach for mouse, track pad, or arrow keys
Minimal chording, with many mnemonic-friendly commands
Sophisticated capabilities for navigating and manipulating text
Highly configurable, enabling you to build a workflow that suits your needs, with many great plugins available
No proprietary format lock-in
But while such reasons might be sound, they remain scant justification to switch away from the familiar word processor. Instead, you need a compelling reason—one that can appeal to a writer’s love for language and the tools of writing.
You can find that reason in Vim’s mysterious command sequences. Take cas for instance. You might see it as a mnemonic for Change Around Sentence to replace an existing sentence. But dig a bit deeper to discover that such commands have a grammar of their own, comprised of nouns, verbs, and modifiers. Think of them as the composable building blocks of a domain specific language for manipulating text, one that can become a powerful tool in expressing yourself. For more details on vi-style editing, see...
For those new to Vim: before installing this plugin, consider getting comfortable with the basics of Vim by working through one of the many tutorials available.
You can manually enable, disable, and toggle pencil as a command:
Pencil - initialize pencil with auto-detect for the current buffer
NoPencil (or PencilOff) - removes navigation mappings and restores buffer to global settings
TogglePencil (or PencilToggle) - if on, turns off; if off, initializes with auto-detect
Because auto-detect might not work as intended, you can invoke a command to set the behavior for the current buffer:
SoftPencil (or PencilSoft) - initialize pencil with soft line wrap mode
HardPencil (or PencilHard) - initialize pencil with hard line break mode (and Vim’s autoformat)
Initializing by file type
Initializing pencil by file type is optional, though doing so will automatically set up your buffers for editing prose.
Add support for your desired file types to your .vimrc:
setnocompatiblefiletypepluginon" may already be in your .vimrcaugrouppencilautocmd!autocmdFileTypemarkdown,mkdcallpencil#init()
autocmdFileTypetextcallpencil#init()
augroupEND
You can initialize several prose-oriented plugins together:
For a list of other prose-oriented plugins, consult the See also section below.
Hard line breaks or soft line wrap?
Coders will have the most experience with the former, and writers the latter. But whatever your background, chances are that you must contend with both conventions. This plugin doesn’t force you to choose a side—you can configure each buffer independently.
In most cases you can set a default to suit your preference and let auto-detection figure out what to do.
letg:pencil#wrapModeDefault ='soft'" default is 'hard'augrouppencilautocmd!autocmdFileTypemarkdown,mkdcallpencil#init()
autocmdFileTypetextcallpencil#init({'wrap': 'hard'})
augroupEND
In the example above, for buffers of type markdown this plugin will auto-detect the line wrap approach, with soft line wrap as the default.
For buffers of type text, it will initialize with hard line breaks, even if auto-detect might suggest soft line wrap.
Automatic formatting
The ‘autoformat’ feature affects HardPencil (hard line break) mode only.
When inserting text while in HardPencil mode, Vim’s powerful autoformat feature will be enabled by default and can offer many of the same benefits as soft line wrap.
...where buffers of type markdown and text will use hard line breaks, but text buffers will have autoformat disabled.
Suspend automatic formatting for the Insert
There are two useful exceptions where autoformat (when enabled for the buffer) will be temporarily disabled for the current Insert:
First is pencil’s ‘blacklisting’ feature: if used with popular prose-oriented syntax plugins, pencil will suspend autoformat when you enter Insert mode from inside a code block or table.
Second, where blacklisting falls short, you can optionally map a buffer-scoped ‘modifier’ key to suspend autoformat during the next Insert:
letg:pencil#map#suspend_af = 'K' " default is no mapping
Using the above mapping, with Ko you’ll enter Insert mode with the cursor on a new line, but autoformat will suspend for that Insert. Using o by itself will retain autoformat.
By default no modifier key is mapped.
(See the advanced section below for details on how blacklisting is implemented and configured).
Manual formatting
Note that you need not rely on Vim’s autoformat exclusively and can manually reformat paragraphs with standard Vim commands:
gqap - format current paragraph (see :help gq for details)
vapJgqap - merge two paragraphs (current and next) and format
ggVGgq or :g/^/norm gqq - format all paragraphs in buffer
Optionally, you can map these operations to underutilized keys in your .vimrc:
Or you may wish to ‘unformat’, (i.e., remove hard line breaks) when using soft line wrap.
vipJ - join all lines in current paragraph
:%norm vipJ - unformat all paragraphs in buffer
Default textwidth
You can configure the textwidth to be used in HardPencil (hard line break) mode when no textwidth is set globally, locally, or available via modeline. It defaults to 74, but you can change that value in your .vimrc:
letg:pencil#textwidth=74
Sentence spacing
By default, when formatting text (through gwip, e.g.) only one space will be inserted after a period(.), exclamation point(!), or question mark(?). You can change this default:
By default, h/l and the left/right cursor keys will move to the previous/next line after reaching first/last character in a line with a hard break. If you wish to retain the default Vim behavior, set the cursorwrap value to 0 in your .vimrc:
Syntax plugins such as tpope/vim-markdown support concealing the markup characters when displaying italic, bold, and ***bold italic*** styled text.
To use Vim’s Conceal feature with Markdown, you will need to install:
tpope/vim-markdown as it’s currently the only Markdown syntax plugin that supports conceal.
A monospaced font (such as Cousine) featuring the italic, bold, and bold italic style variant for styled text.
A colorscheme (such as reedes/vim-colors-pencil) which supports the Markdown-specific highlight groups for styled text.
You should then only see the _ and * markup for the cursor line and in visual selections.
Terminal users: consult your terminal’s documentation to configure your terminal to support bold and italic styles.
Status line indicator
Your status line can reflect the wrap mode for pencil buffers. For example, ␍ to represent HardPencil (hard line break) mode. To configure your status line and ruler, add to your .vimrc:
The default indicators now include ‘auto’ for when Vim’s autoformat is active in hard line break mode. (If autoformat is suspended for the Insert, it’ll show the ‘hard’ indicator.)
Configurable options for pencil#init() include: autoformat, concealcursor, conceallevel, cursorwrap, joinspaces, textwidth, and wrap. These are detailed above.
Autoformat manual control
The ‘autoformat’ feature affects HardPencil (hard line break) mode only.
To suspend autoformat for the next Insert, see above.
When you need to manually enable/disable autoformat for the current buffer, you can do so with a command:
PFormat - enable autoformat for buffer (can still be disabled via blacklisting)
PFormatOff - disable autoformat for buffer
PFormatToggle - toggle to enable if disabled, etc.
You can map a key in your .vimrc to toggle Vim’s autoformat:
The ‘autoformat’ feature affects HardPencil (hard line break) mode only.
When editing formatted text, such as a table or code block, Vim’s autoformat will wreak havoc with the formatting. In these cases you will want autoformat suspended for the duration of the Insert.
When entering Insert mode, pencil will determine the highlight group at the cursor position. If that group has been blacklisted, pencil will suspend autoformat for the Insert. For example, if editing a buffer of type ‘markdown’, autoformat will be suspended if you invoke Insert mode from inside a markdownFencedCodeBlock highlight group.
Blacklists are now declared by file type. The default blacklists (and whitelists) are declared in the plugin/pencil.vim module. Here’s an excerpt showing the configuration for the ‘markdown’ file type:
The whitelist will override the blacklist and enable Vim’s autoformat if text that would normally be blacklisted doesn’t dominate the entire line. This allows autoformat to work with inline code and links.
Auto-detecting wrap mode
If you didn’t explicitly specify a wrap mode during initialization, pencil will attempt to detect it.
It will first look for a textwidth (or tw) specified in a modeline. Failing that, pencil will then sample lines from the start of the buffer.
Detect via modeline
Will the wrap mode be detected accurately? Maybe. But you can improve its chances by giving pencil an explicit hint.
At the bottom of this document is a odd-looking code:
<!-- vim: set tw=73 :-->
This is an optional ‘modeline’ that tells Vim to run the following command upon loading the file into a buffer:
:settextwidth=73
It tells pencil to assume hard line breaks, regardless of whether or not soft line wrap is the default editing mode for buffers of type ‘markdown’.
You explicitly specify soft wrap mode by specifying a textwidth of 0:
<!-- vim: set tw=0 :-->
Note that if the modelines feature is disabled (such as for security reasons) the textwidth will still be set by this plugin.
Detect via sampling
If no modeline with a textwidth is found, pencil will sample the initial lines from the buffer, looking for those excessively-long.
There are two settings you can add to your .vimrc to tweak this behavior.
The maximum number of lines to sample from the start of the buffer:
letg:pencil#softDetectSample =20
Set that value to 0 to disable detection via line sampling.
When the number of bytes on a sampled line per exceeds this next value, then pencil assumes soft line wrap.
letg:pencil#softDetectThreshold =130
If no such lines found, pencil falls back to the default wrap mode.
See also
To Vim - Writer and psychologist Ian Hocking on using Vim for writing
Thanks for the shoutout on the Distraction Free Writing in Vim post. Unfortunately, that post is no longer up at this time. I remedy that and respond here when that is done.
update: Re-posted that article so it doesn't return a 404 anymore.
vim-pencil
Features
The pencil plugin aspires to make Vim as powerful a tool for writers as it is for coders by focusing narrowly on the handful of tweaks needed to smooth the path to writing prose.
modeline
and sampling<C-U>
and word<C-W>
_
and*
markup for styled text in Markdown)␍
and⤸
, e.g.) in the status lineIn addition, when using hard line break mode:
Need spell-check, distraction-free editing, and other features? Vim is about customization. To complete your editing environment, learn to configure Vim and draw upon its rich ecosystem of plugins.
Why use Vim for writing?
With plenty of word processing applications available, including those that specifically cater to writers, why use a modal editor like Vim? Several reasons have been offered:
But while such reasons might be sound, they remain scant justification to switch away from the familiar word processor. Instead, you need a compelling reason—one that can appeal to a writer’s love for language and the tools of writing.
You can find that reason in Vim’s mysterious command sequences. Take
cas
for instance. You might see it as a mnemonic for Change Around Sentence to replace an existing sentence. But dig a bit deeper to discover that such commands have a grammar of their own, comprised of nouns, verbs, and modifiers. Think of them as the composable building blocks of a domain specific language for manipulating text, one that can become a powerful tool in expressing yourself. For more details on vi-style editing, see...Installation
pencil is best installed using a Vim package manager, such as Vundle, Plug, NeoBundle, or Pathogen.
For those new to Vim: before installing this plugin, consider getting comfortable with the basics of Vim by working through one of the many tutorials available.
Vundle
Add to your
.vimrc
and save:…then run the following in Vim:
Plug
Add to your
.vimrc
and save:Plug 'reedes/vim-pencil'
…then run the following in Vim:
NeoBundle
Add to your
.vimrc
and save:NeoBundle 'reedes/vim-pencil'
…then run the following in Vim:
Pathogen
Run the following in a terminal:
Configuration
Initializing by command
You can manually enable, disable, and toggle pencil as a command:
Pencil
- initialize pencil with auto-detect for the current bufferNoPencil
(orPencilOff
) - removes navigation mappings and restores buffer to global settingsTogglePencil
(orPencilToggle
) - if on, turns off; if off, initializes with auto-detectBecause auto-detect might not work as intended, you can invoke a command to set the behavior for the current buffer:
SoftPencil
(orPencilSoft
) - initialize pencil with soft line wrap modeHardPencil
(orPencilHard
) - initialize pencil with hard line break mode (and Vim’s autoformat)Initializing by file type
Initializing pencil by file type is optional, though doing so will automatically set up your buffers for editing prose.
Add support for your desired file types to your
.vimrc
:You can initialize several prose-oriented plugins together:
For a list of other prose-oriented plugins, consult the See also section below.
Hard line breaks or soft line wrap?
Coders will have the most experience with the former, and writers the latter. But whatever your background, chances are that you must contend with both conventions. This plugin doesn’t force you to choose a side—you can configure each buffer independently.
In most cases you can set a default to suit your preference and let auto-detection figure out what to do.
In the example above, for buffers of type
markdown
this plugin will auto-detect the line wrap approach, with soft line wrap as the default.For buffers of type
text
, it will initialize with hard line breaks, even if auto-detect might suggest soft line wrap.Automatic formatting
The ‘autoformat’ feature affects
HardPencil
(hard line break) mode only.When inserting text while in
HardPencil
mode, Vim’s powerful autoformat feature will be enabled by default and can offer many of the same benefits as soft line wrap.To set the default behavior in your
.vimrc
:You can override this default during initialization, as in:
...where buffers of type
markdown
andtext
will use hard line breaks, buttext
buffers will have autoformat disabled.Suspend automatic formatting for the Insert
There are two useful exceptions where autoformat (when enabled for the buffer) will be temporarily disabled for the current Insert:
First is pencil’s ‘blacklisting’ feature: if used with popular prose-oriented syntax plugins, pencil will suspend autoformat when you enter Insert mode from inside a code block or table.
Second, where blacklisting falls short, you can optionally map a buffer-scoped ‘modifier’ key to suspend autoformat during the next Insert:
Using the above mapping, with
Ko
you’ll enter Insert mode with the cursor on a new line, but autoformat will suspend for that Insert. Usingo
by itself will retain autoformat.By default no modifier key is mapped.
(See the advanced section below for details on how blacklisting is implemented and configured).
Manual formatting
Note that you need not rely on Vim’s autoformat exclusively and can manually reformat paragraphs with standard Vim commands:
gqap
- format current paragraph (see:help gq
for details)vapJgqap
- merge two paragraphs (current and next) and formatggVGgq
or:g/^/norm gqq
- format all paragraphs in bufferOptionally, you can map these operations to underutilized keys in your
.vimrc
:Or you may wish to ‘unformat’, (i.e., remove hard line breaks) when using soft line wrap.
vipJ
- join all lines in current paragraph:%norm vipJ
- unformat all paragraphs in bufferDefault textwidth
You can configure the textwidth to be used in
HardPencil
(hard line break) mode when no textwidth is set globally, locally, or available via modeline. It defaults to74
, but you can change that value in your.vimrc
:Sentence spacing
By default, when formatting text (through
gwip
, e.g.) only one space will be inserted after a period(.
), exclamation point(!
), or question mark(?
). You can change this default:Cursor wrap
By default,
h
/l
and the left/right cursor keys will move to the previous/next line after reaching first/last character in a line with a hard break. If you wish to retain the default Vim behavior, set thecursorwrap
value to0
in your.vimrc
:Concealing __markup__
pencil enables Vim’s powerful Conceal feature, although support among Syntax and Colorscheme plugins is currently spotty.
You can change pencil’s default settings for conceal in your
.vimrc
:For more details on Vim’s Conceal feature, see:
Concealing styled text in Markdown
Syntax plugins such as tpope/vim-markdown support concealing the markup characters when displaying italic, bold, and ***bold italic*** styled text.
To use Vim’s Conceal feature with Markdown, you will need to install:
tpope/vim-markdown as it’s currently the only Markdown syntax plugin that supports conceal.
A monospaced font (such as Cousine) featuring the italic, bold, and bold italic style variant for styled text.
A colorscheme (such as reedes/vim-colors-pencil) which supports the Markdown-specific highlight groups for styled text.
You should then only see the
_
and*
markup for the cursor line and in visual selections.Terminal users: consult your terminal’s documentation to configure your terminal to support bold and italic styles.
Status line indicator
Your status line can reflect the wrap mode for pencil buffers. For example,
␍
to representHardPencil
(hard line break) mode. To configure your status line and ruler, add to your.vimrc
:or if using bling/vim-airline:
The default indicators now include ‘auto’ for when Vim’s autoformat is active in hard line break mode. (If autoformat is suspended for the Insert, it’ll show the ‘hard’ indicator.)
If Unicode is detected, the default indicators are:
If you don’t like the default indicators, you can specify your own in your
.vimrc
.Note that
PencilMode()
will return blank for buffers in which pencil has not been initialized.Advanced pencil
Advanced initialization
You may want to refactor initialization statements into a function in your
.vimrc
to set up a buffer for writing:For highly-granular control, you can override pencil and other configuration settings when initializing buffers by file type:
Configurable options for
pencil#init()
include:autoformat
,concealcursor
,conceallevel
,cursorwrap
,joinspaces
,textwidth
, andwrap
. These are detailed above.Autoformat manual control
The ‘autoformat’ feature affects
HardPencil
(hard line break) mode only.To suspend autoformat for the next Insert, see above.
When you need to manually enable/disable autoformat for the current buffer, you can do so with a command:
PFormat
- enable autoformat for buffer (can still be disabled via blacklisting)PFormatOff
- disable autoformat for bufferPFormatToggle
- toggle to enable if disabled, etc.You can map a key in your
.vimrc
to toggle Vim’s autoformat:Autoformat blacklisting (and whitelisting)
The ‘autoformat’ feature affects
HardPencil
(hard line break) mode only.When editing formatted text, such as a table or code block, Vim’s autoformat will wreak havoc with the formatting. In these cases you will want autoformat suspended for the duration of the Insert.
When entering Insert mode, pencil will determine the highlight group at the cursor position. If that group has been blacklisted, pencil will suspend autoformat for the Insert. For example, if editing a buffer of type ‘markdown’, autoformat will be suspended if you invoke Insert mode from inside a
markdownFencedCodeBlock
highlight group.Blacklists are now declared by file type. The default blacklists (and whitelists) are declared in the
plugin/pencil.vim
module. Here’s an excerpt showing the configuration for the ‘markdown’ file type:The whitelist will override the blacklist and enable Vim’s autoformat if text that would normally be blacklisted doesn’t dominate the entire line. This allows autoformat to work with
inline
code and links.Auto-detecting wrap mode
If you didn’t explicitly specify a wrap mode during initialization, pencil will attempt to detect it.
It will first look for a
textwidth
(ortw
) specified in a modeline. Failing that, pencil will then sample lines from the start of the buffer.Detect via modeline
Will the wrap mode be detected accurately? Maybe. But you can improve its chances by giving pencil an explicit hint.
At the bottom of this document is a odd-looking code:
<!-- vim: set tw=73 :-->
This is an optional ‘modeline’ that tells Vim to run the following command upon loading the file into a buffer:
It tells pencil to assume hard line breaks, regardless of whether or not soft line wrap is the default editing mode for buffers of type ‘markdown’.
You explicitly specify soft wrap mode by specifying a textwidth of
0
:<!-- vim: set tw=0 :-->
Note that if the modelines feature is disabled (such as for security reasons) the textwidth will still be set by this plugin.
Detect via sampling
If no modeline with a textwidth is found, pencil will sample the initial lines from the buffer, looking for those excessively-long.
There are two settings you can add to your
.vimrc
to tweak this behavior.The maximum number of lines to sample from the start of the buffer:
Set that value to
0
to disable detection via line sampling.When the number of bytes on a sampled line per exceeds this next value, then pencil assumes soft line wrap.
If no such lines found, pencil falls back to the default wrap mode.
See also
Bloggers and developers discuss pencil and its brethern:
Other plugins of specific interest to writers:
Markdown syntax plugins
Markdown users typically won’t need to install a syntax plugin unless they want the latest version of Pope’s syntax highlighting:
Those using tables and footnotes should consider installing this plugin:
Alternatives to Tim Pope’s syntax highlighting include:
Note that the plasticboy and gabrielelana plugins may incorrectly reformat bulleted lists when Vim’s autoformat is active in pencil’s HardPencil mode.
If you find the pencil plugin useful, check out these others by @reedes:
Unimpressed by pencil? vim-pandoc offers prose-oriented features with its own Markdown variant.
Future development
If you’ve spotted a problem or have an idea on improving pencil, please report it as an issue, or better yet submit a pull request.
The text was updated successfully, but these errors were encountered: