我注意到 this video ,终端提示在分解为新行之前扩展了终端的整个宽度。我如何设置我的 PS1
变量来用一些字符填充剩余的终端空间,就像这个用户所做的那样?
问题是,我不知道如何根据命令更新 PS1
变量。在我看来,PS1
的字符串值只读入一次,就像 .bashrc
文件只读入一次一样。我是否必须在每个命令之后编写某种 Hook ?
我还应该指出,PS1
变量将根据构成它的转义字符计算为不同的长度。例如,\w
打印路径。
我知道我可以使用 $(COLUMNS)
获取终端宽度,并使用 ${#PS1} 获取当前
,做数学运算,并打印正确数量的缓冲区字符,但我如何让它每次都更新。有首选方法吗?PS1
变量的宽度
最佳答案
假设您希望提示看起来像这样:
left text----------------------------------------------------------right text
prompt$
只要 right text
的大小已知,这就非常简单了。 (例如,它可能是当前日期和时间。)我们所做的是打印正确数量的破折号(或者,对于 utf-8 终端,更漂亮的 \u2500
),然后是 右边的文本
,然后是回车符(\r
,不是换行符)和左边的文本,这将覆盖破折号。唯一棘手的一点是“破折号的正确数量”,但我们可以使用 $(tput cols)
来查看终端的宽度,幸运的是 bash
将命令-展开 PS1
。所以,例如:
PS1='\[$(printf "%*s" $(($(tput cols)-20)) "" | sed "s/ /-/g") \d \t\r\u@\h:\w \]\n\$ '
这里,$(($(tput cols)-20))
是终端的宽度减去20,这是基于\d\t
正好20 个字符宽(包括初始空格)。
PS1
不理解 utf-8 转义符 (\uxxxx
),并且将适当的替换插入到 sed
命令中涉及恼人的嵌入式报价问题,虽然这是可能的。然而,printf
确实理解 utf-8 转义,因此以不同的方式生成破折号序列更容易:
PS1='\[$(printf "\\u2500%.0s" $(seq 21 $(tput cols))) \d \t\r\u@\h:\w \]\n\$ '
另一种方法是关闭终端的自动包装,如果您使用 xterm
或实现相同控制代码的终端仿真器(或 linux 控制台本身),这是可能的。要禁用自动包装,请输出序列 ESC[?7l。要重新打开它,请使用 ESC[?7h。禁用自动换行后,一旦输出到达一行的末尾,最后一个字符将被下一个字符覆盖,而不是开始新的一行。使用这种技术,实际上没有必要计算破折号序列的确切长度;我们只需要一串破折号,它比任何控制台的宽度都长,如下所示:
DASHES="$(printf '\u2500%0.s' {1..1000})"
PS1='\[\e[?7l\u@\h:\w $DASHES \e[19D \d \t\e[?7h\]\n\$ '
此处,\e[19D
是“将光标向后移动 19 个字符”的终端仿真器代码。我本可以使用 $(tput cub 19)
代替。 (可能有一个 tput
参数用于打开和关闭自动包装,但我不知道它会是什么。)
视频中的示例还涉及在实际命令行中插入右对齐字符串。我不知道用 bash
做这件事有什么干净的方法;视频中的控制台几乎可以肯定使用了具有 RPROMPT
功能的 zsh
。当然,你可以在 bash
中输出右对齐的提示,使用与上面相同的技术,但是 readline
不会知道任何关于它们的信息,所以一旦你这样做了编辑该行的内容,正确的提示将消失。
关于bash - 如何让我的终端提示符扩展终端的宽度?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21358613/