Emacs: Как сохранить последний параметр, предоставленный пользователем, по умолчанию? - PullRequest
8 голосов
/ 28 сентября 2008

Я пишу интерактивную функцию, которую я хотел бы запомнить последним аргументом, предоставленным пользователем, и использовать его по умолчанию.

(defun run-rake (param)
  (interactive "sTask: ")
  (shell-command (format "rake %s" task)))

При первом вызове функции я хочу, чтобы она запоминала аргумент, предоставленный пользователем, чтобы в следующий раз, когда они вызывали функцию, они могли просто нажать клавишу ввода, и она будет использовать значение, предоставленное в предыдущий раз.

Я не могу найти это в документации - как это сделать в elisp?

Ответы [ 3 ]

8 голосов
/ 29 сентября 2008
read-from-minibuffer

- это то, что вы хотите использовать. У него есть место для переменной истории.

Вот пример кода:

(defvar run-rake-history nil "History for run-rake")
(defun run-rake (cmd)
(interactive (list (read-from-minibuffer "Task: " (car run-rake-history) nil nil 'run-rake-history)))
  (shell-command (format "rake %s " cmd)))

Очевидно, настроить под свои нужды. 'Run-rake-history - это просто переменная, которая используется для хранения истории для этого вызова' read-from-miniuffer '. Другой вариант - использовать «завершение-чтение», но это предполагает, что у вас есть список вариантов, которые вы хотите ограничить пользователем (что обычно не относится к командам, подобным оболочке).

5 голосов
/ 28 сентября 2008

Вы можете видеть, как команда compile делает это. Поднимите текст справки для команды компиляции с помощью C-h f compile, переместите курсор на имя файла, содержащего функцию, затем нажмите RETURN. Это вызовет исходный файл для compile.

По сути, существует динамическая / глобальная переменная compile-command, которая содержит последнюю команду компиляции. Emacs - однопользовательская, однопоточная система, поэтому больше не нужно. Также имейте в виду, что Elisp - очень старый школьный Лисп, и переменные имеют динамический (стек вызовов), а не лексический охват. В такой системе естественно:

(let ((compile-command "gcc -o foo foo.c frobnicate.c"))
     ...
     (compile)
     ...)

Говоря о команде compile, вы пытались использовать ее вместо собственной функции run-rake?

0 голосов
/ 28 сентября 2008

Я выяснил, как сделать это вручную, используя defvar (global), но это похоже на то, что уже должно быть предоставлено базовой библиотекой (вроде как параметр make схемы). Это просто больше кода и больше ручного, чем должно быть:

(defvar *editconf-ruby-run-rake-last-rake-task* nil)

(defun editconf-ruby-run-rake-last-rake-task (&optional new-val)
  (when new-val
    (setf *editconf-ruby-run-rake-last-rake-task* new-val))
  *editconf-ruby-run-rake-last-rake-task*)

(defun editconf-ruby-run-rake (task-name)
  "Execute rake `task-name'.  See                                                                                            
`krb-ruby-get-rakefile-path-for-current-buffer' for how the                                                                  
Rakefile is located.."
  (interactive
   (let* ((rakefile (krb-ruby-get-rakefile-path-for-current-buffer))
          (rake-tasks (krb-ruby-get-rake-tasks rakefile))
          (default-task (or (editconf-ruby-run-rake-last-rake-task)
                            (editconf-ruby-run-rake-last-rake-task (car rake-tasks)))))
     (list
      (read-string (format "Task [%s|%s]: "
                           rake-tasks
                           default-task)
                   nil nil default-task))))
  (editconf-ruby-run-rake-last-rake-task task-name)
  (let ((cmd (format "cd %s; rake %s"
                     (krb-lisp-strip-path-suffix rakefile 1)
                     task-name)))
    (message "editconf-ruby-run-rake: cmd='%s'" cmd)
    (shell-command cmd)))
...