【问题标题】:Alt key shortcuts not working on gnome terminal with VimAlt 键快捷键在 Vim 的 gnome 终端上不起作用
【发布时间】:2011-10-10 08:58:29
【问题描述】:

我在 gnome 终端上运行 Vim。但是 alt 键映射不起作用。
例如(这只是一个例子):

:imap <A-i> <Esc>

它在 GVim 中运行良好。但是当我在 gnome 终端中使用 Vim 运行相同的命令时,它就不起作用了。
所以我认为问题出在终端上,对吧?
我该如何解决?

谢谢

编辑:我在同一台机器上安装了 Windows 7,并且使用 Windows 终端也可以正常工作。

【问题讨论】:

    标签: vim terminal gnome alt-key


    【解决方案1】:

    问题

    终端模拟器有两种方法可以发送 Alt 键(通常称为 Meta 键,因为实际终端没有 Alt)。它可以在使用 Alt 时发送 8 位字符并设置高位,也可以使用转义序列,将 Alt-a 发送为 &lt;Esc&gt;a。 Vim 期望看到 8 位编码而不是转义序列。

    xterm 等一些终端仿真器可以设置为使用任一模式,但 Gnome 终端 不提供任何此类设置。老实说,在这些 Unicode 编辑时代,8 位编码无论如何都不是一个好主意。但是转义序列也不是没有问题的。它们无法区分 &lt;Esc&gt;j 表示 Alt-j 与按 Esc 后跟 j

    在早期的终端使用中,键入 Escj 是另一种在没有 Meta 键的情况下在键盘上发送 Meta 的方法,但这不会'不适合 vi 使用 Esc 离开 insert 模式。

    解决办法

    可以通过配置 vim 将转义序列映射到它们的 Alt 组合来解决这个问题。

    将此添加到您的 .vimrc:

    let c='a'
    while c <= 'z'
      exec "set <A-".c.">=\e".c
      exec "imap \e".c." <A-".c.">"
      let c = nr2char(1+char2nr(c))
    endw
    
    set timeout ttimeoutlen=50
    

    Alt-letter 现在可以被终端中的 vi 和 gvim 识别。 timeout 设置用于解决转义序列的歧义。 50ms 内发送的 Escj 将映射到&lt;A-j&gt;,大于 50ms 将被视为单独的键。这应该有足够的时间来区分元编码和按两个键。

    如果您不喜欢设置 timout,它会为其他映射的键序列超时(默认为一秒后),那么您可以改用 ttimeout。 ttimeout 仅适用于键码,不适用于其他映射。

    set ttimeout ttimeoutlen=50
    

    【讨论】:

    • 非常感谢您!你终于解决了一个困扰我很久的问题!
    • 将 let 更改为 'A' 并将 while 条件更改为 'Z' 以映射大写键。如果你愿意,两者都可以。
    • 我发现使用此设置,如果我处于命令行模式或可视模式,如果我按转义键,它将执行映射或插入多字节字符。我通过使用这些设置按两次 Escape 允许自己离开这些模式来解决此问题:vnoremap &lt;Esc&gt; &lt;C-v&gt;&lt;Esc&gt;cnoremap &lt;Esc&gt; &lt;C-c&gt;
    • @trusktr 是的,您可以设置 ttimeout。将添加到答案。
    • 如果您在 tmux 中使用 vim 并且无法弄清楚为什么上面的方法不起作用,那是因为 tmux 在 ESC 之后插入了一个等待,并且它自己会解释键码。将 set -sg escape-time 0 放入您的 .tmux.conf 以禁用此行为。
    【解决方案2】:

    对于Gnome-terminal,请改用以下内容:

    imap ^[i <Esc>
    

    ^[i 应按Ctrl-v Alt-i 输入

    注意:要复制到别处时需要yankputVim中。如果您只是在gedit 之类的编辑器中复制映射,则映射可能会损坏。

    EDIT这里是一个例子,它使Alt-k在光标上方添加一个空行,Alt-j在当前行之后添加一个空行。

    " Alt-j/k to add a blank line
    if has('gui_running')
        " the following two lines do not work in vim, but work in Gvim
        nnoremap <silent><A-j> :set paste<CR>m`o<Esc>``:set nopaste<CR>
        nnoremap <silent><A-k> :set paste<CR>m`O<Esc>``:set nopaste<CR>
    else
        " these two work in vim
        " shrtcut with alt key: press Ctrl-v then Alt-k
        " ATTENTION: the following two lines should not be 
        " edited under other editors like gedit. ^[k and ^[j will be broken!
        nnoremap ^[k :set paste<CR>m`O<Esc>``:set nopaste<CR>
        nnoremap ^[j :set paste<CR>m`o<Esc>``:set nopaste<CR>
    endif
    

    【讨论】:

    • 我更喜欢 'imap i ...' 拼写,这样我的 .vimrc 中就没有控制字符了
    【解决方案3】:

    试试

    <m-i>
    

    或者,如果键入 alti 插入一个字符(就像在我的情况下,它插入一个小括号:ˆ)只需映射到该字符:

    :inoremap ˆ <esc>
    

    要小心,因为这个不起作用(至少在我的系统中,MacOS 10.6)。插入符号等待一个字母,因为它不完全是插入符号,它是 circumflex

    【讨论】:

    • 感谢您的回复。我也已经试过了。但它也不起作用。
    【解决方案4】:

    同样的事情也发生在我身上。我用“gnome terminal alt key”在谷歌上搜索,发现有人在找到的第一个链接中问了几乎相同的问题:“How to disable the alt-hotkey behavior on gnome terminal?”。 (第二个链接就是这个问题)

    所以,也许你可以试试:

    Edit > Keyboard Shortcuts, and uncheck "Enable menu access keys"
    

    【讨论】:

    • 不知道这是否是 gnome-terminal 上那些奇怪的 ubuntu 上游编辑之一,但 debian 的版本并没有,只需编辑->首选项->快捷方式并禁用它们都对我有用
    【解决方案5】:

    这些快捷方式可能实际上来自 Gnome 桌面。尝试查看 Gnome 键盘快捷键工具(系统菜单、首选项、键盘快捷键),它可以让您查看和修改 Gnome 桌面上定义的快捷键。如果组合键被分配给 Gnome 桌面上的某个功能,则将其删除,然后该组合键应正确过滤到 Vim。

    或者你可能是对的,这是终端的问题。并非所有终端都支持所有按键组合。您的问题可能是在:h map-alt-keys 的 Vim 帮助文档中描述的问题。文档提供了一种解决方法,但不是很好。

    【讨论】:

    • 由于 alt 快捷键适用于 GVim,我认为这不是键盘问题。也许问题真的是终端。你知道我可以试试 Ubuntu 的替代终端吗?
    • 这里是 Ubuntu 的终端替代品列表:howtogeek.com/howto/ubuntu/…
    【解决方案6】:

    查看http://vimdoc.sourceforge.net/htmldoc/map.html 的第 1.10 节。这似乎表明 gnome-terminal 自动转义了 Alt 修饰符,因此它不会以 Vim 期望的方式切换发送的字节。该文档似乎表明,除了使用不同的终端(例如 xterm)之外,实际上没有办法解决这个问题。

    这当然令人沮丧,因为据我所知,Linux 机器也无法使用 D(Mac 的命令或 Linux 的超级)绑定,所以至少就终端而言,我们仅限于 Shift 和 Ctrl 修饰符,如果我们想确保我们可以在终端 Vim 上使用我们在 Gvim 中使用的所有命令,这是令人沮丧的(至少没有切换终端,我可能过于固执 - gnome-terminal 实在是太漂亮了)。我一直在寻找解决此问题的方法,但找不到任何东西。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2012-04-23
      • 2017-08-10
      • 1970-01-01
      • 2022-06-25
      • 2016-03-02
      • 2021-06-20
      • 2015-11-07
      相关资源
      最近更新 更多