Colorized man pages: Understood and customized

2016-08-15 21:18:01 -08:00

man() {
    env \
        LESS_TERMCAP_mb=$(printf "\e[1;31m") \
        LESS_TERMCAP_md=$(printf "\e[1;31m") \
        LESS_TERMCAP_me=$(printf "\e[0m") \
        LESS_TERMCAP_se=$(printf "\e[0m") \
        LESS_TERMCAP_so=$(printf "\e[1;44;33m") \
        LESS_TERMCAP_ue=$(printf "\e[0m") \
        LESS_TERMCAP_us=$(printf "\e[1;32m") \
            man "$@"
}

Screenshot of Terminal showing the zsh manpage with the above customizations.

Pretty neat, right? Let’s tear it apart, see what it’s doing, and think about how we might customize it to suit ourselves.

What even

The first and last lines define a shell function. Your shell recognizes these as commands, just like the tools that are installed on the system, but they’re written in shell language—like shell scripts, but recorded in memory rather than saved as files.

This shell function is named “man”—yes, the same name as the tool that shows manpages! Your shell will prefer your own functions over the pre-installed tools.

This function spans many lines, but runs only one command—the backslashes (\) escape the line breaks so that the shell interprets everything from “env” unto the first line not ending with a \ as one line.

That last physical line is “man "%@"”. But wait, you might ask—how doesn’t this just call itself?

This whole logical line runs a command called env. It passes env a bunch of environment variables (e.g., LESS_TERMCAP_us=…) to set, and then a command to run with that environment. That command, here, is man "%@".

env doesn’t know about shell functions. It’s not part of the shell; it’s a separate tool. So it doesn’t know about our function—the only man it knows about is the pre-installed tool called man.

So the one line that is the body of this function uses the tool called env to set up a custom environment in which to then run the tool called man.

"%@" tells the shell to insert all of the arguments to the function there, so that that line will run the real man with the arguments you passed in to the fake man—most probably, the optional section number and the manpage name.

Cool, but what’s LESS_TERMCAP_mb and such?

less is a pager—a tool that shows you a screenful at a time of some output, and waits for you to acknowledge it before showing you more. (It’s an expansion of an older pager called more.) It’s what man uses to do that—man just runs less and pipes the rendered manpage text into it.

I don’t know that the LESS_TERMCAP_xx trick is actually documented—it’s not mentioned in less’s manpage. But here’s what it does:

termcap is a database of terminal capabilities. (I won’t fault you for imagining a terminal in a cape right now.) The idea is that every kind of terminal is different (which it was, back in the day), so there needs to be a database in which you can look up whichever terminal the user has in order to find out what that kind of terminal is capable of and how to tell it to do each thing.

Each capability has a two-letter identifier, and maps to a string of characters. Here are the capabilities used in the Gist, as defined in the aforelinked manpage:

mb
Start blinking
md
Start bold mode
me
End all mode like so, us, mb, md and mr
so
Start standout mode
se
End standout mode
us
Start underlining
ue
End underlining

“Standout” mode, in case you’re wondering, is inverse video: use the assigned foreground color as the background color and vice versa.

zsh has a built-in command called echotc that lets us play with these records. For example:

% echotc us && echo -n 'This should be underlined' && \
echotc ue && echo ' and this should not.'
This should be underlined and this should not.

Yup, us starts underlining and ue ends underlining!

So when we want something underlined (for example), the us and ue entries in our terminal’s termcap record are what we need to send to the terminal to start and end underlining that section of text.

And less, it seems, provides this handy way to override those entries using environment variables. We can make us and ue and any other termcap string do whatever we want!

What are these strings you speak of?

Yes, just what are these termcap strings doing now?

All of them follow a similar format:

