doc: update "horrific mess when resizing terminal window" section
- Don't say it's a bug in Zsh. It's more complicated than that. - Mention that the the issue can be avoided by using Kitty. - Do not mention that the issue can also be avoided by using zsh4humans (even though it's true). If you, dear reader, see this, give zsh4humans a try.pull/1676/head
parent
4543076483
commit
a23c4314a1
64
README.md
64
README.md
|
@ -1649,19 +1649,25 @@ There are several ways to fix this.
|
||||||
|
|
||||||
### Horrific mess when resizing terminal window
|
### Horrific mess when resizing terminal window
|
||||||
|
|
||||||
When you resize terminal window horizontally back and forth a few times, you might see this ugly
|
When you resize a terminal window horizontally back and forth a few times, you might see this ugly
|
||||||
picture.
|
picture.
|
||||||
|
|
||||||
![Powerlevel10k Resizing Mess](
|
![Powerlevel10k Resizing Mess](
|
||||||
https://raw.githubusercontent.com/romkatv/powerlevel10k-media/master/resizing-mess.png)
|
https://raw.githubusercontent.com/romkatv/powerlevel10k-media/master/resizing-mess.png)
|
||||||
|
|
||||||
tl;dr: This is a bug in Zsh that isn't specific to Powerlevel10k. See [mitigation](#mitigation).
|
tl;dr: This issue arises when a terminal reflows Zsh prompt upon resizing. It isn't specific to
|
||||||
|
Powerlevel10k. See [mitigation](#mitigation).
|
||||||
|
|
||||||
#### Zsh bug
|
*Note: This section [used to say](
|
||||||
|
https://github.com/romkatv/powerlevel10k/blob/dce00cdb5daaa8a519df234a7012ba3257b644d4/README.md#horrific-mess-when-resizing-terminal-window)
|
||||||
|
that the problem is caused by a bug in Zsh. While it's true that it's possible to avoid the problem
|
||||||
|
in many circumstances by modifying Zsh, it cannot be completely resolved this way. Thus it's unfair
|
||||||
|
to pin the blame on Zsh.*
|
||||||
|
|
||||||
This issue is caused by a bug in Zsh that gets triggered when the vertical distance between the
|
#### The anatomy of the problem
|
||||||
start of the current prompt and the cursor (henceforth `VD`) changes when the terminal window is
|
|
||||||
resized. This bug is not specific to Powerlevel10k.
|
The issue is manifested when the vertical distance between the start of the current prompt and the
|
||||||
|
cursor (henceforth `VD`) changes when the terminal window is resized.
|
||||||
|
|
||||||
When a terminal window gets shrunk horizontally, there are two ways for a terminal to handle long
|
When a terminal window gets shrunk horizontally, there are two ways for a terminal to handle long
|
||||||
lines that no longer fit: *reflow* or *truncate*.
|
lines that no longer fit: *reflow* or *truncate*.
|
||||||
|
@ -1683,9 +1689,9 @@ Terminal truncates text when shrinking:
|
||||||
|
|
||||||
Reflowing strategy can change the height of terminal content. If such content happens to be between
|
Reflowing strategy can change the height of terminal content. If such content happens to be between
|
||||||
the start of the current prompt and the cursor, Zsh will print prompt on the wrong line. Truncation
|
the start of the current prompt and the cursor, Zsh will print prompt on the wrong line. Truncation
|
||||||
strategy never changes the height of terminal content, so it doesn't trigger this bug in Zsh.
|
strategy never changes the height of terminal content, so it doesn't trigger this issue.
|
||||||
|
|
||||||
Let's see how the bug plays out in slow motion. We'll start by launching `zsh -df` and pasting
|
Let's see how the issue plays out in slow motion. We'll start by launching `zsh -f` and pasting
|
||||||
the following code:
|
the following code:
|
||||||
|
|
||||||
```zsh
|
```zsh
|
||||||
|
@ -1701,13 +1707,13 @@ PROMPT=$'${$((pause()))+}left>${(pl.$((COLUMNS-12))..-.)}<right\n> '
|
||||||
When `PROMPT` gets expanded, it calls `pause` to let us observe the state of the terminal. Here's
|
When `PROMPT` gets expanded, it calls `pause` to let us observe the state of the terminal. Here's
|
||||||
the initial state:
|
the initial state:
|
||||||
|
|
||||||
![Zsh Resizing Bug 1](
|
![Terminal Resizing Bug 1](
|
||||||
https://raw.githubusercontent.com/romkatv/powerlevel10k-media/master/resize-bug-1.png)
|
https://raw.githubusercontent.com/romkatv/powerlevel10k-media/master/resize-bug-1.png)
|
||||||
|
|
||||||
Zsh keeps track of the cursor position relative to the start of the current prompt. In this case it
|
Zsh keeps track of the cursor position relative to the start of the current prompt. In this case it
|
||||||
knows that the cursor is one line below. When we shrink the terminal window, it looks like this:
|
knows that the cursor is one line below. When we shrink the terminal window, it looks like this:
|
||||||
|
|
||||||
![Zsh Resizing Bug 2](
|
![Terminal Resizing Bug 2](
|
||||||
https://raw.githubusercontent.com/romkatv/powerlevel10k-media/master/resize-bug-2.png)
|
https://raw.githubusercontent.com/romkatv/powerlevel10k-media/master/resize-bug-2.png)
|
||||||
|
|
||||||
At this point the terminal sends `SIGWINCH` to Zsh to notify it about changes in the terminal
|
At this point the terminal sends `SIGWINCH` to Zsh to notify it about changes in the terminal
|
||||||
|
@ -1719,7 +1725,7 @@ terminal content that follows and prints reexpanded prompt there. However, after
|
||||||
no longer one line above the cursor. It's two lines above! Zsh ends up printing new prompt one line
|
no longer one line above the cursor. It's two lines above! Zsh ends up printing new prompt one line
|
||||||
too low.
|
too low.
|
||||||
|
|
||||||
![Zsh Resizing Bug 3](
|
![Terminal Resizing Bug 3](
|
||||||
https://raw.githubusercontent.com/romkatv/powerlevel10k-media/master/resize-bug-3.png)
|
https://raw.githubusercontent.com/romkatv/powerlevel10k-media/master/resize-bug-3.png)
|
||||||
|
|
||||||
In this case we ended up with unwanted junk content because `VD` has *increased*. When you make
|
In this case we ended up with unwanted junk content because `VD` has *increased*. When you make
|
||||||
|
@ -1728,33 +1734,30 @@ higher than intended, potentially erasing useful content in the process.
|
||||||
|
|
||||||
Here are a few more examples where shrinking terminal window increased `VD`.
|
Here are a few more examples where shrinking terminal window increased `VD`.
|
||||||
|
|
||||||
Simple one-line left prompt with right prompt. No `prompt_subst`. Note that the cursor is below the
|
- Simple one-line left prompt with right prompt. No `prompt_subst`. Note that the cursor is below
|
||||||
prompt line (hit *ESC-ENTER* to get it there).
|
the prompt line (hit *ESC-ENTER* to get it there).
|
||||||
|
![Zsh Prompt That Breaks on Terminal Shrinking 1](
|
||||||
![Zsh Prompt That Breaks on Terminal Shrinking 1](
|
|
||||||
https://raw.githubusercontent.com/romkatv/powerlevel10k-media/master/resize-breakable-1.png)
|
https://raw.githubusercontent.com/romkatv/powerlevel10k-media/master/resize-breakable-1.png)
|
||||||
|
- Simple one-line left prompt. No `prompt_subst`, no right prompt. Here `VD` is bound to increase
|
||||||
Simple one-line left prompt. No `prompt_subst`, no right prompt. Here `VD` is bound to increase
|
upon terminal shrinking due to the command line wrapping around.
|
||||||
upon terminal shrinking due to the command line wrapping around.
|
![Zsh Prompt That Breaks on Terminal Shrinking 2](
|
||||||
|
|
||||||
![Zsh Prompt That Breaks on Terminal Shrinking 2](
|
|
||||||
https://raw.githubusercontent.com/romkatv/powerlevel10k-media/master/resize-breakable-2.png)
|
https://raw.githubusercontent.com/romkatv/powerlevel10k-media/master/resize-breakable-2.png)
|
||||||
|
|
||||||
#### Zsh patch
|
#### Zsh patch
|
||||||
|
|
||||||
The bug described above has been partially fixed (only for some terminals) in [this branch](
|
[This Zsh patch](https://github.com/romkatv/zsh/tree/fix-winchanged) fixes the issue on some
|
||||||
https://github.com/romkatv/zsh/tree/fix-winchanged). The idea behind the fix is to use `sc` (save
|
terminals. The idea behind the patch is to use `sc` (save cursor) terminal capability before
|
||||||
cursor) terminal capability before printing prompt and `rc` (restore cursor) to move cursor back
|
printing prompt and `rc` (restore cursor) to move cursor back to the original position when prompt
|
||||||
to the original position when prompt needs to be refreshed.
|
needs to be refreshed.
|
||||||
|
|
||||||
The patch works only on terminals that reflow saved cursor position together with text when the
|
The patch works only on terminals that reflow saved cursor position together with text when the
|
||||||
terminal window is resized. The patch has no observable effect on terminals that don't reflow text
|
terminal window is resized. The patch has no observable effect on terminals that don't reflow text
|
||||||
on resize (both patched and unpatched Zsh behave correctly) and on terminals that reflow text but
|
on resize (both patched and unpatched Zsh behave correctly) and on terminals that reflow text but
|
||||||
not saved cursor position (both patched and unpatched Zsh redraw prompt at the same incorrect
|
not the saved cursor position (both patched and unpatched Zsh redraw prompt at the same incorrect
|
||||||
position). In other words, the patch fixes the resizing issue on some terminals while keeping the
|
position). In other words, the patch fixes the resizing issue on some terminals while keeping the
|
||||||
behavior unchanged on others.
|
behavior unchanged on others.
|
||||||
|
|
||||||
There are two alternative approaches to fixing the bug that may seem to work at first glance but in
|
There are two alternative approaches to patching Zsh that may seem to work at first glance but in
|
||||||
fact don't:
|
fact don't:
|
||||||
|
|
||||||
- Instead of `sc`, use `u7` terminal capability to query the current cursor position and then `cup`
|
- Instead of `sc`, use `u7` terminal capability to query the current cursor position and then `cup`
|
||||||
|
@ -1773,6 +1776,15 @@ There is no ETA for the patch making its way into upstream Zsh. See [discussion]
|
||||||
|
|
||||||
There are a few mitigation options for this issue.
|
There are a few mitigation options for this issue.
|
||||||
|
|
||||||
|
- Use [Kitty](https://sw.kovidgoyal.net/kitty/) and enable terminal-shell integration in
|
||||||
|
Powerlevel10k. The latter can be done by appending these lines to `~/.zshrc`:
|
||||||
|
```zsh
|
||||||
|
unset KITTY_SHELL_INTEGRATION
|
||||||
|
ITERM_SHELL_INTEGRATION_INSTALLED=Yes
|
||||||
|
```
|
||||||
|
At the time of this writing (Dec 2021) the latest release of Kitty (0.23.1) doesn't have the
|
||||||
|
necessary code to fix the issue. If by the time you are reading this there is no newer version,
|
||||||
|
you'll have to build from source.
|
||||||
- Apply [the patch](#zsh-patch) and [rebuild Zsh from source](
|
- Apply [the patch](#zsh-patch) and [rebuild Zsh from source](
|
||||||
https://github.com/zsh-users/zsh/blob/master/INSTALL). It won't help if you are using Alacritty,
|
https://github.com/zsh-users/zsh/blob/master/INSTALL). It won't help if you are using Alacritty,
|
||||||
Kitty or some other terminal that reflows text on resize but doesn't reflow saved cursor position.
|
Kitty or some other terminal that reflows text on resize but doesn't reflow saved cursor position.
|
||||||
|
|
Loading…
Reference in New Issue