From 7d668f2c1873456ec81ae9a481189fd318b3b5d2 Mon Sep 17 00:00:00 2001 From: Chong Yidong Date: Wed, 6 Apr 2011 17:13:17 -0400 Subject: [PATCH] Reimplement list-processes in Lisp. * lisp/simple.el: Lisp reimplement of list-processes. Based on an earlier reimplementation by Leo Liu, but using tabulated-list.el. (process-menu-mode): New major mode. (list-processes--refresh, list-processes): (process-menu-visit-buffer): New functions. * lisp/files.el (save-buffers-kill-emacs): Don't assume any return value of list-processes, which is undocumented anyway. --- lisp/ChangeLog | 11 +++++++ lisp/files.el | 4 +-- lisp/simple.el | 88 +++++++++++++++++++++++++++++++++++++++++++++++++- 3 files changed, 100 insertions(+), 3 deletions(-) diff --git a/lisp/ChangeLog b/lisp/ChangeLog index 313b2e94a30..b09313ddda0 100644 --- a/lisp/ChangeLog +++ b/lisp/ChangeLog @@ -1,3 +1,14 @@ +2011-04-06 Chong Yidong + + * simple.el: Lisp reimplement of list-processes. Based on an + earlier reimplementation by Leo Liu, but using tabulated-list.el. + (process-menu-mode): New major mode. + (list-processes--refresh, list-processes): + (process-menu-visit-buffer): New functions. + + * files.el (save-buffers-kill-emacs): Don't assume any return + value of list-processes, which is undocumented anyway. + 2011-04-06 Chong Yidong * emacs-lisp/tabulated-list.el: New file. diff --git a/lisp/files.el b/lisp/files.el index 6bfb4f00d32..7d8f3ee4503 100644 --- a/lisp/files.el +++ b/lisp/files.el @@ -6146,8 +6146,8 @@ With prefix ARG, silently save all file-visiting buffers, then kill." (setq active t)) (setq processes (cdr processes))) (or (not active) - (list-processes t) - (yes-or-no-p "Active processes exist; kill them and exit anyway? ")))) + (progn (list-processes t) + (yes-or-no-p "Active processes exist; kill them and exit anyway? "))))) ;; Query the user for other things, perhaps. (run-hook-with-args-until-failure 'kill-emacs-query-functions) (or (null confirm-kill-emacs) diff --git a/lisp/simple.el b/lisp/simple.el index a414fc77a39..a9a5b50283b 100644 --- a/lisp/simple.el +++ b/lisp/simple.el @@ -2690,7 +2690,93 @@ support pty association, if PROGRAM is nil." (let ((fh (find-file-name-handler default-directory 'start-file-process))) (if fh (apply fh 'start-file-process name buffer program program-args) (apply 'start-process name buffer program program-args)))) - + +;;;; Process menu + +(defvar tabulated-list-format) +(defvar tabulated-list-entries) +(defvar tabulated-list-sort-key) +(declare-function tabulated-list-init-header "tabulated-list" ()) +(declare-function tabulated-list-print "tabulated-list" ()) + +(defvar process-menu-query-only nil) + +(define-derived-mode process-menu-mode tabulated-list-mode "Process Menu" + "Major mode for listing the processes called by Emacs." + (setq tabulated-list-format [("Process" 15 t) + ("Status" 7 t) + ("Buffer" 15 t) + ("TTY" 12 t) + ("Command" 0 t)]) + (make-local-variable 'process-menu-query-only) + (setq tabulated-list-sort-key (cons "Process" nil)) + (add-hook 'tabulated-list-revert-hook 'list-processes--refresh nil t) + (tabulated-list-init-header)) + +(defun list-processes--refresh () + "Recompute the list of processes for the Process List buffer." + (setq tabulated-list-entries nil) + (dolist (p (process-list)) + (when (or (not process-menu-query-only) + (process-query-on-exit-flag p)) + (let* ((buf (process-buffer p)) + (type (process-type p)) + (name (process-name p)) + (status (symbol-name (process-status p))) + (buf-label (if (buffer-live-p buf) + `(,(buffer-name buf) + face link + help-echo ,(concat "Visit buffer `" + (buffer-name buf) "'") + follow-link t + process-buffer ,buf + action process-menu-visit-buffer) + "--")) + (tty (or (process-tty-name p) "--")) + (cmd + (if (memq type '(network serial)) + (let ((contact (process-contact p t))) + (if (eq type 'network) + (format "(%s %s)" + (if (plist-get contact :type) + "datagram" + "network") + (if (plist-get contact :server) + (format "server on %s" + (plist-get contact :server)) + (format "connection to %s" + (plist-get contact :host)))) + (format "(serial port %s%s)" + (or (plist-get contact :port) "?") + (let ((speed (plist-get contact :speed))) + (if speed + (format " at %s b/s" speed) + ""))))) + (mapconcat 'identity (process-command p) " ")))) + (push (list p (vector name status buf-label tty cmd)) + tabulated-list-entries))))) + +(defun process-menu-visit-buffer (button) + (display-buffer (button-get button 'process-buffer))) + +(defun list-processes (&optional query-only buffer) + "Display a list of all processes. +If optional argument QUERY-ONLY is non-nil, only processes with +the query-on-exit flag set are listed. +Any process listed as exited or signaled is actually eliminated +after the listing is made. +Optional argument BUFFER specifies a buffer to use, instead of +\"*Process List\". +The return value is always nil." + (interactive) + (unless (bufferp buffer) + (setq buffer (get-buffer-create "*Process List*"))) + (with-current-buffer buffer + (process-menu-mode) + (setq process-menu-query-only query-only) + (list-processes--refresh) + (tabulated-list-print)) + (display-buffer buffer)) (defvar universal-argument-map (let ((map (make-sparse-keymap))) -- 2.39.2