Автозаполнение Bash в режиме оболочки Emacs

93

В терминале GNOME Bash выполняет интеллектуальное автозаполнение. Например

apt-get in<TAB>

становится

apt-get install

В режиме оболочки Emacs это автозаполнение не работает даже после того, как я явно исходил /etc/bash_completion. Приведенный выше пример придерживается inили автоматически завершается с именем файла в текущем каталоге, а не с допустимым apt-getпараметром команды. Предположительно, это потому, что Emacs перехватывает нажатие клавиши Tab. Как включить интеллектуальное автозаполнение в shell-mode?

Крис Конвей
источник
Для всех, кто плохо знаком с emacs ... в emacs есть и другие режимы оболочки . Например, у eshell-modeкоторого есть завершение табуляции. Подробнее здесь: masteringemacs.org/articles/2010/11/01/…
Росс
3
Автозаполнение eshell работает только для локальных каталогов, если вы подключитесь по ssh к другой машине, вы внезапно потеряете эту возможность.
hajovonta

Ответы:

93

Я знаю, что этому вопросу уже три года, но я тоже был заинтересован в решении этого вопроса. Поиск в сети направил меня на часть elisp, которая заставляет Emacs использовать bash для завершения в режиме оболочки. В любом случае у меня работает.

Проверьте это на https://github.com/szermatt/emacs-bash-completion .

Дэвид Кристиансен
источник
6
+1 за то, что у меня хватило смелости ответить на вопрос трехлетней давности, но все еще актуальный. Благодарность!
djhaskin987
3
Это уже есть в melpa melpa.org/#/bash-completion , установка проста M-x package-list-packages, но в конце концов у меня это не сработало, я не знаю почему, это может быть мой emacs (24.3.1) или bash (4.3.30) версия. В документации написано, что «bash-Completion.el весьма чувствителен к версии ОС и BASH ...», тогда это список, в котором мои версии отсутствуют.
boclodoa
К сожалению, нерешенная проблема №6 кажется не очень полезной для текущего набора инструментов CLI.
Джесси Глик,
Эта проблема 6 была закрыта как <по крайней мере, в основном исправлена ​​/ требуется пример того, что она не работает> (перефразировано).
Мэтью Элви,
21

В оболочке emacs автозаполнение выполняет именно emacs, а не bash. Если оболочка и emacs не синхронизированы (например, при использовании pushd, popd или какой-либо пользовательской функции bash, которая изменяет текущий каталог оболочки), то автозаполнение перестает работать.

Чтобы исправить это, просто введите «dirs» в оболочку, и все снова синхронизируется.

В моем .emacs также есть следующее:

(global-set-key "\M-\r" 'shell-resync-dirs)

Затем простое нажатие Esc-return повторно синхронизирует автозаполнение.

Стив Лэйси
источник
9
Это хороший ответ на другой вопрос!
Крис Конвей,
Спасибо. Я использую автопереход, и это была проблема, из-за которой каталоги не синхронизировались.
Plankalkül
Крис Конвей, да, к этому: emacs.stackexchange.com/questions/30550/… :-)
unhammer
15

Я не знаю ответа на это. Но причина того, что это не работает так, как вы ожидаете, вероятно, заключается в том, что завершение в оболочках emacs обрабатывается внутри emacs (функцией comint-dynamic-complete) и не имеет встроенных функций интеллектуального завершения.

Боюсь, это нелегко исправить.

Изменить: предложение njsf об использовании режима термина, вероятно, так же хорошо, как и получается. Начни с

Mx срок
Он включен в стандартный дистрибутив emacs (и, по крайней мере, в emacs21-common или emacs22-common в Ubuntu и Debian).

Матли
источник
2
Завершение табуляции еще хуже с M-x term.
mcandre
7

Пожалуйста, рассмотрите другой режим M-x term, как я сделал это, когда возникла проблема с попаданием в 2011. Я пытался собрать все усилия через Inet в то время, чтобы заставить оболочку работать с завершением Bash, включая этот вопрос. Но, открыв альтернативу перед лицом, term-modeя даже не хочу пробовать eshell.

Это полноценный эмулятор терминала, поэтому вы можете запускать внутри интерактивную программу, например Midnight Commander. Или переключитесь на zshзавершение, чтобы не терять время на настройку Emacs.

