11

I am trying to print two string separated by a TAB. I have tried:

echo -e 'foo\tbar'
printf '%s\t%s\n' foo bar

Both of them print:

foo     bar

Where the whitespace between the two is actually 5 spaces (as per selecting the output with mouse in Putty).

I have also tried using CTRL+V and pressing TAB when typing the command, with the same result.

What is the correct way to force tab being printed as tab, so I can select the output and copy it to somewhere else, with tabs?

And the secondary question: why is bash expanding tabs into spaces?

Update: Apparently, this is a problem of Putty: https://superuser.com/questions/656838/how-to-make-putty-display-tabs-within-a-file-instead-of-changing-them-to-spaces

5
  • 5
    Related: Output tab character on terminal window Commented Dec 15, 2018 at 16:34
  • Why just not escape it? printf '%s\\t%s\n' foo bar Commented Dec 15, 2018 at 17:11
  • @steeldriver Thanks that's very similar to what I need, but ultimately there isn't a solution... Commented Dec 15, 2018 at 17:19
  • 1
    @Valentin That outputs foo\tbar ... Commented Dec 16, 2018 at 1:17
  • 2
    Despite that fact that you already know that you have an issue with your terminal: Bash by itself interprets $'\t' as tabulator. So you can always concatenate strings like this - for example as assignment: v='This is'$'\t''a test' And the print it literally, e.g. printf '%s' "$v" Commented Dec 17, 2018 at 7:11

4 Answers 4

18

the whitespace between the two is actually 5 spaces.

No, it's not. Not in the output of echo or printf.

$ echo -e 'foo\tbar' | od -c
0000000   f   o   o  \t   b   a   r  \n
0000010

What is the correct way to force tab being printed as tab, so I can select the output and copy it to somewhere else, with tabs?

This is a different issue. It's not about the shell but the terminal emulator, which converts the tabs to spaces on output. Many, but not all of them do that.

