Всплывающий минибуфер в центре
Вот способ сделать именно то, что вы просили: отобразить минибуфер в центре экрана .
- Есть отдельная рама для минибуфера
- Расположите его в центре
- Поднимайте этот кадр всякий раз, когда минибуфер фокусируется.
Этого относительно легко достичь, и это то, что вы получите с
oneonone.el
пакетом (как указывает @Drew). Проблема этого подхода заключается в том, что минибуфер также используется для сообщений, поэтому скрывать его не очень удобно. Не беспокоиться! Я придумал другое решение.
- Пусть второй минибуфер отображается в отдельной рамке.
- Поместите это в центр.
- Используйте второй кадр для интерактивных команд, в то время как оригинальный (первый) минибуфер используется для отображения сообщений.
Реализация
Чтобы реализовать это, нам нужно создать фрейм минибуфера при запуске emacs.
(defvar endless/popup-frame-parameters
'((name . "MINIBUFFER")
(minibuffer . only)
(height . 1)
;; Ajust this one to your preference.
(top . 200))
"Parameters for the minibuffer popup frame.")
(defvar endless/minibuffer-frame
(let ((mf (make-frame endless/popup-frame-parameters)))
(iconify-frame mf) mf)
"Frame holding the extra minibuffer.")
(defvar endless/minibuffer-window
(car (window-list endless/minibuffer-frame t))
"")
Затем мы исправляем read-from-minibuffer
использование второго минибуфера вместо минибуфера оригинального кадра.
(defmacro with-popup-minibuffer (&rest body)
"Execute BODY using a popup minibuffer."
(let ((frame-symbol (make-symbol "selected-frame")))
`(let* ((,frame-symbol (selected-frame)))
(unwind-protect
(progn
(make-frame-visible endless/minibuffer-frame)
(when (fboundp 'point-screen-height)
(set-frame-parameter
endless/minibuffer-frame
'top (point-screen-height)))
(select-frame-set-input-focus endless/minibuffer-frame 'norecord)
,@body)
(select-frame-set-input-focus ,frame-symbol)))))
(defun use-popup-minibuffer (function)
"Rebind FUNCTION so that it uses a popup minibuffer."
(let* ((back-symb (intern (format "endless/backup-%s" function)))
(func-symb (intern (format "endless/%s-with-popup-minibuffer"
function)))
(defs `(progn
(defvar ,back-symb (symbol-function ',function))
(defun ,func-symb (&rest rest)
,(format "Call `%s' with a poupup minibuffer." function)
,@(list (interactive-form function))
(with-popup-minibuffer
(apply ,back-symb rest))))))
(message "%s" defs)
(when (and (boundp back-symb) (eval back-symb))
(error "`%s' already defined! Can't override twice" back-symb))
(eval defs)
(setf (symbol-function function) func-symb)))
;;; Try at own risk.
(use-popup-minibuffer 'read-from-minibuffer)
;;; This will revert the effect.
;; (setf (symbol-function #'read-from-minibuffer) endless/backup-read-from-minibuffer)
;; (setq endless/backup-read-from-minibuffer nil)
Это не может работать абсолютно со всем, но он работал на все , что я пытался до сих пор --- find-file
, ido-switch-buffer
,
eval-expression
. Если вы делаете найти какое - либо исключение, вы можете исправить эти функции на индивидуальной основе случая, позвонив
use-popup-minibuffer
на них.
Положение рядом с точкой
Чтобы расположить эту рамку минибуфера рядом с высотой точки, просто определите что-то вроде следующей функции. Он не идеален (на самом деле, он ужасен во многих случаях), но он отлично справляется с оценкой высоты точки.
(defun point-screen-height ()
(* (/ (face-attribute 'default :height) 10) 2
(- (line-number-at-pos (point))
(line-number-at-pos (window-start)))))