Вы получаете завершение TAB в bash бесплатно. Но что еще важнее, вы получаете все возможности Readline, такие как инкрементный или префиксный поиск команд . Чтобы сделать эту настройку более удобной, проверьте мои .inputrc , .bashrc , .emacs .

Существенная часть .inputrc:

# I like this!
set editing-mode emacs

# Don't strip characters to 7 bits when reading.
set input-meta on

# Allow iso-latin1 characters to be inserted rather than converted to
# prefix-meta sequences.
set convert-meta off

# Display characters with the eighth bit set directly rather than as
# meta-prefixed characters.
set output-meta on

# Ignore hidden files.
set match-hidden-files off

# Ignore case (on/off).
set completion-ignore-case on

set completion-query-items 100

# First tab suggests ambiguous variants.
set show-all-if-ambiguous on

# Replace common prefix with ...
set completion-prefix-display-length 1

set skip-completed-text off

# If set to 'on', completed directory names have a slash appended. The default is 'on'.
set mark-directories on
set mark-symlinked-directories on

# If set to 'on', a character denoting a file's type is appended to the
# filename when listing possible completions. The default is 'off'.
set visible-stats on

set horizontal-scroll-mode off

$if Bash
"\C-x\C-e": edit-and-execute-command
$endif

# Define my favorite Emacs key bindings.
"\C-@": set-mark
"\C-w": kill-region
"\M-w": copy-region-as-kill

# Ctrl+Left/Right to move by whole words.
"\e[1;5C": forward-word
"\e[1;5D": backward-word
# Same with Shift pressed.
"\e[1;6C": forward-word
"\e[1;6D": backward-word

# Ctrl+Backspace/Delete to delete whole words.
"\e[3;5~": kill-word
"\C-_": backward-kill-word

# UP/DOWN filter history by typed string as prefix.
"\e[A": history-search-backward
"\C-p": history-search-backward
"\eOA": history-search-backward
"\e[B": history-search-forward
"\C-n": history-search-forward
"\eOB": history-search-forward

# Bind 'Shift+TAB' to complete as in Python TAB was need for another purpose.
"\e[Z": complete
# Cycling possible completion forward and backward in place.
"\e[1;3C": menu-complete                    # M-Right
"\e[1;3D": menu-complete-backward           # M-Left
"\e[1;5I": menu-complete                    # C-TAB

.bashrc(ДА! В Bash есть dabbrev от любого слова в ~/.bash_history):

set -o emacs

if [[ $- == *i* ]]; then
  bind '"\e/": dabbrev-expand'
  bind '"\ee": edit-and-execute-command'
fi

.emacs для удобства навигации в буфере терминов:

(setq term-buffer-maximum-size (lsh 1 14))

