我在this video注意到终端提示在扩展到新行之前扩展了终端的整个宽度。如何设置PS1
变量以使用某个字符填充剩余的终端空间,就像这个用户的方式一样?
问题是,我不知道如何更新每个命令的PS1
变量。在我看来,PS1
的字符串值只读入一次,就像.bashrc
文件只读入一次一样。在每个命令之后我是否必须编写某种钩子?
我还应该指出,PS1
变量将根据构成它的转义字符计算到不同的长度。例如,\w
打印路径。
我知道我可以使用$(COLUMNS)
获取终端宽度,并使用PS1
获取当前${#PS1}
变量的宽度,进行数学计算,并打印正确数量的缓冲区字符,但是如何让它每次更新。有没有首选方式?
答案 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"