\e[(one or more numbers separated by semicolons)m

The \e[ essentially tells the terminal to start listening for a command that will change its behavior. m is actually the command here; all the inputs to the command come before it. The m command tells the terminal to change how it renders subsequent text until further notice.

(\e is the ASCII/Unicode character U+001B ESCAPE, just in case you were wondering. In other languages, you might refer to it as \x1b or \033.)

These are ANSI escape sequences, which come in a lot more varieties than these, but these are the most common. The m command takes a list of arguments, all of which are numbers, separated by semicolons:

0
Reset to standard configuration
1
Bold
31
Set foreground color to red
32
Set foreground color to green
33
Set foreground color to yellow
44
Set background color to blue

There are more than just those options for our foreground and background colors, of course! We’ll see more shortly.

So, for example, the first variable:

LESS_TERMCAP_mb=$(printf "\e[1;31m") \

says that when we want to start blinking, display bold red text instead.

Customizing

Let’s start by seeing it in my own Terminal theme:

Screenshot of Terminal showing the zsh manpage with the above customizations.

I’m also going to make one change to start with:

man() {
    env \
        LESS_TERMCAP_mb=$'\e[1;31m' \
        LESS_TERMCAP_md=$'\e[1;31m' \
        LESS_TERMCAP_me=$'\e[0m' \
        LESS_TERMCAP_se=$'\e[0m' \
        LESS_TERMCAP_so=$'\e[1;44;33m' \
        LESS_TERMCAP_ue=$'\e[0m' \
        LESS_TERMCAP_us=$'\e[1;32m' \
            man "$@"
}

$(…) runs a command and is replaced with the output, so $(printf …) replaces the $(…) with the string that printf prints.

That’s redundant! We can simplify this to $'…', which will let us have the \e sequences without calling the shell built-in function printf for every one of these.

OK, let’s start with…

The bountiful rainbow of color

We have a lot more than those four colors to choose from. The basic ANSI set is 16, and most terminal emulators have supported 256-color sequences for years now. That Wikipedia article has charts of all the different color codes; here are the basic groups:

30–37
Foreground color, dark
90–97
Foreground color, bright
40–47
Background color, dark
100–107
Background color, bright
38;5;###
Set foreground color to color ### (256-color extension)
48;5;###
Set background color to color ### (256-color extension)

Now that we have more colors to work with, let’s explore…

What things in a manpage use which capabilities?

I changed the function so that every capability got a unique character—i.e., I changed blinking (mb) to 35, which is fuchsia—and then looked up a few things.

It doesn’t seem like anything uses blinking.

That leaves bold (mb), underline (us), and standout/inverse (so).

man and less use:

  • bold for headings, command synopses, and code font
  • underline for proper names (for example, “termcap” and “terminfo” in the termcap manpage), variable names (“name”, “bp”, “id”, etc.), and type names in some manpages (such as dispatch_queue_create(3))
  • inverse for the prompt at the bottom

(Some of these are more consistent than others. This is the problem with looking things from the styling end rather than the semantic end—which is why web developers had to invent CSS!)

My own changes

I made these changes:

  • Cut out mb entirely, since it seems unused
  • Change md from setting 31 (red) to setting 36 (cyan) for the foreground color
  • Change so from 44;33 (yellow on blue) to 40;92 (bright green on black)

I left the us override alone; I think the green generally works where that’s used.

So here’s what that looks like:

Screenshot of the zsh manpage with the customized style.
Screenshot of the dispatch_queue_create manpage with the customized style.

6 Responses to “Colorized man pages: Understood and customized”

  1. Darren Embry Says:

    A modified version of this that does not invoke external programs other than man. Works in bash and zsh.

    man () {
    LESS_TERMCAP_mb=$’\e'”[1;31m” \
    LESS_TERMCAP_md=$’\e'”[1;31m” \
    LESS_TERMCAP_me=$’\e'”[0m” \
    LESS_TERMCAP_se=$’\e'”[0m” \
    LESS_TERMCAP_so=$’\e'”[1;44;33m” \
    LESS_TERMCAP_ue=$’\e'”[0m” \
    LESS_TERMCAP_us=$’\e'”[1;32m” \
    command man “$@”
    }

    The $’…’ construct is for escape sequences, e.g., $’\e’. The `command` builtin in bash and zsh avoids recursively invoking the alias. You can specify = as many times as you want before specifying a command and arguments to make those variable assignments temporary to the invocation of that command. No `env` required.

  2. Darren Embry Says:

    That should say: you can specify var=val as many times…

  3. Adam Says:

    On my Mac (10.10) I had to put this in ~/.bash_profile instead of ~/.bash_rc for the man override to stick, in case anyone else has this problem.

  4. John Macdonald Says:

    Adam, it is usually ~/.bashrc (with no underscore). ~/.bash_profile *does* have the underscore, so that works too – it just sets up the same command for your non-interactive shell sessions as well as the interactive ones, which wastes a few microseconds of startup time and a bit (well, bytes) of memory.

  5. Ingo Schwarze Says:

    This is all pointless. For a real answer, see:

    https://lists.gnu.org/archive/html/groff/2016-08/msg00016.html

  6. Jack Jack Says:

    Hi Ingo,

    Thanks for your reply, but how could I make your code (below) work under my OSX 10.10.5? man on OSX does not have -T, and -t is alias for /usr/bin/groff -Tps -mandoc -c.

    Thanks!

    export MANPAGER=’lynx -force_html’
    alias man=’man -Thtml’

Leave a Reply

Do not delete the second sentence.