如何使终端提示延长终端的宽度?

时间:2014-01-26 01:45:54

标签: bash terminal command-prompt terminal-emulator ps1

我在this video注意到终端提示在扩展到新行之前扩展了终端的整个宽度。如何设置PS1变量以使用某个字符填充剩余的终端空间,就像这个用户的方式一样?

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

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

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

2 个答案:

答案 0 :(得分:4)

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

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 [ 7 l 。要重新打开它,请使用 ESC [ 7 h 。禁用自动换行后,一旦输出到达行尾,最后一个字符将被下一个字符覆盖而不是开始换行。使用这种技术,没有必要计算短划线序列的确切长度;我们只需要一个破折号字符串,比任何控制台都宽,比如下:

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对它们一无所知,所以只要你做了一些编辑线的事情,正确的提示将消失。

答案 1 :(得分:1)

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

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

虽然某些系统脚本(或您自己)可能已经使用PROMPT_COMMAND作为某些内容,但在这种情况下您只需添加它即可。

PROMPT_COMMAND="$PROMPT_COMMAND; set_prompt"