6

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

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

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

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

4

2 回答 2

6

假设您希望您的提示看起来像这样:

left text----------------------------------------------------------right text
prompt$ 

right text只要具有已知的大小,这非常简单。(例如,它可能是当前日期和时间。)我们所做的是打印正确数量的破折号(或者,对于 utf-8 终端,更漂亮的\u2500),然后是right text,然后是回车符 ( \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;视频中的控制台几乎可以肯定zsh与该RPROMPT功能一起使用。当然,你可以在 中输出右对齐的提示bash,使用与上面相同的技术,但readline对它们一无所知,所以一旦你对行进行编辑,右提示就会消失。

于 2014-01-26T20:37:41.250 回答
3

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

PROMPT_COMMAND=set_prompt
set_prompt () {
    PS1=...
}

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

PROMPT_COMMAND="$PROMPT_COMMAND; set_prompt"
于 2014-01-26T16:00:23.627 回答