如何在Bash中根据VI模式更改光标形状?

28

我在我的 .bashrc 文件中有以下代码:

set -o vi

当我处于插入模式时,我希望我的光标呈现出管状形状;当我处于命令模式时,我希望我的光标呈现成块状形状。就像在Vim中将以下内容放入我的.vimrc文件中所实现的那样:

let &t_SI = "\e[6 q"
let &t_SR = "\e[4 q"
let &t_EI = "\e[2 q"

除此之外,在这种情况下,我希望在命令行上拥有相同的行为。

我在这里找到了对我的问题的部分答案 - https://unix.stackexchange.com/questions/22527/change-cursor-shape-or-color-to-indicate-vi-mode-in-bash - 由 @gogolb 编写。

以下是答案,复制:

#!/bin/bash
# Script "kmtest.sh"

TEST=`bind -v | awk '/keymap/ {print $NF}'`
if [ "$TEST" = 'vi-insert' ]; then
    echo -ne "\033]12;Green\007"
else
    echo -ne "\033]12;Red\007"
fi

export PS1="\u@\h \$(kmtest.sh)> "

不幸的是,如答案所解释的那样,示例脚本仅在回车后更改光标形状,而我想要的是在按<Esc>时更改光标形状(即当我更改模式时)。

我正在运行Linux本地终端应用程序,使用Bash 4.4.7和我的$ TERM变量设置为xterm-256color。此外,我不知道tmux是否对我所要求的有任何影响,但我理想情况下希望解决方案既适用于tmux会话内部也适用于外部。


解决方案

最终我自己发现了这个问题的答案,并在另一个我发布的问题中描述了它:

如何将补丁修补的GNU readline库正确链接到所有现有程序?

不用担心,解决方案不需要打补丁。 ;)

5个回答

46

解决方法:

我按照建议在这里发布了自己问题的答案。

此解决方案适用于Bash 4.4+,因为从该版本开始,使用GNU readline库的7.0版本,其中包括vi-cmd-mode-stringvi-ins-mode-string变量所需的添加。

您可以在.inputrc文件中设置这些变量,以实现上述功能:

set show-mode-in-prompt on
set vi-cmd-mode-string "\1\e[2 q\2"
set vi-ins-mode-string "\1\e[6 q\2"

解释:

对于那些真正对以上解决方案如何工作感兴趣的人。

这两个变量vi-cmd-mode-stringvi-ins-mode-string与命令提示符一起打印到您的终端上,以提供一种视觉指示器,表明您当前处于哪种模式(即命令模式与插入模式)。

这两个变量的默认值分别为“(cmd)”和“(ins)”,用于命令和插入模式。因此,如果你将它们留在默认状态,并且有一个命令提示符,比如PS1='>>>',那么你的提示符看起来就像下面这样:

  • 命令模式:

      (cmd) >>>
    
  • 插入模式:

  •   (ins) >>>
    

根据readline的手册(见下文),你还可以通过在\1和\2转义字符之间嵌入序列来指定非可打印字符,例如终端控制序列。

vi-cmd-mode-string ((cmd))
       This  string  is  displayed immediately before the last line of the primary prompt when vi editing mode is active and in command mode.  The value is expanded like a key binding, so the
       standard set of meta- and control prefixes and backslash escape sequences is available.  Use the \1 and \2 escapes to begin and end sequences of non-printing characters, which  can  be
       used to embed a terminal control sequence into the mode string.
vi-ins-mode-string ((ins))
       This  string is displayed immediately before the last line of the primary prompt when vi editing mode is active and in insertion mode.  The value is expanded like a key binding, so the
       standard set of meta- and control prefixes and backslash escape sequences is available.  Use the \1 and \2 escapes to begin and end sequences of non-printing characters, which  can  be
       used to embed a terminal control sequence into the mode string.
因此,在我的上述解决方案中,我将终端控制序列\e[2 q(使光标成为垂直条)和\e[6 q(使光标成为管道符号)嵌入到这些\1和\2转义字符之间,从而导致我的光标在命令模式下呈现垂直条形状,在插入模式下呈现管道符形状。

1
太棒了,谢谢!不过我对这个解决方案有一个问题:当我把这三行代码放到.inputrc文件中时,在vim中(比如编辑文件时)光标也会变成管道符号。我不想要这个效果——你有什么办法可以防止这种情况发生吗? - andreas-h
1
出于某种原因,我没有遇到这个问题。我正在使用来自Github的NextJump/jarvis,并且在我的vimrc.local文件中,我已经编写了let &t_SI = "\e[6 q"等命令(请参见原始问题描述)...也许添加这些命令可以解决你的问题。 - jinscoe123
谢谢,确实如此。 - andreas-h
这在Windows上的Git Bash中可行,但光标不会闪烁,你知道如何启用闪烁吗? - user1068352
如果你想要闪烁效果,可以尝试使用上述控制序列的闪烁版本。例如,闪烁块光标 = "\e[1 q",闪烁下划线光标 = "\e[3 q",闪烁竖线光标 = "\e[5 q"。 - jinscoe123

5

这很棒。除了调整光标之外,还可以有文本模式状态消息。以下代码可行:

set show-mode-in-prompt on
set vi-cmd-mode-string "\1\e[2 q\2cmd"
set vi-ins-mode-string "\1\e[6 q\2ins"

基于模式,cmdins将显示在提示符的左侧。


2
如果你想在其他程序运行之前将光标重置为正常状态,那么你可以使用环境变量PS0。从man bash中得知:

这个参数的值被展开(见下面的PROMPING),并在读取命令之后,在执行命令之前由交互式shell显示。

使用此命令设置PS0将使其恢复光标为非闪烁块:
PS0="\e[2 q"

2

Alan Barnett的答案是关于以下内容的:

使用此命令设置PS0将导致将光标恢复为非闪烁块:

PS0="\e[2 q"

解决了我的Vim问题,但我写成了PS0="\e[2 q\2",而不是PS0="\e[2 q"。Alan的答案在我的终端输出中添加了一个]


0

这是我正在使用的,是Unicode编码。唯一的缺点是当你没有运行X服务器时,Unicode会出错!:-)

 set show-mode-in-prompt on
 set vi-ins-mode-string \1\e[34;1m\2└──[ins] \1\e[0m\2
 set vi-cmd-mode-string \1\e[33;1m\2└──[ cmd] \1\e[0m\2

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接