diff options
author | Richard M. Stallman <rms@gnu.org> | 1994-12-23 17:58:46 +0000 |
---|---|---|
committer | Richard M. Stallman <rms@gnu.org> | 1994-12-23 17:58:46 +0000 |
commit | b9a5a6afd95c56a43a555e077781607e188d3c15 (patch) | |
tree | f0f89655617c6d77747fd8cdbc34dd4d59260028 /lisp/msb.el | |
parent | 313f3cb427b60b87f0e6249a33c6db6156d34662 (diff) | |
download | emacs-b9a5a6afd95c56a43a555e077781607e188d3c15.tar.gz |
Initial revision
Diffstat (limited to 'lisp/msb.el')
-rw-r--r-- | lisp/msb.el | 1009 |
1 files changed, 1009 insertions, 0 deletions
diff --git a/lisp/msb.el b/lisp/msb.el new file mode 100644 index 00000000000..8cee0b779bf --- /dev/null +++ b/lisp/msb.el @@ -0,0 +1,1009 @@ +;;; msb.el --- Customizable buffer-selection with multiple menus. +;; Copyright (C) 1993, 1994 Lars Lindberg <Lars.Lindberg@sypro.cap.se> +;; +;; Author: Lars Lindberg <Lars.Lindberg@sypro.cap.se> +;; Created: 8 Oct 1993 +;; $Revision: 3.21 $ +;; $Date: 1994/12/22 07:58:27 $ +;; Keywords: mouse buffer menu +;; +;; This program is free software; you can redistribute it and/or modify +;; it under the terms of the GNU General Public License as published by +;; the Free Software Foundation; either version 2 of the License, or +;; (at your option) any later version. +;; +;; This program is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; GNU General Public License for more details. +;; +;; You should have received a copy of the GNU General Public License +;; along with this program; if not, write to the Free Software +;; Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. + +;; LCD Archive Entry: +;; msb|Lars Lindberg|Lars.Lindberg@sypro.cap.se| +;; Choose buffer with the mouse. +;; $Date: 1994/12/22 07:58:27 $|$Revision: 3.21 $|~/packages/msb.el.Z| + +;;; Commentary: +;; +;; Purpose of this package: +;; 1. Offer a function for letting the user choose buffer, +;; not necessarily for switching to it. +;; 2. Make a better mouse-buffer-menu. +;; +;; Installation: +;; (require 'msb) +;; Note! You now use msb instead of mouse-buffer-menu. +;; +;; Now try c-mouse-down-1 (Press <CTRL> and mouse button 1 at the +;; same time). +;; +;; Customization: +;; Look at the variable 'msb-menu-cond' for deciding what menus you +;; want. It's not that hard to customize, despite my not-so-good +;; doc-string. Feel free to send me a better doc-string. +;; There are some constants for you to try here: +;; msb--few-menus +;; msb--very-many-menus (default) +;; +;; Look at the variable 'msb-item-handling-function' for customization +;; of the appearance of every menu item. Try for instance setting +;; it to 'msb-alon-item-handler. +;; +;; Look at the variable 'msb-item-sort-function' for customization +;; of sorting the menus. Set it to t for instance, which means no +;; sorting - you will get latest used buffer first. +;; +;; Also check out the variable 'msb-display-invisible-buffers-p' + +;; Known bugs: +;; - `msb' does not work on a non-X-toolkit Emacs. +;; Future enhancements: +;; - [Mattes] had a suggestion about sorting files by extension. +;; I (Lars Lindberg) think this case could be solved if msb.el was +;; rewritten to handle more dynamic splitting. It's now completely +;; static, depending on the menu-cond. If the splitting could also +;; be done by a user-defined function a lot of cases would be +;; solved. +;; - [Jim] suggested that the Frame menu became a part of the buffer menu. + +;;; Change Log +;; 3.21 22/12-94 +;; Fixed bug that occured in non X-toolkit versions of Emacs. +;; [Chalupsky] pointed out that a global `save-match-data' is +;; necessary. +;; Bug found. Thanks [kifer]. +;; 3.16 20/12-94 +;; Added separators to the menu. New variable `msb-separator-diff'. +;; New variable `msb-files-by-directory-sort-key'. +;; Removed `msb--many-menus.' +;; Fixed bugs. +;; 3.13 20/12-94 +;; Menu fix for non-X-toolkit Emacsen and new "process" +;; menu. Thanks [jim]. +;; Bug for 'files-by-type'. +;; Restored the call to `msb-after-load-hooks'. Thanks [larry]. +;; Major fixes by [Ake]. +;; Menu-bar buffer-menu now has menu-toggle at top level. +;; 3.6 16/12-94 +;; Added variable `msb-max-file-menu-items'. +;; Removed a large part of the change log. +;; Found bug. +;; 3.3 16/12-94 +;; Found bugs. +;; 3.1 16/12-94 +;; Now has two types of menus - "files by directory" and "files +;; by type". +;; Added variable `msb-files-by-directory'. +;; Fixed a number of bugs for older versions. +;; 2.11 16/12-94 +;; Added 'no-multi to msb-menu-cond. +;; Added possibility to shift the menu leftwards. Thanks [kifer]. +;; 2.8 15/12-94 +;; Now aware of earlier versions of Emacs that doesn't have the +;; function `frame-or-buffer-changed-p' or the variable +;; `menu-bar-update-hook'. Thanks [will]. +;; 2.7 14/12-94 +;; Better installation. +;; 2.6 14/12-94 +;; Now only makes up the menu when necessary. +;; Added menu-bar support. +;; Now handles errors in msb-menu-cond better. Thanks [jaalto]. +;; Added MH-awareness. Thanks [kifer]. +;; Added autoload statements. +;; 2.3 8/12-94 +;; Now uses RCS version numbering for msb.el version number. +;; Submitted this to LCD. + +;;; Thanks goes to +;; [msb] - Mark Brader <msb@sq.com> +;; [Chalupsky] - Hans Chalupsky <hans@cs.Buffalo.EDU> +;; [jim] - Jim Berry <m1jhb00@FRB.GOV> +;; [larry] - Larry Rosenberg <ljr@ictv.com> +;; [will] - Will Henney <will@astroscu.unam.mx> +;; [jaalto] - Jari Aalto <jaalto@tre.tele.nokia.fi> +;; [kifer] - Michael Kifer <kifer@sbkifer.cs.sunysb.edu> +;; [Gael] - Gael Marziou <gael@gnlab030.grenoble.hp.com> +;; [Gillespie] - Dave Gillespie <daveg@thymus.synaptics.com> +;; [Alon] - Alon Albert <alon@milcse.rtsg.mot.com> +;; [KevinB] - Kevin Broadey, <KevinB@bartley.demon.co.uk> +;; [Ake] - Ake Stenhof <ake@cadpoint.se> +;; [RMS] - Richard Stallman <rms@gnu.ai.mit.edu> +;; [Fisk] - Steve Fisk <fisk@medved.bowdoin.edu> + +;;; Code: + +(require 'cl) + +;;; +;;; Some example constants to be used for 'msb-menu-cond'. See that +;;; variable for more information. Please note that if the condition +;;; returns 'multi, then the buffer can appear in several menus. +;;; +(defconst msb--few-menus + '(((and (boundp 'server-buffer-clients) + server-buffer-clients + 'multi) + 3030 + "Clients (%d)") + ((and msb-display-invisible-buffers-p + (msb-invisible-buffer-p) + 'multi) + 3090 + "Invisible buffers (%d)") + ((eq major-mode 'dired-mode) + 2010 + "Dired (%d)" + msb-dired-item-handler + msb-sort-by-directory) + ((eq major-mode 'Man-mode) + 4090 + "Manuals (%d)") + ((eq major-mode 'w3-mode) + 4020 + "WWW (%d)") + ((or (memq major-mode '(rmail-mode vm-summary-mode vm-mode mail-mode)) + (memq major-mode '(mh-letter-mode + mh-show-mode + mh-folder-mode)) + (memq major-mode '(gnus-summary-mode + news-reply-mode + gnus-group-mode + gnus-article-mode + gnus-kill-file-mode + gnus-browse-killed-mode))) + 4010 + "Mail (%d)") + ((not buffer-file-name) + 4099 + "Buffers (%d)") + ('no-multi + 1099 + "Files (%d)"))) + +(defconst msb--very-many-menus + '(((and (boundp 'server-buffer-clients) + server-buffer-clients + 'multi) + 1010 + "Clients (%d)") + ((and (boundp 'vc-mode) vc-mode 'multi) + 1020 + "Version Control (%d)") + ((and buffer-file-name + (buffer-modified-p) + 'multi) + 1030 + "Changed files (%d)") + ((and (get-buffer-process (current-buffer)) + 'multi) + 1040 + "Processes (%d)") + ((and msb-display-invisible-buffers-p + (msb-invisible-buffer-p) + 'multi) + 1090 + "Invisible buffers (%d)") + ((eq major-mode 'dired-mode) + 2010 + "Dired (%d)" + ;; Note this different menu-handler + msb-dired-item-handler + ;; Also note this item-sorter + msb-sort-by-directory) + ((eq major-mode 'Man-mode) + 4030 + "Manuals (%d)") + ((eq major-mode 'w3-mode) + 4020 + "WWW (%d)") + ((or (memq major-mode '(rmail-mode vm-summary-mode vm-mode mail-mode)) + (memq major-mode '(mh-letter-mode + mh-show-mode + mh-folder-mode)) + (memq major-mode '(gnus-summary-mode + news-reply-mode + gnus-group-mode + gnus-article-mode + gnus-kill-file-mode + gnus-browse-killed-mode))) + 4010 + "Mail (%d)") + ;; Catchup for all non-file buffers + ((and (not buffer-file-name) + 'no-multi) + 4099 + "Other non-file buffers (%d)") + ((and (string-match "/\\.[^/]*$" buffer-file-name) + 'multi) + 3090 + "Hidden Files (%d)") + ((memq major-mode '(c-mode c++-mode)) + 3010 + "C/C++ Files (%d)") + ((eq major-mode 'emacs-lisp-mode) + 3020 + "Elisp Files (%d)") + ((eq major-mode 'latex-mode) + 3030 + "LaTex Files (%d)") + ('no-multi + 3099 + "Other files (%d)"))) + +;; msb--many-menus is obsolete +(defvar msb--many-menus msb--very-many-menus) + +;;; +;;; Customizable variables +;;; + +(defvar msb-separator-diff 100 + "*Non-nil means use separators. +The separators will appear between all menus that have a sorting key that differs by this value or more.") + +(defvar msb-files-by-directory-sort-key 0 + "*The sort key for files sorted by directory") + +(defvar msb-max-menu-items 25 + "*The maximum number of items in a menu. +If this variable is set to 15 for instance, then the 15 latest used +buffer that fits in a certain submenu will appear in that submenu. +Nil means no limit.") + +(defvar msb-max-file-menu-items 10 + "*The maximum number of items from different directories. + +When the menu is of type 'file by directory', this is the maximum +number of buffers that are clumped togehter from different +directories. + +If the value is not a number, then the value 10 is used.") + +(defvar msb-most-recently-used-sort-key -1010 + "*Where should the menu with the most recently used buffers be placed?") + +(defvar msb-display-most-recently-used t + "*How many buffers should be in the most-recently-used menu. +No buffers at all if less than 1 or nil. +T means use the value of `msb-max-menu-items' in the way it is defined.") + +(defvar msb-most-recently-used-title "Most recently used (%d)" + "*The title for the most-recently-used menu.") + +(defvar msb-horizontal-shift-function '(lambda () 0) + "*Function that specifies a number of pixels by which the top menu should +be shifted leftwards.") + +(defvar msb-display-invisible-buffers-p nil + "*Show invisible buffers or not. +Non-nil means that the buffer menu should include buffers that have +names that starts with a space character.") + +(defvar msb-item-handling-function 'msb-item-handler + "*The appearance of a buffer menu. + +The default function to call for handling the appearance of a menu +item. It should take to arguments, BUFFER and MAX-BUFFER-NAME-LENGTH, +where the latter is the max length of all buffer names. +When the function is called, BUFFER is the current buffer. +This function is called for items in the variable 'msb-menu-cond' that +have nil as ITEM-HANDLING-FUNCTION. See 'msb-menu-cond' for more +information.") + +(defvar msb-item-sort-function 'msb-sort-by-name + "*The order of items in a buffer menu. +The default function to call for handling the order of items in a menu +item. This function is called like a sort function. The items +look like (ITEM-NAME . BUFFER). +ITEM-NAME is the name of the item that will appear in the menu. +BUFFER is the buffer, this is not necessarily the current buffer. + +Set this to nil or t if you don't want any sorting (faster).") + +(defvar msb-files-by-directory nil + "*Non-nil means that files should be sorted by directory instead of +the groups in msb-menu-cond.") + +(defvar msb-menu-cond msb--very-many-menus + "*List of criterias for splitting the mouse buffer menu. +The elements in the list should be of this type: + (CONDITION MENU-SORT-KEY MENU-TITLE ITEM-HANDLING-FN ITEM-SORT-FN). + +When making the split, the buffers are tested one by one against the +CONDITION, just like a lisp cond: When hitting a true condition, the +other criterias are *not* tested and the buffer name will appear in +the menu with the menu-title corresponding to the true condition. + +If the condition returns the symbol 'multi, then the buffer will be +added to this menu *and* tested for other menus too. If it returns +'no-multi, then the buffer will only be added if it hasn't been added +to any other menu. + +During this test, the buffer in question is the current buffer, and +the test is surrounded by calls to `save-excursion' and +`save-match-data' + +The categories are sorted by MENU-SORT-KEY. Smaller keys are on +top. nil means don't display this menu. + +MENU-TITLE is really a format. If you add %d in it, the %d is replaced +with the number of items in that menu. + +ITEM-HANDLING-FN, is optional. If it is supplied and is a +function, than it is used for displaying the items in that particular +buffer menu, otherwise the function pointed out by +'msb-item-handling-function' is used. + +ITEM-SORT-FN, is also optional. +If it is not supplied, the function pointed out by +'msb-item-sort-function' is used. +If it is nil, then no sort takes place and the buffers are presented +in least-recently-used order. +If it is t, then no sort takes place and the buffers are presented in +most-recently-used order. +If it is supplied and non-nil and not t than it is used for sorting +the items in that particular buffer menu. + +Note1: There should always be a 'catch-all' as last element, +in this list. That is an element like (t TITLE ITEM-HANDLING-FUNCTION). +Note2: A buffer menu appears only if it has at least one buffer in it. +Note3: If you have a CONDITION that can't be evaluated you will get an +error every time you do \\[msb].") + +(defvar msb-after-load-hooks nil + "Hooks to be run after the msb package has been loaded.") + +;;; +;;; Internal variables +;;; + +;; The last calculated menu. +(defvar msb--last-buffer-menu nil) + +;; If this is non-nil, then it is a string that describes the error. +(defvar msb--error nil) + +;;; +;;; Some example function to be used for 'msb-item-sort-function'. +;;; +(defun msb-item-handler (buffer &optional maxbuf) + "Create one string item, concerning BUFFER, for the buffer menu. +The item looks like: +*% <buffer-name> +The '*' appears only if the buffer is marked as modified. +The '%' appears only if the buffer is read-only. +Optional second argument MAXBUF is completely ignored." + (let ((name (buffer-name)) + (modified (if (buffer-modified-p) "*" " ")) + (read-only (if buffer-read-only "%" " "))) + (format "%s%s %s" modified read-only name))) + + +(eval-when-compile (require 'dired)) + +;; 'dired' can be called with a list of the form (directory file1 file2 ...) +;; which causes 'dired-directory' to be in the same form. +(defun msb--dired-directory () + (cond ((stringp dired-directory) + (abbreviate-file-name (expand-file-name dired-directory))) + ((consp dired-directory) + (abbreviate-file-name (expand-file-name (car dired-directory)))) + (t + (error "Unknown type of 'dired-directory' in buffer %s" + (buffer-name))))) + +(defun msb-dired-item-handler (buffer &optional maxbuf) + "Create one string item, concerning a dired BUFFER, for the buffer menu. +The item looks like: +*% <buffer-name> +The '*' appears only if the buffer is marked as modified. +The '%' appears only if the buffer is read-only. +Optional second argument MAXBUF is completely ignored." + (let ((name (msb--dired-directory)) + (modified (if (buffer-modified-p) "*" " ")) + (read-only (if buffer-read-only "%" " "))) + (format "%s%s %s" modified read-only name))) + +(defun msb-alon-item-handler (buffer maxbuf) + "Create one string item for the buffer menu. +The item looks like: +<buffer-name> *%# <file-name> +The '*' appears only if the buffer is marked as modified. +The '%' appears only if the buffer is read-only. +The '#' appears only version control file (SCCS/RCS)." + (format (format "%%%ds %%s%%s%%s %%s" maxbuf) + (buffer-name buffer) + (if (buffer-modified-p) "*" " ") + (if buffer-read-only "%" " ") + (if (and (boundp 'vc-mode) vc-mode) "#" " ") + (or buffer-file-name ""))) + +;;; +;;; Some example function to be used for 'msb-item-handling-function'. +;;; +(defun msb-sort-by-name (item1 item2) + "Sorts the items depending on their buffer-name +An item look like (NAME . BUFFER)." + (string-lessp (buffer-name (cdr item1)) + (buffer-name (cdr item2)))) + + +(defun msb-sort-by-directory (item1 item2) + "Sorts the items depending on their directory. Made for dired. +An item look like (NAME . BUFFER)." + (string-lessp (save-excursion (set-buffer (cdr item1)) (msb--dired-directory)) + (save-excursion (set-buffer (cdr item2)) (msb--dired-directory)))) + +;;; +;;; msb +;;; +;;; This function can be used instead of (mouse-buffer-menu EVENT) +;;; function in "mouse.el". +;;; +(defun msb (event) + "Pop up several menus of buffers for selection with the mouse. +This command switches buffers in the window that you clicked on, and +selects that window. + +See the function 'mouse-select-buffer' and the variable +'msb-menu-cond' for more information about how the menus are split." + (interactive "e") + (let ((buffer (mouse-select-buffer event)) + (window (posn-window (event-start event)))) + (cond + (buffer + (or (framep window) (select-window window)) + (switch-to-buffer (car (cdr buffer)))))) + nil) + +;;; +;;; Some supportive functions +;;; +(defun msb-invisible-buffer-p (&optional buffer) + "Return t if optional BUFFER is an \"invisible\" buffer. +If the argument is left out or nil, then the current buffer is considered." + (and (> (length (buffer-name buffer)) 0) + (eq ?\ (aref (buffer-name buffer) 0)))) + +;; Strip one hierarcy level from the end of PATH. +(defun msb--strip-path (path) + (save-match-data + (if (string-match "\\(.+\\)/[^/]+$" path) + (substring path (match-beginning 1) (match-end 1)) + "/"))) + +;; Create an alist with all buffers from LIST that lies under the same +;; directory will be in the same item as the directory string as +;;'((PATH1 . (BUFFER-1 BUFFER-2 ...)) (PATH2 . (BUFFER-K BUFFER-K+1...)) ...) +(defun msb--init-file-alist (list) + (let ((buffer-alist + (sort (mapcan + (function + (lambda (buffer) + (let ((file-name (buffer-file-name buffer))) + (when file-name + (list (cons (msb--strip-path file-name) buffer)))))) + list) + (function (lambda (item1 item2) + (string< (car item1) (car item2))))))) + ;; Make alist that looks like + ;;'((PATH1 . (BUFFER-1 BUFFER-2 ...)) (PATH2 . (BUFFER-K)) ...) + (let ((path nil) + (buffers nil) + (result nil)) + (append + (mapcan (function + (lambda (item) + (cond + ((and path + msb-max-menu-items + (< (length buffers) msb-max-menu-items) + (string= path (car item))) + (push (cdr item) buffers) + nil) + (t + (when path + (setq result (cons path buffers))) + (setq path (car item)) + (setq buffers (list (cdr item))) + (and result (list result)))))) + buffer-alist) + (list (cons path buffers)))))) + +;; Choose file-menu with respect to directory for every buffer in LIST. +(defun msb--choose-file-menu (list) + (let ((buffer-alist (msb--init-file-alist list)) + (final-list nil) + (max-clumped-together (if (numberp msb-max-file-menu-items) + msb-max-file-menu-items + 10)) + (top-found-p nil) + (last-path nil) + first rest path buffers) + (setq first (car buffer-alist)) + (setq rest (cdr buffer-alist)) + (setq path (car first)) + (setq buffers (cdr first)) + (while rest + (let ((found-p nil) + (tmp-rest rest) + new-path item) + (setq item (car tmp-rest)) + (while (and tmp-rest + (<= (length buffers) max-clumped-together) + (>= (length (car item)) (length path)) + (string= path (substring (car item) 0 (length path)))) + (setq found-p t) + (setq buffers (append buffers (cdr item))) + (setq tmp-rest (cdr tmp-rest)) + (setq item (car tmp-rest))) + (cond + ((> (length buffers) max-clumped-together) + (setq last-path (car first)) + (when top-found-p + (setq first (cons (concat (car first) "/...") + (cdr first))) + (setq top-found-p nil)) + (push first final-list) + (setq first (car rest) + rest (cdr rest)) + (setq path (car first) + buffers (cdr first))) + (t + (when found-p + (setq top-found-p t) + (setq first (cons path buffers) + rest tmp-rest)) + (setq path (msb--strip-path path) + buffers (cdr first)) + (when (and last-path + (or (and (>= (length path) (length last-path)) + (string= last-path + (substring path 0 (length last-path)))) + (and (< (length path) (length last-path)) + (string= path + (substring last-path 0 (length path)))))) + + (when top-found-p + (setq first (cons (concat (car first) "/...") + (cdr first))) + (setq top-found-p nil)) + (push first final-list) + (setq first (car rest) + rest (cdr rest)) + (setq path (car first) + buffers (cdr first))))))) + (when top-found-p + (setq first (cons (concat (car first) + (if (string-match "/$" (car first)) + "..." + "/...")) + (cdr first))) + (setq top-found-p nil)) + (push first final-list) + (nreverse final-list))) + +;; Create a vector as: +;; [BUFFER-LIST-VARIABLE CONDITION MENU-SORT-KEY MENU-TITLE ITEM-HANDLER SORTER) +;; from an element in 'msb-menu-cond'. See that variable for a +;; description of it's elements. +(defun msb--create-function-info (menu-cond-elt) + (let* ((list-symbol (make-symbol "-msb-buffer-list")) + (tmp-ih (and (> (length menu-cond-elt) 3) + (nth 3 menu-cond-elt))) + (item-handler (if (and tmp-ih (fboundp tmp-ih)) + tmp-ih + msb-item-handling-function)) + (tmp-s (if (> (length menu-cond-elt) 4) + (nth 4 menu-cond-elt) + msb-item-sort-function)) + (sorter (if (or (fboundp tmp-s) + (null tmp-s) + (eq tmp-s 't)) + tmp-s + msb-item-sort-function))) + (when (< (length menu-cond-elt) 3) + (error "Wrong format of msb-menu-cond.")) + (when (and (> (length menu-cond-elt) 3) + (not (fboundp tmp-ih))) + (signal 'invalid-function (list tmp-ih))) + (when (and (> (length menu-cond-elt) 4) + tmp-s + (not (fboundp tmp-s)) + (not (eq tmp-s 't))) + (signal 'invalid-function (list tmp-s))) + (set list-symbol '()) + (vector list-symbol ;BUFFER-LIST-VARIABLE + (nth 0 menu-cond-elt) ;CONDITION + (nth 1 menu-cond-elt) ;SORT-KEY + (nth 2 menu-cond-elt) ;MENU-TITLE + item-handler ;ITEM-HANDLER + sorter) ;SORTER + )) + +;; This defsubst is only used in `msb--choose-menu' below. It was +;; pulled out merely to make the code somewhat clearer. The indention +;; level was too big. +(defsubst msb--collect (function-info-vector) + (let ((result nil) + (multi-flag nil) + function-info-list) + (setq function-info-list + (loop for fi + across function-info-vector + if (and (setq result + (eval (aref fi 1))) ;Test CONDITION + (not (and (eq result 'no-multi) + multi-flag)) + (progn (when (eq result 'multi) + (setq multi-flag t)) + t) + (or (not msb-max-menu-items) + (< (length (eval (aref fi 0))) + msb-max-menu-items))) + collect fi + until (and result + (not (eq result 'multi))))) + (when (and (not function-info-list) + (not result)) + (error "No catch-all in msb-menu-cond!")) + function-info-list)) + +;; Adds BUFFER to the menu depicted by FUNCTION-INFO +;; All side-effects. Adds an element of type '(BUFFER-TITLE . BUFFER) +;; to the buffer-list variable in function-info. +(defun msb--add-to-menu (buffer function-info max-buffer-name-length) + (let ((list-symbol (aref function-info 0))) ;BUFFER-LIST-VARIABLE + ;; Here comes the hairy side-effect! + (set list-symbol + (cons (cons (funcall (aref function-info 4) ;ITEM-HANDLER + buffer + max-buffer-name-length) + buffer) + (eval list-symbol))))) + +;; Selects the appropriate menu for BUFFER. +;; This is all side-effects, folks! +;; This should be optimized. +(defsubst msb--choose-menu (buffer function-info-vector max-buffer-name-length) + (unless (and (not msb-display-invisible-buffers-p) + (msb-invisible-buffer-p buffer)) + (condition-case nil + (save-excursion + (set-buffer buffer) + ;; Menu found. Add to this menu + (mapc (function + (lambda (function-info) + (msb--add-to-menu buffer function-info max-buffer-name-length))) + (msb--collect function-info-vector))) + (error (unless msb--error + (setq msb--error + (format + "Variable `msb-menu-cond': Error for buffer \"%s\"." + (buffer-name buffer))) + (error msb--error)))))) + +;; Return (SORT-KEY TITLE . BUFFER-LIST) or nil if the +;; buffer-list is empty. +(defun msb--create-sort-item (function-info) + (let ((buffer-list (eval (aref function-info 0)))) + (when buffer-list + (let ((sorter (aref function-info 5)) ;SORTER + (sort-key (aref function-info 2))) ;MENU-SORT-KEY + (when sort-key + (cons sort-key + (cons (format (aref function-info 3) ;MENU-TITLE + (length buffer-list)) + (cond + ((null sorter) + buffer-list) + ((eq sorter 't) + (nreverse buffer-list)) + (t + (sort buffer-list sorter)))))))))) + +;; Returns a list on the form ((TITLE . BUFFER-LIST)) for +;; the most recently used buffers. +(defun msb--most-recently-used-menu (max-buffer-name-length) + (when (and msb-display-most-recently-used + (or (not (numberp msb-display-most-recently-used)) + (> msb-display-most-recently-used 0))) + (let* ((max-in-menu + (if (numberp msb-display-most-recently-used) + msb-display-most-recently-used + msb-max-menu-items)) + + (most-recently-used + (loop with n = 0 + for buffer in (cdr (buffer-list)) + if (save-excursion + (set-buffer buffer) + (and (not (msb-invisible-buffer-p)) + (not (eq major-mode 'dired-mode)))) + collect (save-excursion + (set-buffer buffer) + (cons (funcall msb-item-handling-function + buffer + max-buffer-name-length) + buffer)) + and do (incf n) + until (and max-in-menu (>= n max-in-menu))))) + (cons (if (stringp msb-most-recently-used-title) + (format msb-most-recently-used-title + (length most-recently-used)) + (signal 'wrong-type-argument (list msb-most-recently-used-title))) + most-recently-used)))) + +(defun msb--create-buffer-menu-2 () + (let ((max-buffer-name-length 0) + file-buffers + function-info-vector) + ;; Calculate the longest buffer name. + (mapc + (function + (lambda (buffer) + (if (or msb-display-invisible-buffers-p + (not (msb-invisible-buffer-p))) + (setq max-buffer-name-length + (max max-buffer-name-length + (length (buffer-name buffer))))))) + (buffer-list)) + ;; Make a list with elements of type + ;; (BUFFER-LIST-VARIABLE + ;; CONDITION + ;; MENU-SORT-KEY + ;; MENU-TITLE + ;; ITEM-HANDLER + ;; SORTER) + ;; Uses "function-global" variables: + ;; function-info-vector + (setq function-info-vector + (apply (function vector) + (mapcar (function msb--create-function-info) + msb-menu-cond))) + ;; Split the buffer-list into several lists; one list for each + ;; criteria. This is the most critical part with respect to time. + (mapc (function (lambda (buffer) + (cond ((and msb-files-by-directory + (buffer-file-name buffer)) + (push buffer file-buffers)) + (t + (msb--choose-menu buffer + function-info-vector + max-buffer-name-length))))) + (buffer-list)) + (when file-buffers + (setq file-buffers + (mapcar (function + (lambda (buffer-list) + (cons msb-files-by-directory-sort-key + (cons (car buffer-list) + (sort + (mapcar (function + (lambda (buffer) + (cons (buffer-name buffer) + buffer))) + (cdr buffer-list)) + (function + (lambda (item1 item2) + (string< (car item1) (car item2))))))))) + (msb--choose-file-menu file-buffers)))) + ;; Now make the menu - a list of (TITLE . BUFFER-LIST) + (let* ((buffers (buffer-list)) + menu + (most-recently-used + (msb--most-recently-used-menu max-buffer-name-length)) + (others (append file-buffers + (loop for elt + across function-info-vector + for value = (msb--create-sort-item elt) + if value collect value)))) + (setq menu + (mapcar 'cdr ;Remove the SORT-KEY + ;; Sort the menus - not the items. + (msb--add-separators + (sort + ;; Get a list of (SORT-KEY TITLE . BUFFER-LIST) + ;; Also sorts the items within the menus. + (if (cdr most-recently-used) + (cons + ;; Add most recent used buffers + (cons msb-most-recently-used-sort-key + most-recently-used) + others) + others) + (function (lambda (elt1 elt2) + (< (car elt1) (car elt2)))))))) + ;; Now make it a keymap menu + (append + '(keymap "Select Buffer") + (msb--make-keymap-menu menu) + (when msb-separator-diff + (list (list 'separator "---"))) + (list (cons 'toggle + (cons + (if msb-files-by-directory + "*Files by type*" + "*Files by directory*") + 'msb--toggle-menu-type))))))) + +(defun msb--create-buffer-menu () + (save-match-data + (save-excursion + (msb--create-buffer-menu-2)))) + +;;; +;;; Multi purpose function for selecting a buffer with the mouse. +;;; +(defun msb--toggle-menu-type () + (interactive) + (setq msb-files-by-directory (not msb-files-by-directory)) + (menu-bar-update-buffers t)) + +(defun mouse-select-buffer (event) + "Pop up several menus of buffers, for selection with the mouse. +Returns the selected buffer or nil if no buffer is selected. + +The way the buffers are splitted is conveniently handled with the +variable 'msb-menu-cond'." + ;; Popup the menu and return the selected buffer. + (when (or msb--error + (not msb--last-buffer-menu) + (not (fboundp 'frame-or-buffer-changed-p)) + (frame-or-buffer-changed-p)) + (setq msb--error nil) + (setq msb--last-buffer-menu (msb--create-buffer-menu))) + (let ((position event)) + (when (and (fboundp 'posn-x-y) + (fboundp 'posn-window)) + (let ((posX (car (posn-x-y (event-start event)))) + (posY (cdr (posn-x-y (event-start event)))) + (posWind (posn-window (event-start event))) + name) + ;; adjust position + (setq posX (- posX (funcall msb-horizontal-shift-function)) + position (list (list posX posY) posWind)))) + (setq name (x-popup-menu position msb--last-buffer-menu)) + ;; If toggle bring up the + (cond + ((eq (car name) 'toggle) + (msb--toggle-menu-type) + (mouse-select-buffer event)) + ((and (numberp (car name)) + (null (cdr name))) + (let ((msb--last-buffer-menu (nthcdr 3 (assq (car name) msb--last-buffer-menu)))) + (mouse-select-buffer event))) + ((and (stringp (car name)) + (null (cdr name))) + (cons nil name)) + (t + name)))) + +;; Add separators +(defun msb--add-separators (sorted-list) + (cond + ((or (not msb-separator-diff) + (not (numberp msb-separator-diff))) + sorted-list) + (t + (let ((last-key nil)) + (mapcan + (function + (lambda (item) + (cond + ((and msb-separator-diff + last-key + (> (- (car item) last-key) + msb-separator-diff)) + (setq last-key (car item)) + (list (cons last-key 'separator) + item)) + (t + (setq last-key (car item)) + (list item))))) + sorted-list))))) + +(defun msb--make-keymap-menu (raw-menu) + (let ((end (cons '(nil) 'menu-bar-select-buffer)) + (mcount 0)) + (mapcar + (function + (lambda (sub-menu) + (cond + ((eq 'separator sub-menu) + (list 'separator "---")) + (t + (append (list (incf mcount) (car sub-menu) + 'keymap (car sub-menu)) + (mapcar (function + (lambda (item) + (let ((string (car item)) + (buffer (cdr item))) + (cons (buffer-name buffer) + (cons string end))))) + (cdr sub-menu))))))) + raw-menu))) + +(defun menu-bar-update-buffers (&optional arg) + ;; If user discards the Buffers item, play along. + (when (and (lookup-key (current-global-map) [menu-bar buffer]) + (or (not (fboundp 'frame-or-buffer-changed-p)) + (frame-or-buffer-changed-p) + arg)) + (let ((buffers (buffer-list)) + (frames (frame-list)) + buffers-menu frames-menu) + ;; If requested, list only the N most recently selected buffers. + (when (and (integerp buffers-menu-max-size) + (> buffers-menu-max-size 1) + (> (length buffers) buffers-menu-max-size)) + (setcdr (nthcdr buffers-menu-max-size buffers) nil)) + ;; Make the menu of buffers proper. + (setq msb--last-buffer-menu (msb--create-buffer-menu)) + (setq buffers-menu msb--last-buffer-menu) + ;; Make a Frames menu if we have more than one frame. + (if (cdr frames) + (setq frames-menu + (cons "Select Frame" + (mapcar + (function + (lambda (frame) + (nconc + (list frame + (cdr (assq 'name + (frame-parameters frame))) + (cons nil nil)) + 'menu-bar-select-frame))) + frames)))) + (when frames-menu + (setq frames-menu (cons 'keymap frames-menu))) + (define-key (current-global-map) [menu-bar buffer] + (cons "Buffers" + (if (and buffers-menu frames-menu) + (list 'keymap "Buffers and Frames" + (cons 'buffers (cons "Buffers" buffers-menu)) + (cons 'frames (cons "Frames" frames-menu))) + (or buffers-menu frames-menu 'undefined))))))) + +(when (and (boundp 'menu-bar-update-hook) + (not (fboundp 'frame-or-buffer-changed-p))) + (defvar msb--buffer-count 0) + (defun frame-or-buffer-changed-p () + (let ((count (length (buffer-list)))) + (when (/= count msb--buffer-count) + (setq msb--buffer-count count) + t)))) + +(unless (or (not (boundp 'menu-bar-update-hook)) + (memq 'menu-bar-update-buffers menu-bar-update-hook)) + (add-hook 'menu-bar-update-hook 'menu-bar-update-buffers)) + +(and (fboundp 'mouse-buffer-menu) + (substitute-key-definition 'mouse-buffer-menu 'msb (current-global-map))) + +(provide 'msb) +(eval-after-load 'msb (run-hooks 'msb-after-load-hooks)) +;;; msb.el ends here |