如何让我的终端提示扩展终端的宽度?

Tre*_*key 5 bash terminal command-prompt terminal-emulator ps1

我在此视频中注意到,终端提示在分解为新行之前扩展了终端的整个宽度。我怎样才能设置我的PS1变量来用一些字符填充剩余的终端空间,就像这个用户所做的那样?

问题是,我不知道如何更新PS1每个命令的变量。在我看来,for 的字符串值PS1只读取一次,就像.bashrc文件只读取一次一样。我是否必须在每个命令之后编写某种钩子?

我还应该指出,该PS1变量将根据组成它的转义字符计算为不同的长度。例如,\w打印路径。

我知道我可以使用 获取终端宽度,使用 获取$(COLUMNS)当前PS1变量的宽度${#PS1},进行数学运算并打印适量的缓冲区字符,但是我如何让它每次都更新。有没有首选的方法?

ric*_*ici 5

假设您希望提示如下所示:

left text----------------------------------------------------------right text
prompt$ 
Run Code Online (Sandbox Code Playgroud)

如果right text具有已知的大小,这是非常简单的。(例如,它可能是当前日期和时间。)我们所做的是打印正确数量的破折号(或者,对于 utf-8 终端,更漂亮的\u2500),然后是right text,然后是回车(\r而不是换行符) 和左边的文本,这将覆盖破折号。唯一棘手的一点是“正确数量的破折号”,但我们可以使用它$(tput cols)来查看终端的宽度,幸运的是bashcommand-expand PS1。因此,例如:

PS1='\[$(printf "%*s" $(($(tput cols)-20)) "" | sed "s/ /-/g") \d \t\r\u@\h:\w \]\n\$ '
Run Code Online (Sandbox Code Playgroud)

这里,$(($(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\$ '
Run Code Online (Sandbox Code Playgroud)

另一种方法是关闭终端的自动换行,如果您正在使用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\$ '
Run Code Online (Sandbox Code Playgroud)

\e[19D是“将光标向后移动 19 个字符”的终端仿真器代码。我本来可以用的$(tput cub 19)。(可能有一个tput用于打开和关闭自动换行的参数,但我不知道它会是什么。)

视频中的示例还涉及在实际命令行中插入右对齐的字符串。我不知道这样做的任何干净的方法bash;视频中的控制台几乎可以肯定正在使用zshRPROMPT功能。当然,您可以bash使用与上述相同的技术在 中输出右对齐提示,但readline对它们一无所知,因此只要您对行进行了某些操作,正确的提示就会消失。


che*_*ner 3

用于在每个命令之前PROMPT_COMMAND重置值PS1

PROMPT_COMMAND=set_prompt
set_prompt () {
    PS1=...
}
Run Code Online (Sandbox Code Playgroud)

尽管某些系统脚本(或您自己)可能已经用于PROMPT_COMMAND某些用途,在这种情况下您可以简单地添加到它。

PROMPT_COMMAND="$PROMPT_COMMAND; set_prompt"
Run Code Online (Sandbox Code Playgroud)