Reputation: 377
I want to auto close the compile buffer when there is no error and no warning,but i want to show it when there is warnings. Anyone can help me? This code from emacswiki only do the first requirement. How to change it?
;; Helper for compilation. Close the compilation window if
;; there was no error at all.
(defun compilation-exit-autoclose (status code msg)
;; If M-x compile exists with a 0
(when (and (eq status 'exit) (zerop code))
;; then bury the *compilation* buffer, so that C-x b doesn't go there
(bury-buffer)
;; and delete the *compilation* window
(delete-window (get-buffer-window (get-buffer "*compilation*"))))
;; Always return the anticipated result of compilation-exit-message-function
(cons msg code))
;; Specify my function (maybe I should have done a lambda function)
(setq compilation-exit-message-function 'compilation-exit-autoclose)
Upvotes: 17
Views: 5841
Reputation: 322
@jpkotta's answer is fine for me, except that when I only have a single buffer, it keeps the opened compilation buffer. I check for this case and close the window:
(defun bury-compile-buffer-if-successful (buffer string)
"Bury a compilation buffer if succeeded without warnings."
(when (and
(buffer-live-p buffer)
(string-match "compilation" (buffer-name buffer))
(string-match "finished" string)
)
(run-with-timer 1 nil
(lambda (buf)
(bury-buffer buf)
(switch-to-prev-buffer (get-buffer-window buf) 'kill)
;; delete window if it was opened by the compilation process
;; (have two windows with the same buffer)
(when
(and (equal 2 (length (window-list)))
(eq (window-buffer (car (window-list))) (window-buffer (cadr (window-list)))))
(delete-window (selected-window))
)
)
buffer)))
(add-hook 'compilation-finish-functions 'bury-compile-buffer-if-successful)
Upvotes: 0
Reputation: 71
I've tweaked above answers with better logic and tested it, working perfectly:
(add-hook 'compilation-start-hook 'compilation-started)
(add-hook 'compilation-finish-functions 'hide-compile-buffer-if-successful)
M-x customize-variable RET auto-hide-compile-buffer-delay
): (defcustom auto-hide-compile-buffer-delay 0
"Time in seconds before auto hiding compile buffer."
:group 'compilation
:type 'number
)
compilation-num-*
for warnings and errors count in compilation buffer: (defun hide-compile-buffer-if-successful (buffer string)
(setq compilation-total-time (time-subtract nil compilation-start-time))
(setq time-str (concat " (Time: " (format-time-string "%s.%3N" compilation-total-time) "s)"))
(if
(with-current-buffer buffer
(setq warnings (eval compilation-num-warnings-found))
(setq warnings-str (concat " (Warnings: " (number-to-string warnings) ")"))
(setq errors (eval compilation-num-errors-found))
(if (eq errors 0) nil t)
)
;;If Errors then
(message (concat "Compiled with Errors" warnings-str time-str))
;;If Compiled Successfully or with Warnings then
(progn
(bury-buffer buffer)
(run-with-timer auto-hide-compile-buffer-delay nil 'delete-window (get-buffer-window buffer 'visible))
(message (concat "Compiled Successfully" warnings-str time-str))
)
)
)
(make-variable-buffer-local 'compilation-start-time)
(defun compilation-started (proc)
(setq compilation-start-time (current-time))
)
Upvotes: 1
Reputation: 9437
I use the following for compilation. It keeps the compilation buffer if there are warnings or errors, and buries it otherwise (after 1 second).
(defun bury-compile-buffer-if-successful (buffer string)
"Bury a compilation buffer if succeeded without warnings "
(when (and
(buffer-live-p buffer)
(string-match "compilation" (buffer-name buffer))
(string-match "finished" string)
(not
(with-current-buffer buffer
(goto-char (point-min))
(search-forward "warning" nil t))))
(run-with-timer 1 nil
(lambda (buf)
(bury-buffer buf)
(switch-to-prev-buffer (get-buffer-window buf) 'kill))
buffer)))
(add-hook 'compilation-finish-functions 'bury-compile-buffer-if-successful)
Upvotes: 23
Reputation: 146
jpkotta, it does work most of the times. Sometimes, even if there is a warning, it doesn't switch to compilation buffer. So I made a change to your form & it does work now:
(defun bury-compile-buffer-if-successful (buffer string)
"Bury a compilation buffer if succeeded without warnings "
(if (and
(string-match "compilation" (buffer-name buffer))
(string-match "finished" string)
(not
(with-current-buffer buffer
**(goto-char 1)**
(search-forward "warning" nil t))))
(run-with-timer 1 nil
(lambda (buf)
(bury-buffer buf)
(switch-to-prev-buffer (get-buffer-window buf) 'kill))
buffer)))
(add-hook 'compilation-finish-functions 'bury-compile-buffer-if-successful)
Upvotes: 2