emacs shell:用ido更改目录

时间:2014-01-06 15:13:16

标签: emacs elisp

我越来越多地使用emacs shell-mode,并且我希望可以改进一些东西:更改目录时的完成。我很乐意使用idoprojectile-find-dir

我的工作流程

截至今天,我尽可能外部的emacs外壳,尽可能使用emacs的强大功能(使用ido访问文件,使用projectile查找项目中的文件,探索里面的树,...)。

我不经常这样做。当我在一个不同的项目中工作时,我打开另一个shell缓冲区。但是当我不得不这样做的时候,我真的很想念ido或者 fasd shell实用程序(它有效,但是没有它的完成界面,这对于zsh来说很棒,并且不像使用ido那样强大是https://github.com/clvv/fasd)。

如何在elisp中连接?

我知道我们可以列出ido-completing-read;

的列表

在shell中,键入cd ../<TAB>会打开一个新的* Completions *缓冲区。它使用comint-dynamic-completion,但是如何在elisp列表中获取该列表,而不是在缓冲区中?

  • 是否可以将完成列表连接到ido? (或抛射物或舵或其他)
  • 如果你把我与准确的文件联系起来我也会很感激(有很多,很难知道什么对我有用)
  • 或者是否存在解决方案?

谢谢!

编辑:这是cd到最近访问过的目录的另一种好方法,使用fasd实用程序和ido完成:https://gitlab.com/emacs-stuff/fasd-shell/blob/master/README.org

请参阅another SO question

ps:eshell不适用于某些shell脚本,我想保持shell模式。

1 个答案:

答案 0 :(得分:3)

试试这个,这是一个快速而肮脏的黑客,在某些情况下可能会失败,但应该通常工作。还原谅我的elisp

(require 'ido)
(require 'cl-lib)
(require 'shell)

(defvar my-dir-selected nil "Flag to indicate that user has selected the directory")

(defun my-filter-cd-input (current-input)
  "Takes current user input for `cd' the a list
    whose car is the 'maximum possible directory path'
    and cdr is remaining string.

    Examples:
    '~/.emacs.d/in => ('~./emacs.d/' 'in')
    '/home/gue' => ('/home/' 'gue')
    '~/../' => ('~/../' '')"
  (let* ((unquoted-input (shell-unquote-argument current-input))
     (components (split-string unquoted-input "/"))
         (directory-parts (butlast components))
         (possible-prefix (car (last components))))
    (list (if (string= possible-prefix "")
              unquoted-input
            (concat (mapconcat 'identity directory-parts "/")
                    (when directory-parts "/")))
          possible-prefix)))

(defun my-complete-directory-name (directory current-input)
  "Prompts user for directories in `directory', `current-input'
    is the string entered by the user till now"
  (let* ((filtered-input (my-filter-cd-input current-input))
         (directory-path (car filtered-input))
         (partial-input (cadr filtered-input))
         (directory-choices (mapcar 'file-name-nondirectory
                                    (condition-case nil
                                        (cl-remove-if-not 'file-directory-p
                                                          (directory-files (concat directory directory-path) t))
                                      ('file-error (list)))))
         (selected-name (ido-completing-read "Directory: "
                                             directory-choices
                                             nil nil partial-input)))
    (comint-delete-input)
    (insert (concat "cd " 
            (shell-quote-argument (concat directory-path selected-name "/"))))))

(defun my-prompt-for-dir-or-fallback ()
  "If current shell command is `cd' prompt for directory
    using ido otherwise fallback to normal completion"
  (interactive)
  (let* ((user-input (buffer-substring-no-properties (comint-line-beginning-position)
                                                     (point-max))))
    (if (and (>= (length user-input) 3)
             (string= (substring user-input 0 3) "cd "))
        (progn 
          (setq my-dir-selected nil)
          (while (not my-dir-selected)
            (my-complete-directory-name default-directory 
                    (buffer-substring-no-properties (+ (comint-line-beginning-position) 3) 
                                    (point-max))))
          (comint-send-input))
      (call-interactively 'completion-at-point))))

(define-key shell-mode-map (kbd "<tab>") 'my-prompt-for-dir-or-fallback)

(add-hook 'ido-setup-hook 'ido-my-keys)

(defun ido-my-keys ()
  "Add my keybindings for ido."
  (define-key ido-completion-map (kbd "<C-return>") (lambda ()
                                                        (interactive)
                                                        (setq my-dir-selected t)
                                                        (ido-exit-minibuffer))))

如果当前输入的命令为<tab>,则在shell中点击cd将提示使用ido可用的目录,否则它将回退到默认完成状态,退出按C-RET