2017-06-08 11:47:56 +02:00
|
|
|
;;; core/autoload/popups.el -*- lexical-binding: t; -*-
|
2017-01-28 02:00:38 -05:00
|
|
|
|
2017-07-12 23:54:56 +02:00
|
|
|
(defvar doom-popup-remember-history)
|
|
|
|
|
2017-01-28 02:00:38 -05:00
|
|
|
;;;###autoload
|
2017-02-24 03:12:14 -05:00
|
|
|
(defun doom-popup-p (&optional target)
|
2017-03-01 22:12:30 -05:00
|
|
|
"Return TARGET (a window) if TARGET (a window or buffer) is a popup. Uses
|
|
|
|
current window if omitted."
|
2017-02-24 03:12:14 -05:00
|
|
|
(when-let (target (or target (selected-window)))
|
|
|
|
(cond ((bufferp target)
|
2017-09-24 18:41:12 +02:00
|
|
|
(and (buffer-live-p target)
|
|
|
|
(buffer-local-value 'doom-popup-mode target)))
|
2017-02-24 03:12:14 -05:00
|
|
|
((windowp target)
|
2017-09-24 18:41:12 +02:00
|
|
|
(and (window-live-p target)
|
|
|
|
(window-parameter target 'popup)
|
2017-03-01 22:12:30 -05:00
|
|
|
target)))))
|
2017-01-28 02:00:38 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
2017-02-03 19:21:33 -05:00
|
|
|
(defun doom-popup-buffer (buffer &rest plist)
|
2017-06-08 11:47:56 +02:00
|
|
|
"Display BUFFER in a shackle popup. See `shackle-rules' for possible rules.
|
|
|
|
Returns the new popup window."
|
2017-03-08 14:41:32 -05:00
|
|
|
(declare (indent defun))
|
2017-02-08 01:58:11 -05:00
|
|
|
(unless (bufferp buffer)
|
|
|
|
(error "%s is not a valid buffer" buffer))
|
2017-05-13 00:14:15 +02:00
|
|
|
(setq plist (append plist (shackle-match buffer)))
|
2017-02-08 01:58:11 -05:00
|
|
|
(shackle-display-buffer
|
|
|
|
buffer
|
|
|
|
nil (or plist (shackle-match buffer))))
|
2017-01-28 02:00:38 -05:00
|
|
|
|
2017-05-17 17:28:04 +02:00
|
|
|
;;;###autoload
|
|
|
|
(defun doom-popup-switch-to-buffer (buffer)
|
2017-05-19 03:01:49 +02:00
|
|
|
"Switch the current (or closest) pop-up window to BUFFER."
|
2017-05-17 17:28:04 +02:00
|
|
|
(unless (doom-popup-p)
|
|
|
|
(let ((popups (doom-popup-windows)))
|
|
|
|
(unless popups
|
|
|
|
(error "No popups to switch"))
|
|
|
|
(select-window (car popups))))
|
|
|
|
(set-window-dedicated-p nil nil)
|
|
|
|
(switch-to-buffer buffer nil t)
|
|
|
|
(prog1 (selected-window)
|
|
|
|
(set-window-dedicated-p nil t)))
|
|
|
|
|
2017-01-28 02:00:38 -05:00
|
|
|
;;;###autoload
|
2017-02-03 19:21:33 -05:00
|
|
|
(defun doom-popup-file (file &rest plist)
|
2017-01-28 02:00:38 -05:00
|
|
|
"Display FILE in a shackle popup, with PLIST rules. See `shackle-rules' for
|
|
|
|
possible rules."
|
|
|
|
(unless (file-exists-p file)
|
|
|
|
(user-error "Can't display file in popup, it doesn't exist: %s" file))
|
|
|
|
(doom-popup-buffer (find-file-noselect file t) plist))
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun doom-popup-windows ()
|
2017-05-19 03:01:49 +02:00
|
|
|
"Get a list of open pop up windows."
|
2017-07-06 17:43:17 +02:00
|
|
|
(cl-remove-if-not #'doom-popup-p doom-popup-windows))
|
2017-01-28 02:00:38 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun doom/popup-restore ()
|
2017-03-01 22:14:21 -05:00
|
|
|
"Restore the last open popups. If the buffers have been killed, and
|
2017-05-14 14:37:06 +02:00
|
|
|
represented real files, they will be restored. Dead special buffers or buffers
|
|
|
|
with non-nil :autokill properties will not be.
|
2017-03-01 22:14:21 -05:00
|
|
|
|
|
|
|
Returns t if popups were restored, nil otherwise."
|
2017-01-28 02:00:38 -05:00
|
|
|
(interactive)
|
|
|
|
(unless doom-popup-history
|
|
|
|
(error "No popups to restore"))
|
2017-03-01 22:14:21 -05:00
|
|
|
(let (any-p)
|
|
|
|
(dolist (spec doom-popup-history)
|
|
|
|
(let ((buffer (get-buffer (car spec)))
|
2017-05-14 14:37:06 +02:00
|
|
|
(file (plist-get (cdr spec) :file))
|
2017-05-19 03:01:49 +02:00
|
|
|
(rules (plist-get (cdr spec) :rules))
|
|
|
|
(size (plist-get (cdr spec) :size)))
|
2017-05-14 14:37:06 +02:00
|
|
|
(when (and (not buffer) file)
|
|
|
|
(setq buffer
|
|
|
|
(if-let (buf (get-file-buffer file))
|
|
|
|
(clone-indirect-buffer (buffer-name buf) nil t)
|
|
|
|
(find-file-noselect file t))))
|
2017-05-19 03:01:49 +02:00
|
|
|
(when size
|
|
|
|
(setq rules (plist-put rules :size size)))
|
2017-04-17 02:17:10 -04:00
|
|
|
(when (and buffer (apply #'doom-popup-buffer buffer rules) (not any-p))
|
2017-03-01 22:14:21 -05:00
|
|
|
(setq any-p t))))
|
|
|
|
(when any-p
|
|
|
|
(setq doom-popup-history '()))
|
|
|
|
any-p))
|
2017-01-28 02:00:38 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
2017-02-20 00:10:34 -05:00
|
|
|
(defun doom/popup-toggle ()
|
|
|
|
"Toggle popups."
|
2017-01-28 02:00:38 -05:00
|
|
|
(interactive)
|
2017-02-20 00:10:34 -05:00
|
|
|
(when (doom-popup-p)
|
|
|
|
(if doom-popup-other-window
|
|
|
|
(select-window doom-popup-other-window)
|
|
|
|
(other-window 1)))
|
|
|
|
(if (doom-popup-windows)
|
2017-03-01 22:14:51 -05:00
|
|
|
(doom/popup-close-all t)
|
2017-02-20 00:10:34 -05:00
|
|
|
(doom/popup-restore)))
|
2017-01-28 02:00:38 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun doom/popup-close (&optional window)
|
2017-03-01 22:15:32 -05:00
|
|
|
"Find and close WINDOW if it's a popup. If WINDOW is omitted, defaults to
|
|
|
|
`selected-window'. The contained buffer is buried, unless it has an :autokill
|
|
|
|
property."
|
2017-01-28 02:00:38 -05:00
|
|
|
(interactive)
|
2017-03-01 22:15:32 -05:00
|
|
|
(when-let (window (doom-popup-p window))
|
|
|
|
(delete-window window)))
|
2017-01-28 02:00:38 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
2017-03-01 21:38:26 -05:00
|
|
|
(defun doom/popup-close-all (&optional force-p)
|
|
|
|
"Closes all open popups. If FORCE-P is non-nil, or this function is called
|
|
|
|
interactively, it will close all popups without question. Otherwise, it will
|
|
|
|
only close popups that have an :autoclose property in their rule (see
|
|
|
|
`shackle-rules')."
|
2017-01-28 02:00:38 -05:00
|
|
|
(interactive)
|
2017-05-19 03:01:49 +02:00
|
|
|
(when-let (popups (doom-popup-windows))
|
2017-06-08 11:47:56 +02:00
|
|
|
(let (success doom-popup-remember-history)
|
2017-04-17 02:17:10 -04:00
|
|
|
(setq doom-popup-history (mapcar #'doom--popup-data popups))
|
2017-05-19 03:01:49 +02:00
|
|
|
(dolist (window popups)
|
|
|
|
(when (or force-p
|
|
|
|
(called-interactively-p 'interactive)
|
|
|
|
(doom-popup-prop :autoclose window))
|
2017-05-25 12:20:51 +02:00
|
|
|
(delete-window window)
|
|
|
|
(setq success t)))
|
|
|
|
success)))
|
2017-01-28 02:00:38 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun doom/popup-close-maybe ()
|
|
|
|
"Close the current popup *if* its window doesn't have a noesc parameter."
|
|
|
|
(interactive)
|
2017-05-19 03:01:49 +02:00
|
|
|
(if (doom-popup-prop :noesc)
|
|
|
|
(call-interactively
|
|
|
|
(if (featurep 'evil)
|
|
|
|
#'evil-force-normal-state
|
|
|
|
#'keyboard-escape-quit))
|
|
|
|
(delete-window)))
|
2017-01-28 02:00:38 -05:00
|
|
|
|
2017-02-20 00:11:05 -05:00
|
|
|
;;;###autoload
|
2017-07-06 15:56:24 +02:00
|
|
|
(defun doom/popup-this-buffer ()
|
2017-02-20 00:11:05 -05:00
|
|
|
"Display currently selected buffer in a popup window."
|
2017-05-12 12:11:40 +02:00
|
|
|
(interactive)
|
|
|
|
(doom-popup-buffer (current-buffer) :align t :autokill t))
|
2017-02-20 00:11:05 -05:00
|
|
|
|
2017-05-17 21:07:41 +02:00
|
|
|
;;;###autoload
|
|
|
|
(defun doom/popup-toggle-messages ()
|
|
|
|
"Toggle *Messages* buffer."
|
|
|
|
(interactive)
|
|
|
|
(if-let (win (get-buffer-window "*Messages*"))
|
|
|
|
(doom/popup-close win)
|
|
|
|
(doom-popup-buffer (get-buffer "*Messages*"))))
|
2017-05-19 03:01:49 +02:00
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun doom-popup-prop (prop &optional window)
|
2017-06-08 11:47:56 +02:00
|
|
|
"Returns a `doom-popup-rules' PROPerty from WINDOW."
|
2017-05-19 03:01:49 +02:00
|
|
|
(or (plist-get (or (if window
|
2017-07-11 01:06:10 +02:00
|
|
|
(ignore-errors
|
|
|
|
(buffer-local-value 'doom-popup-rules
|
|
|
|
(window-buffer window)))
|
2017-05-19 03:01:49 +02:00
|
|
|
doom-popup-rules)
|
|
|
|
(window-parameter window 'popup))
|
|
|
|
prop)
|
2017-07-11 01:06:26 +02:00
|
|
|
(pcase prop
|
|
|
|
(:size shackle-default-size)
|
|
|
|
(:align shackle-default-alignment))))
|
2017-05-19 03:01:49 +02:00
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun doom-popup-side (&optional window)
|
2017-06-08 11:47:56 +02:00
|
|
|
"Return what side a popup WINDOW came from ('left 'right 'above or 'below)."
|
2017-05-19 03:01:49 +02:00
|
|
|
(let ((align (doom-popup-prop :align window)))
|
|
|
|
(when (eq align t)
|
|
|
|
(setq align shackle-default-alignment))
|
|
|
|
(when (functionp align)
|
|
|
|
(setq align (funcall align)))
|
|
|
|
align))
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun doom-popup-size (&optional window)
|
2017-06-08 11:47:56 +02:00
|
|
|
"Return the size of a popup WINDOW."
|
2017-07-11 01:08:30 +02:00
|
|
|
(pcase (doom-popup-side window)
|
|
|
|
((or 'left 'right) (window-width window))
|
|
|
|
((or 'above 'below) (window-height window))))
|
2017-05-19 03:01:49 +02:00
|
|
|
|
|
|
|
(defun doom--popup-data (window)
|
|
|
|
(when-let (buffer (window-buffer window))
|
|
|
|
`(,(buffer-name buffer)
|
|
|
|
:file ,(buffer-file-name buffer)
|
|
|
|
:rules ,(window-parameter window 'popup)
|
|
|
|
:size ,(doom-popup-size window))))
|
2017-07-04 19:54:50 +02:00
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defmacro with-popup-rules! (rules &rest body)
|
2017-07-09 02:14:45 +02:00
|
|
|
"TODO"
|
2017-07-04 19:54:50 +02:00
|
|
|
(declare (indent defun))
|
|
|
|
`(let ((old-shackle-rules shackle-rules))
|
|
|
|
,@(cl-loop for rule in rules
|
|
|
|
collect `(set! :popup ,@rule))
|
|
|
|
,@body
|
|
|
|
(setq shackle-rules old-shackle-rules)))
|
2017-07-06 17:45:11 +02:00
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun doom/other-popup (count)
|
|
|
|
"Cycle through popup windows. Like `other-window', but for popups."
|
|
|
|
(interactive "p")
|
|
|
|
(if-let (popups (if (doom-popup-p)
|
|
|
|
(cdr (memq (selected-window) doom-popup-windows))
|
|
|
|
(setq doom-popup-other-window (selected-window))
|
|
|
|
doom-popup-windows))
|
|
|
|
(select-window (nth (mod (1- count) (length popups)) popups))
|
|
|
|
(unless (eq (selected-window) doom-popup-other-window)
|
|
|
|
(when doom-popup-other-window
|
|
|
|
(select-window doom-popup-other-window t)
|
|
|
|
(cl-decf count))
|
|
|
|
(when (/= count 0)
|
|
|
|
(other-window count)))))
|