(eval-after-load 'term
  '(progn
    (defun my-term-send-delete-word-forward () (interactive) (term-send-raw-string "\ed"))
    (defun my-term-send-delete-word-backward () (interactive) (term-send-raw-string "\e\C-h"))
    (define-key term-raw-map [C-delete] 'my-term-send-delete-word-forward)
    (define-key term-raw-map [C-backspace] 'my-term-send-delete-word-backward)
    (defun my-term-send-forward-word () (interactive) (term-send-raw-string "\ef"))
    (defun my-term-send-backward-word () (interactive) (term-send-raw-string "\eb"))
    (define-key term-raw-map [C-left] 'my-term-send-backward-word)
    (define-key term-raw-map [C-right] 'my-term-send-forward-word)
    (defun my-term-send-m-right () (interactive) (term-send-raw-string "\e[1;3C"))
    (defun my-term-send-m-left () (interactive) (term-send-raw-string "\e[1;3D"))
    (define-key term-raw-map [M-right] 'my-term-send-m-right)
    (define-key term-raw-map [M-left] 'my-term-send-m-left)
    ))

(defun my-term-mode-hook ()
  (goto-address-mode 1))
(add-hook 'term-mode-hook #'my-term-mode-hook)

Как и любые обычные команды, C-x oкоторые не работают в режиме эмуляции терминала, я расширил раскладку клавиш:

(unless
    (ignore-errors
      (require 'ido)
      (ido-mode 1)
      (global-set-key [?\s-d] #'ido-dired)
      (global-set-key [?\s-f] #'ido-find-file)
      t)
  (global-set-key [?\s-d] #'dired)
  (global-set-key [?\s-f] #'find-file))

(defun my--kill-this-buffer-maybe-switch-to-next ()
  "Kill current buffer. Switch to next buffer if previous command
was switching to next buffer or this command itself allowing
sequential closing of uninteresting buffers."
  (interactive)
  (let ( (cmd last-command) )
    (kill-buffer (current-buffer))
    (when (memq cmd (list 'next-buffer this-command))
      (next-buffer))))
(global-set-key [s-delete] 'my--kill-this-buffer-maybe-switch-to-next)
(defun my--backward-other-window ()
  (interactive)
  (other-window -1))
(global-set-key [s-up] #'my--backward-other-window)
(global-set-key [s-down] #'other-window)
(global-set-key [s-tab] 'other-window)

Обратите внимание, что я использую superключ, поэтому, term-raw-mapвозможно, любая другая раскладка не конфликтует с моими привязками клавиш. Чтобы сделать superключ из левого Winключа, я использую .xmodmaprc:

! To load this config run:
!   $ xmodmap .xmodmaprc

! Win key.
clear mod3
clear mod4

keycode 133 = Super_L
keycode 134 = Hyper_R
add mod3 = Super_L
add mod4 = Hyper_R

Вам просто нужно запомнить 2 команды: C-c C-j- войти в обычный режим редактирования Emacs (для копирования или поиска текста в буфере), C-c C-k- вернуться в режим эмуляции терминала.

Выбор мыши и Shift-Insertработа как в xterm.

гавенкоа
источник
1
Спасибо, это золото! особенно .inputrcчасть!
cmantas
6

Как сказал Матли, это непростая задача, поскольку bash запускается с --noediting, а TAB привязан к comint-dynamic-complete.

Можно было бы повторно привязать TAB к self-insert-command в shell-comand-hook с помощью local-set-key и сделать так, чтобы режим оболочки не запускался с --noediting с помощью Mx customize-variable RET explicit-bash-args, но я подозреваю, что это не будет хорошо сочетаться с любым другим редактированием.

Возможно, вы захотите попробовать term-mode, но у него есть другой набор проблем, потому что некоторые другие обычные сочетания клавиш заменяются term-mode.

РЕДАКТИРОВАТЬ: под другими обычными ключевыми предложениями, которые обгоняют терминологический режим, я имею в виду все, кроме Cc, который становится побегом, позволяющим переключать буферы. Поэтому вместо Cx k, чтобы убить буфер, вам нужно будет Cc Cx k. Или переключиться на другой буфер 'Cc Cx o' или 'Cc Cx 2'

njsf
источник
self-insert-command - это не так: она вставляет символ TAB, а не передает нажатие клавиши TAB в Bash.
Крис Конвей,
У меня нет команды term-mode, и я не могу понять, где ее взять.
Крис Конвей,
2

Я знаю, что этому посту уже более 11 лет. Но я создал функцию для завершения собственной оболочки в Emacs. Он просто отправляет клавишу табуляции базовому процессу и перехватывает вывод, поэтому он точно такой же, как и в самой оболочке.

https://github.com/CeleritasCelery/emacs-native-shell-complete

Prgrm.celeritas
источник
Вот это да! Удивительный ! (Первое впечатление, основанное на превью.)
Мэтью Элви,
1

Я использую Prelude, и когда я нажимаю Meta + Tab, он завершается за меня.

Кроме того, Ctrl + i, похоже, делает то же самое.

Дума
источник
0

Я использую режим руля. Он имеет такую ​​функциональность (после нажатия «TAB»): введите описание изображения здесь

a_subscriber
источник
-2

Я не претендую на звание эксперта по emacs, но это должно решить вашу проблему:

Создать: ~ / .emacs

Добавьте к этому:

(требуется 'команда-оболочки) (режим-завершение-команды-оболочки)

Emacs берет на себя управление оболочкой, поэтому настройки BASH не сохраняются. Это установит автоматическое завершение для самого EMACS.

Скотт Алан Миллер
источник
Нет, это не решает проблему. Он работает только с командой оболочки, но не в режиме оболочки. Кроме того, он не включает интеллектуальное завершение, которое было запрошено в вопросе (оно будет выполнять только команды и имена файлов).
matli 02
1
режим-завершения-команд-оболочки выполняет завершение имени файла и включен по умолчанию. Мне нужна функция завершения Bash (которая расширяема, включая, например, подкоманды для apt-get и svn).
Крис Конвей,