;;; efc.el -- Emacs Foundation Classes ;; $Revision: 1.1 $ $Date: 2007-11-26 15:16:44 $ ;; Author: Paul Kinnucan ;; Maintainer: Paul Kinnucan ;; Keywords: lisp, tools, classes ;; Copyright (C) 2001, 2002, 2003, 2004, 2005 Paul Kinnucan. ;; GNU Emacs 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, or (at your option) ;; any later version. ;; GNU Emacs 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 GNU Emacs; see the file COPYING. If not, write to the ;; Free Software Foundation, Inc., 59 Temple Place - Suite 330, ;; Boston, MA 02111-1307, US ;;; Commentary: ;; This package contains a set of eieio-based foundation classes ;; for Emacs. ;; Please send bug reports and enhancement suggestions ;; to Paul Kinnucan at ;; See end of this file for change history. ;;; Code: (require 'eieio) (require 'wid-edit) (defvar efc-query-options-function nil "If non-nil the function to use for interactively querying options. If nil then the default efc custom-based dialogs will be used.") ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; ;; Dialog Class ;; ;; ;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (defclass efc-dialog () ((title :initarg :title :type string :initform "Dialog" :documentation "Title of dialog") (buf :initarg :buf :type buffer :documentation "Dialog buffer") (initbuf :initarg :initbuf :type buffer :documentation "Buffer from which dialog was called.") ) "Super class of EFC dialogs." ) (defmethod initialize-instance ((this efc-dialog) &rest fields) "Constructor for dialog." ;; Call parent initializer. (call-next-method)) (defmethod efc-dialog-create ((this efc-dialog))) (defmethod efc-dialog-ok ((this efc-dialog)) "Invoked when the user clicks the dialog's okay button. The default method kills the dialog buffer." (kill-buffer (current-buffer))) (defmethod efc-dialog-cancel ((this efc-dialog)) "Invoked when the user clicks the dialog's Cancel button. The default method kills the dialog buffer." (delete-window) (set-buffer (oref this initbuf)) (pop-to-buffer (oref this initbuf)) (kill-buffer (oref this buf))) (defmethod efc-dialog-show ((this efc-dialog)) (oset this initbuf (current-buffer)) (oset this buf (get-buffer-create (oref this title))) (set-buffer (oref this buf)) (efc-dialog-create this) (widget-put (widget-create 'push-button :notify (lambda (button &rest ignore) (efc-dialog-ok (widget-get button :dialog))) "Ok") :dialog this) (widget-insert " ") (widget-put (widget-create 'push-button :notify (lambda (button &rest ignore) (efc-dialog-cancel (widget-get button :dialog))) "Cancel") :dialog this) (use-local-map widget-keymap) (widget-setup) ;; Position cursor over OK button. ;; (forward-line 0) (goto-char (point-min)) (pop-to-buffer (oref this buf))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; ;; Option Dialog ;; ;; ;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (defclass efc-option-dialog (efc-dialog) ((options :initarg :options :documentation "Options from from which to choose.") (radio-buttons :initarg :radio-buttons :documentation "Buttons for selecting options.") (text :initarg :text :type string :initform "Select option." :documentation "Text to be inserted at top of dialog.") (selection :initarg :selection :documentation "Option chosen by the user.")) "This dialog allows a user to choose one of a set of OPTIONS by clicking a radio button next to the option. The dialog sets SELECTION to the option chosen by the user when the user selects the OK button on the dialog. This dialog uses recursive edit to emulate a modal dialog.") (defmethod initialize-instance ((this efc-option-dialog) &rest fields) "Dialog constructor." (call-next-method)) (defmethod efc-dialog-create ((this efc-option-dialog)) (widget-insert (oref this text)) (widget-insert "\n\n") (oset this radio-buttons (widget-create (list 'radio-button-choice :value (car (oref this options)) :args (mapcar (lambda (x) (list 'item x)) (oref this options))))) (widget-insert "\n")) (defmethod efc-dialog-show ((this efc-option-dialog)) "Shows the options dialog buffer. After showing the dialog buffer, this method invokes recursive-edit to emulate the behavior of a modal dialog. This suspends the current command until the user has selected an option or canceled the dialog. See `efc-dialog-ok' and `efc-dialog-cancel' for more information." (save-window-excursion (call-next-method) (recursive-edit))) (defmethod efc-dialog-ok ((this efc-option-dialog)) "Invoked when the user selects the OK button on the options dialog. Sets the :selection field of THIS to the option chosen by the user, kills the dialog buffer, and exits recursive-edit mode." (oset this selection (widget-value (oref this radio-buttons))) (delete-window) (set-buffer (oref this initbuf)) (pop-to-buffer (oref this initbuf)) (kill-buffer (oref this buf)) (exit-recursive-edit)) (defmethod efc-dialog-cancel ((this efc-option-dialog)) "Invoked when the user clicks the dialog's Cancel button. Invokes the default cancel method, sets the :selection field of THIS to nil, and then exits recursive edit mode." (call-next-method) (oset this selection nil) (exit-recursive-edit)) (defun efc-query-options (options &optional prompt title) "Ask user to choose among a set of options." (if efc-query-options-function (funcall efc-query-options-function options prompt title) (let ((dialog (efc-option-dialog (or title "option dialog") :text (or prompt "Select option:") :options options))) (efc-dialog-show dialog) (oref dialog selection)))) ;; The following code is a patch that implements Richard Stallman's fix ;; for the following error that occurs only in Emacs 21.1.1. ;; ;; Debugger entered--Lisp error: (wrong-type-argument window-live-p #) ;; select-window(#) ;; exit-recursive-edit() ;; This replacement macro fixes the problem with exit-recursive-edit on Emacs 21. ;; You'll have to recompile wid-edit.el with it. ;; (defmacro save-selected-window (&rest body) ;; "Execute BODY, then select the window that was selected before BODY. ;; However, if that window has become dead, don't get an error, ;; just refrain from switching to it." ;; `(let ((save-selected-window-window (selected-window))) ;; (unwind-protect ;; (progn ,@body) ;; (if (window-live-p save-selected-window-window) ;; (select-window save-selected-window-window))))) (if (and (not (featurep 'xemacs)) (or (string-match "21\\.1" (emacs-version)) (string-match "21\\.2" (emacs-version)))) (progn ;; Need to load wid-edit first to ensure that ;; it does not get loaded after this patch and ;; hence override the patch. (require 'wid-edit) ;; Patched version of save-selected-window. (defmacro save-selected-window (&rest body) "Execute BODY, then select the window that was selected before BODY. However, if that window has become dead, don't get an error, just refrain from switching to it." `(let ((save-selected-window-window (selected-window))) (unwind-protect (progn ,@body) (if (window-live-p save-selected-window-window) (select-window save-selected-window-window))))) ;; Redefine widget-button-click to use the patched ;; version of save-selected-window (defun widget-button-click (event) "Invoke the button that the mouse is pointing at." (interactive "@e") (if (widget-event-point event) (let* ((pos (widget-event-point event)) (button (get-char-property pos 'button))) (if button ;; Mouse click on a widget button. Do the following ;; in a save-excursion so that the click on the button ;; doesn't change point. (save-selected-window (save-excursion (mouse-set-point event) (let* ((overlay (widget-get button :button-overlay)) (face (overlay-get overlay 'face)) (mouse-face (overlay-get overlay 'mouse-face))) (unwind-protect ;; Read events, including mouse-movement events ;; until we receive a release event. Highlight/ ;; unhighlight the button the mouse was initially ;; on when we move over it. (let ((track-mouse t)) (save-excursion (when face ; avoid changing around image (overlay-put overlay 'face widget-button-pressed-face) (overlay-put overlay 'mouse-face widget-button-pressed-face)) (unless (widget-apply button :mouse-down-action event) (while (not (widget-button-release-event-p event)) (setq event (read-event) pos (widget-event-point event)) (if (and pos (eq (get-char-property pos 'button) button)) (when face (overlay-put overlay 'face widget-button-pressed-face) (overlay-put overlay 'mouse-face widget-button-pressed-face)) (overlay-put overlay 'face face) (overlay-put overlay 'mouse-face mouse-face)))) ;; When mouse is released over the button, run ;; its action function. (when (and pos (eq (get-char-property pos 'button) button)) (widget-apply-action button event)))) (overlay-put overlay 'face face) (overlay-put overlay 'mouse-face mouse-face)))) (unless (pos-visible-in-window-p (widget-event-point event)) (mouse-set-point event) (beginning-of-line) (recenter))) (let ((up t) command) ;; Mouse click not on a widget button. Find the global ;; command to run, and check whether it is bound to an ;; up event. (mouse-set-point event) (if (memq (event-basic-type event) '(mouse-1 down-mouse-1)) (cond ((setq command ;down event (lookup-key widget-global-map [down-mouse-1])) (setq up nil)) ((setq command ;up event (lookup-key widget-global-map [mouse-1])))) (cond ((setq command ;down event (lookup-key widget-global-map [down-mouse-2])) (setq up nil)) ((setq command ;up event (lookup-key widget-global-map [mouse-2]))))) (when up ;; Don't execute up events twice. (while (not (widget-button-release-event-p event)) (setq event (read-event)))) (when command (call-interactively command))))) (message "You clicked somewhere weird."))) )) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; ;; Multiple Option Dialog ;; ;; ;; ;; Contributed by Philip Lord. ;; ;; ;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (defclass efc-multi-option-dialog (efc-option-dialog) ((build-message :initarg :text :type string :initform "Building Dialog" :documentation "Warning message while building dialog, as this can be slow")) "Provides a dialog with several sets of OPTIONS. The dialog sets SELECTION to the options selected by the user.") (defmethod initialize-instance ((this efc-multi-option-dialog) &rest fields) "Dialog constructor." (call-next-method)) (defmethod efc-dialog-create ((this efc-multi-option-dialog)) (message "%s..." (oref this build-message)) (widget-insert (oref this text)) (widget-insert "\n\n") ;; use radio buttons slot as list of radio buttons rather than. (oset this radio-buttons (mapcar (lambda(list) (prog1 (widget-create (list 'radio-button-choice :value (efc-multi-option-dialog-default this list) :args (mapcar (lambda (x) (list 'item x)) list))) (widget-insert "\n"))) (efc-multi-option-dialog-sort this (oref this options)))) (widget-insert "\n") (message "%s...done" (oref this text))) (defmethod efc-dialog-ok((this efc-multi-option-dialog)) ;; set the selection up as a list rather a simple result (oset this selection (mapcar (lambda(widget) (widget-value widget)) (oref this radio-buttons))) (delete-window) (set-buffer (oref this initbuf)) (pop-to-buffer (oref this initbuf)) (kill-buffer (oref this buf)) (exit-recursive-edit)) (defmethod efc-multi-option-dialog-default ((this efc-multi-option-dialog) list) "Pick the default from a collection of options." (if (= 1 (length list)) (car list))) (defmethod efc-multi-option-dialog-sort ((this efc-multi-option-dialog) list) "Sort the options." ;; sort the ones with the most options first... (sort list (lambda(a b) (> (length a) (length b))))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; ;; Compiler Class ;; ;; ;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (defclass efc-compiler () ((name :initarg :name :type string :documentation "Compiler name.") (buffer :initarg :buffer :type buffer :documentation "Compilation buffer") (window :initarg :window :type window :documentation "Window that displays the compilation buffer.") (exec-path :initarg :exec-path :type string :documentation "Path of compiler executable.") (comp-finish-fcn :initarg :comp-finish-fcn :type function :documentation "Function to invoke at end of compilation.")) "Class of compiler-like applications.") (defmethod create-buffer ((this efc-compiler)) "Create a buffer to display the output of a compiler process." (save-excursion (let ((buf (get-buffer-create (format "*%s*" (oref this name)))) (error-regexp-alist compilation-error-regexp-alist) (enter-regexp-alist (if (boundp 'compilation-enter-directory-regexp-alist) compilation-enter-directory-regexp-alist)) (leave-regexp-alist (if (boundp 'compilation-leave-directory-regexp-alist) compilation-leave-directory-regexp-alist)) (file-regexp-alist (if (boundp 'compilation-file-regexp-alist) compilation-file-regexp-alist)) (nomessage-regexp-alist (if (not jde-xemacsp) compilation-nomessage-regexp-alist)) (parser compilation-parse-errors-function) (error-message "No further errors") (thisdir default-directory)) (oset this :buffer buf) (set-buffer buf) ;; Make sure a compiler process is not ;; already running. (let ((compiler-proc (get-buffer-process (current-buffer)))) (if compiler-proc (if (or (not (eq (process-status compiler-proc) 'run)) (yes-or-no-p (format "A %s process is running; kill it?" (oref this name)))) (condition-case () (progn (interrupt-process compiler-proc) (sit-for 1) (delete-process compiler-proc)) (error nil)) (error "Cannot have two processes in `%s' at once" (buffer-name))))) ;; In case the compiler buffer is current, make sure we get the global ;; values of compilation-error-regexp-alist, etc. (kill-all-local-variables) ;; Clear out the compilation buffer and make it writable. (setq buffer-read-only nil) (buffer-disable-undo (current-buffer)) (erase-buffer) (buffer-enable-undo (current-buffer)) (compilation-mode (oref this name)) (set (make-local-variable 'compilation-parse-errors-function) parser) (set (make-local-variable 'compilation-error-message) error-message) (set (make-local-variable 'compilation-error-regexp-alist) error-regexp-alist) (if (not jde-xemacsp) (progn (set (make-local-variable 'compilation-enter-directory-regexp-alist) enter-regexp-alist) (set (make-local-variable 'compilation-leave-directory-regexp-alist) leave-regexp-alist) (set (make-local-variable 'compilation-file-regexp-alist) file-regexp-alist) (set (make-local-variable 'compilation-nomessage-regexp-alist) nomessage-regexp-alist))) (if (slot-boundp this 'comp-finish-fcn) (set (make-local-variable 'compilation-finish-function) (oref this comp-finish-fcn))) (setq default-directory thisdir compilation-directory-stack (list default-directory))))) (defmethod get-args ((this efc-compiler)) "Get a list of command-line arguments to pass to the compiler process.") (defmethod exec ((this efc-compiler)) "Start the compiler process." (create-buffer this) ;; Pop to checker buffer. (let ((outwin (display-buffer (oref this :buffer)))) (compilation-set-window-height outwin) (oset this :window outwin)) (if (not (featurep 'xemacs)) (if compilation-process-setup-function (funcall compilation-process-setup-function))) (let* ((outbuf (oref this :buffer)) (executable-path (oref this exec-path)) (args (get-args this))) (save-excursion (set-buffer outbuf) (insert (format "cd %s\n" default-directory)) (insert (concat executable-path " " (mapconcat 'identity args " ") "\n\n")) (let* ((process-environment (cons "EMACS=t" process-environment)) (w32-quote-process-args ?\") (win32-quote-process-args ?\") ;; XEmacs (proc (apply 'start-process (downcase mode-name) outbuf executable-path args))) (set-process-sentinel proc 'compilation-sentinel) (set-process-filter proc 'compilation-filter) (set-marker (process-mark proc) (point) outbuf) (setq compilation-in-progress (cons proc compilation-in-progress))) (set-buffer-modified-p nil) (setq compilation-last-buffer (oref this :buffer))))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; ;; Collection Class ;; ;; ;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (defclass efc-collection () ((elem-type :initarg :elem-type :type (or null symbol) :initform nil :documentation "Type of element that this collection contains.")) "A collection of objects. The collection can be either homogeneous, i.e., composed of elements of one type, or heterogeneous. The ELEM-TYPE property of a heterogeneous collection is nil.") (defmethod efc-coll-type-compatible-p ((this efc-collection) item) "Returns t if ITEM is type-compatible with this collection. An item is type-compatible with a collection if the collection is heterogeneous or the item's type is the same as the collection's element type." (let ((element-type (oref this elem-type))) (or (eq element-type nil) (typep item element-type)))) (defmethod efc-coll-iterator ((this efc-collection)) "Returns an iterator for this collection." (error "Abstract method.")) (defmethod efc-coll-visit ((this efc-collection) visitor) "Maps VISITOR to each element of the collection. VISITOR is an object of efc-visitor class." (let ((iter (efc-coll-iterator this))) (while (efc-iter-has-next iter) (efc-visitor-visit visitor (efc-iter-next iter))))) (defmethod efc-coll-memberp ((this efc-collection) member) "Returns nonil if this contains item." (error "Abstract method.")) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; ;; Iterator Class ;; ;; ;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (defclass efc-iterator () () "Iterates over a collection.") (defmethod efc-iter-has-next ((this efc-iterator)) "Returns nonnil if the iterator has not returned all of the collection's elements." (error "Abstract method.")) (defmethod efc-iter-next ((this efc-iterator)) "Return the next element of the collection." (error "Abstract method.")) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; ;; Visitor Class ;; ;; ;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (defclass efc-visitor () () "Visits each member of a collection.") (defmethod efc-visitor-visit ((this efc-visitor) member) "Visits MEMBER, a member of a collection." (error "Abstract method.")) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; ;; List Class ;; ;; ;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (defclass efc-list (efc-collection) ((items :initarg :items :type list :initform nil :documentation "List of items.")) "List of items.") (defmethod initialize-instance ((this efc-list) &rest fields) "Iterator constructor." (call-next-method)) (defmethod efc-coll-add ((this efc-list) item) "Adds an item to the list." (if (efc-coll-type-compatible-p this item) (oset this items (append (oref this items) (list item))) (error "Tried to add an item of type %s to a list of items of type %s" (typep item) (oref this elem-type)))) (defmethod efc-coll-iterator ((this efc-list)) "Return an iterator for this list." (efc-list-iterator "list iterator" :list-obj this)) (defmethod efc-coll-memberp ((this efc-list) item) "Returns nonil if this list contains item." (member item (oref this items))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; ;; List Iterator Class ;; ;; ;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (defclass efc-list-iterator (efc-iterator) ((list-obj :initarg :list-obj :type efc-list :documentation "List that this iterator iterates.") (list :type list :documentation "Lisp list.")) "Iterates over a list.") (defmethod initialize-instance ((this efc-list-iterator) &rest fields) "Iterator constructor." (call-next-method) (assert (oref this list-obj)) (assert (typep (oref this list-obj) efc-list)) (oset this list (oref (oref this list-obj) items))) (defmethod efc-iter-has-next ((this efc-list-iterator)) "Returns true if this iterator has another list item to return." (oref this list)) (defmethod efc-iter-next ((this efc-list-iterator)) "Get next item in the list." (let* ((list (oref this list)) (next (car list))) (oset this list (cdr list)) next)) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; ;; List Set Class ;; ;; ;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (defclass efc-list-set (efc-list) () "List that contains no duplicates.") (defmethod efc-coll-add ((this efc-list-set) item) "Adds an item to a set only if the set does not already contain the item." (if (efc-coll-memberp this item) (error "This set already contains %s" item) (call-next-method))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; ;; Association Class ;; ;; ;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (defclass efc-assoc (efc-list) () "Association") (defmethod efc-coll-put ((this efc-assoc) key value) "Put an item into the association list." (oset this items (append (oref this items) (list (cons key value))))) (defmethod efc-coll-get ((this efc-assoc) key) "Get an item from the association list." (cdr (assq key (oref this items)))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; ;; Association Set Class ;; ;; ;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (defclass efc-assoc-set (efc-assoc) () "Association that contains no duplicate keys.") (defmethod efc-coll-put ((this efc-assoc-set) key value) "Adds an item to a set only if the set does not already contain the item." (if (efc-coll-get this key) (error "This set already contains %s" key) (call-next-method))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; ;; Hash Table Class ;; ;; ;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (defclass efc-hash-table (efc-collection) ((table :type hash-table :documentation "Lisp table object.")) "Hash table.") (defmethod initialize-instance ((this efc-hash-table) &rest fields) "Hash table constructor." (call-next-method) (oset table (make-hash-table))) (defmethod efc-coll-put ((this efc-hash-table) key value) "Put an item into the table." (if (efc-coll-type-compatible-p this value) (puthash key value (oref this table)) (error "Tried to add an item of type %s to a hash table of items of type %s" (typep value) (oref this elem-type)))) (defmethod efc-coll-get ((this efc-hash-table) key) "Get an item from the table." (gethash key (oref this table))) (defmethod efc-coll-visit ((this efc-hash-table) visitor) "Visit each item in the hash table. VISITOR is an instance of efc-visitor class." (maphash (lambda (key value) (efc-visitor-visit visitor value)) (oref this table))) (defmethod efc-coll-iterator ((this efc-hash-table)) "Return an iterator for this hash table." (efc-list-iterator "hash table iterator" :list-obj (let (values) (maphash (lambda (key value) (setq values (append values (list value)))) (oref this table)) values))) (provide 'efc) ;; Change History ;; $Log: efc.el.upstream,v $ ;; Revision 1.1 2007-11-26 15:16:44 michaels ;; Update jde to author version 2.3.5.1. ;; ;; Revision 1.18 2005/03/19 03:50:31 paulk ;; Define an association set. ;; ;; Revision 1.17 2005/03/18 04:53:14 paulk ;; Define a set of collection classes. ;; ;; Revision 1.16 2004/12/10 03:38:04 paulk ;; Fix efc-compiler to make and set a buffer-local version of compilation-finish-function. ;; Thanks To David Evers. ;; ;; Revision 1.15 2004/07/01 14:04:39 jslopez ;; Compatibility fix for emacs in CVS. Replaces jde-xemacsp check for boundp for ;; the following variables: compilation-nomessage-regexp-alist, ;; compilation-file-regexp-alist, compilation-leave-directory-regexp-alist, ;; compilation-enter-directory-regexp-alist. Uses the compilation-mode without a ;; parameter. The emacs in CVS does not contain the variables, or the parameter ;; for compilation mode. ;; ;; Revision 1.14 2004/03/16 07:42:09 paulk ;; Define new efc-multi-option-dialog. Thanks to Philip Lord. ;; ;; Revision 1.13 2003/11/29 05:50:18 paulk ;; The efc-dialog-show method ofr efc-option-dialog now uses save-window-excursion ;; to restore the user's original window conversion after showing the selection ;; buffer. ;; ;; Revision 1.12 2003/08/25 04:57:30 paulk ;; Adds efc-compiler class. This class implements an Emacs interface to an external process ;; that generates compiler-like output. ;; ;; Revision 1.11 2003/06/07 04:04:10 paulk ;; Fix regexp for matching Emacs versions. Thanks to David Ponce. ;; ;; Revision 1.10 2003/03/28 05:33:29 andyp ;; XEmacs optimizations for JDEbug and efc. ;; ;; Revision 1.9 2002/03/29 12:40:27 paulk ;; Adds efc-query-option function. ;; ;; Revision 1.8 2002/03/19 12:24:47 paulk ;; Updated live-window error patch to work for Emacs 21.2. ;; ;; Revision 1.7 2002/02/21 05:35:39 paulk ;; efc-dialog class now creates the dialog buffer in ;; the efc-dialog-show method instead of in the ;; intialize-instance method. This permits reuse ;; of the dialog buffer object and hence persistance ;; of user settings in the dialog. ;; ;; Revision 1.6 2002/01/25 10:41:55 paulk ;; Fixes Lisp error: (wrong-type-argument window-live-p #) that ;; occurs in Emacs 21.1.1 when the user clicks an efc dialog box button. ;; ;; Revision 1.5 2002/01/06 06:54:06 paulk ;; Finally found a fix for the efc dialog class that works around ;; the delete-other-windows bug in Emacs 21. ;; ;; Revision 1.4 2001/12/04 14:45:55 jslopez ;; Change jde-xemacs for (featurep 'xemacs). ;; ;; Revision 1.3 2001/12/04 12:32:34 jslopez ;; Fixes typo (efc-xemacsp to jde-xemacsp). ;; ;; Revision 1.2 2001/12/04 06:06:34 paulk ;; Remove carriage returns. ;; ;; Revision 1.1 2001/12/04 05:23:20 paulk ;; Initial revision. ;; ;; ;; End of efc.el