It may be easier to redirect the output with tabs to a file, and copy it from there, or to use unexpand on the output to convert spaces to tabs. (Though it also can't know what whitespace was tabs to begin with, and will convert all of it to tabs, if possible.) This of course would depend on what, exactly, you need to do with the output.

4
  • I meant that when I try to select the output, it is being treated as 5 spaces. Thanks for the 'od -c' to verify the contents of the command output. Commented Dec 15, 2018 at 17:14
  • 2
    @Asu I think he understands that. His solution is to get the output via other means since the terminal emulator is not guaranteed to leave tabs as tabs when you select them in the window. However, I just checked and while putty, xterm, and konsole convert tabs to spaces, urxvt and gnome-terminal do not. So, another solution is to switch terminals. Commented Dec 15, 2018 at 17:47
  • @JoL Yes, that's the conclusion I just came to a minute ago, and I think it would be the accepted answer if somebody cares to post it as such... Commented Dec 15, 2018 at 17:50
  • 2
    @Asu, yeah, I thought about just working around the issue manually. It would be annoying to have to do that, but then I admit I hadn't realized that there are terminal emulators that do support copying tabs. Changing to one that does, would of course be a much better solution! Commented Dec 15, 2018 at 22:16
9

Like ilkkachu said, this isn't an issue with bash, but with the terminal emulator which converts tabs to spaces on output.

Checking different terminals, putty, xterm, and konsole convert tabs to spaces, while urxvt and gnome-terminal do not. So, another solution is to switch terminals.

1
  • 5
    It can also be done by the tty driver after you run stty tab3. Commented Dec 15, 2018 at 19:09
6

In printf '%s\t%s\n' foo bar, printf does output foo<TAB>bar<LF>.

f, o, b, a and r are single-width graphical characters.

Upon receiving those characters, the terminal will display a corresponding glyph and move the cursor one column to the right, unless it's already reached the right edge of the screen (paper in original tele-typewriters)), in which case it may feed a line and return to the left edge of the screen (wrap) or just discard the character depending on the terminal and how it's been configured.

<Tab> and <LF> are two control characters. <LF> (aka newline) is the line delimiter in Unix text, but for terminals, it just feeds a line (move the cursor one position down). So the terminal driver in the kernel will actually translate it to <CR> (return to the left edge of the screen), <LF> (cursor down) (stty onlcr generally on by default).

<Tab> tells the terminal to move the cursor to the next tab stop (which on most terminals are 8 positions apart by default but can also be configured to be set anywhere) without filling the gap with blanks.

So if those characters are sent to a terminal with tab stops every 8 columns whilst the cursor is at the start of an empty line, that will result in:

foo     bar

printed on the screen at that line. If they are sent whilst the cursor is in third position in a line that contains xxxxyyyyzzzz, that will result in:

xxfooyyybarz

On terminals that don't support tabulation, the terminal driver can be configured to translate those tabs to sequences of spaces. (stty tab3).

The SPC character, in original tele-typewriters would move the cursor to the right, while backspace (\b) would move it to the left. Now in modern terminals, SPC moves to the right and also erases (writes a space character as you'd expect). So the pendant of \b had to be something newer than ASCII. On most modern terminals, it's actually a sequence of characters: <Esc>, [, C.

There are more escape sequences to move n characters left, right, up, down or at any position on the screen. There are other escape sequences to erase (fill with blank) parts of lines or regions of the screen, etc.

Those sequences are typically used by visual applications like vi, lynx, mutt, dialog where text is written at arbitrary positions on the screen.

Now, all X11 terminal emulators and a few other non-X11 ones like GNU screen let you select areas of the screen for copy paste. When you select a part of what you see in the vi editor, you don't want to copy all the escape sequences that have been used to produce that output. You want to select the text you see there.

For instance if you run:

printf 'abC\rAC\bB\t\e[C\b\bD\n'

Which simulates an editor session where you enter abC, go back to the beginning, replace ab with AC, C with B, move to the next tab stop, then one more column to the right, then two columns to the left, then enter D.

You see:

ABC    D

That is, ABC, a 4 column gap and D.

If you select that with the mouse in xterm or putty, they will store in the selection ABC, 4 space characters and D, not abC<CR>AC<BS>B<Tab><Esc>[C<BS><BS>D.

What ends up in the selection is what has been sent by printf but post-processed by both the terminal driver and the terminal emulator.

For other kinds of transformation, see the <U+0065><U+0301> (e followed by a combining acute accent) changed to <U+00E9> (é the pre-composed form) by xterm.

Or echo abc that ends up being translated to ABC by the terminal driver before sending to the terminal after a stty olcuc.

Now, <Tab>, like <LF> is one of those few control characters that are actually sometimes found in text files (also <CR> in MSDOS text files, and sometimes <FF> for page break).

So some terminal emulators do choose to copy them when possible in the copy-paste buffers to preserve them (that's generally not the case of <CR> nor <LF> though).

For instance, in VTE-based terminals like gnome-terminal, you may see that, when you select the output of printf 'a\tb\n' on an empty line, gnome-terminal actually stores a\tb in the X11 selection instead of a, 7 spaces and b.

But for the output of printf 'a\t\bb\n', it stores a, 6 spaces and b, and for printf 'a\r\tb\n', a, 7 spaces and b.

There are other cases where the terminals will try to copy the actual input, like when you select two lines after running printf 'a \nb\n' where that invisible trailing space will be preserved. Or when selecting two lines doesn't include a LF character when the two lines result from wrapping at the right margin.

Now, if you want to store the output of printf into the CLIPBOARD X11 select, best is to do it directly like with:

printf 'foo\tbar\n' | xclip -sel c

Note that when you paste that in xterm or most other terminals, xterm actually replaces that \n with \r because that's the character xterm sends when you press Enter (and the terminal driver may translate it back to \n).

2
  • This is very insightful, thank you. I have tried the xclip solution and it works. But it doesn't do exactly what I had in mind and requires X11. May be this will come in handy at some point, thanks! Commented Dec 17, 2018 at 1:04
  • @Asu, X11 is what handles the copy-paste selection in terminal emulators like xterm or putty on Unix. Other terminal emulators may have they own copy-paste mechanism and ways to store arbitrary content in there, like the readbuf and register commands in GNU screen. Commented Dec 17, 2018 at 6:53
-2

You can use awk.

awk -v line="$line" 'BEGIN {print line}'

You must log in to answer this question.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.