;; jde-complete.el -- Smart completion for the JDE ;; ;; $Revision: 1.84 $ ;; ;; Author: Rodrigo Reyes ;; Maintainers: Rodrigo Reyes ;; Paul Kinnucan ;; David Ponce ;; Howard Spector ;; Stephane Nicolas , ;; Javier Lopez ;; Keywords: java, intellisense, completion ;; Copyright (C) 1999, 2000, 2001, 2002, 2003, 2004 Rodrigo Reyes, Paul Kinnucan, ;; Stephane Nicolas, David Ponce, Javier Lopez ;; This package follows the GNU General Public Licence (GPL), see the ;; COPYING file that comes along with GNU Emacs. This is free software, ;; you can redistribute it and/or modify it under the GNU GPL terms. ;; ;; Java is a registered trademark of Sun Microsystem, Inc. ;; ;;; Commentary: ;; This is one of a set of packages that make up the ;; Java Development Environment for Emacs (JDEE). See the ;; JDEE User's Guide for more information. ;; ;; This package adds smart completion to the JDE. How it works is ;; simple : put the cursor at the end of a statement "under ;; construction", eg. "myVariable.rem and call the ;; jde-complete-at-point emacs-lisp function (this is by default ;; C-.). A completion is then inserted. If multiple completions are ;; possible, calling the completion function again will cycle through ;; all the possibilities (as dabbrev-mode does). ;; To retrieve all the possible completions, it uses the java code in ;; jde.util.Completion.getClassInfo(), called by beanshell. That ;; need the class to be compiled (but that's not worst than an etag ;; call). ;; Known bugs/problems : ;; ;; TODO ;; Move all the strings manipulation to the java side so that ;; class info can be loaded in the background. ;; ;; The latest version of the JDEE is available at ;; . ;; Please send any comments, bugs, or upgrade requests to ;; Paul Kinnucan at pkinnucan@attbi.com ;;jde-eldoc for completion signatures (require 'eldoc) (require 'semantic-idle) (defgroup jde-complete nil "JDE Completion" :group 'jde :prefix "jde-complete-") (defconst jde-complete-public 0 "Represent a public java modifiers") (defconst jde-complete-protected 1 "Represent a protected java modifiers") (defconst jde-complete-package 2 "Represent a package java modifiers") (defconst jde-complete-private 3 "Represent a private java modifiers") (defconst jde-complete-fields 0 "Represents the positions where the fields are placed.") (defconst jde-complete-constructors 1 "Represents the positions where the constructors are placed.") (defconst jde-complete-methods 2 "Represents the positions where the methods are placed.") (defconst jde-complete-classes 3 "Represents the positions where the classes are placed.") (defvar jde-complete-temp-process "*Temp*" "Used as the name of a temporary process") (defvar jde-complete-last-compiled-class nil "Contains the name of the class that was compiled last") (defvar jde-complete-current-signature nil "Contains the signature of the last method inserted by either `jde-complete-in-line' or `jde-complete-menu'") (defconst jde-complete-signature-buffer "*Signature*" "Buffer to display signatures") (defvar jde-complete-display-signature nil "If non nil it displays the `jde-complete-current-signaure' in the minibuffer") (defcustom jde-complete-signature-display (list "Eldoc") "Display modes used to show the method signature after a completion. The display modes choices are using eldoc-mode(the signature shows in the minibuffer), a buffer(a one line buffer shows the signature and then dissapears), or none." :group 'jde-complete :type '(list (radio-button-choice (const "Eldoc") (const "Buffer") (const "None")))) (defcustom jde-complete-signature-display-time 5 "Amount of time in seconds to display the method signature in the minibuffer after a completion." :group 'jde-complete :type 'integer) (defcustom jde-complete-add-space-after-method nil "*If non nil it will add a space between the method name and the first parenthesis. i.e foo () instead of foo() when using the completion methods `jde-complete-in-line' and `jde-complete-menu' After customizing this variable, be sure to use `jde-complete-flush-classinfo-cache', any class information that was previously cache is not going to be affected by this setting." :group 'jde-complete :type 'boolean) (defcustom jde-complete-unique-method-names nil "If non nil it will display methods with the same name but with different signature only once, the signature for ths methods will be the signature of the first method in the list of completions The list of completion is sorted in alphabetical order. This variable modifies the completion for `jde-complete-menu' and `jde-complete-in-line'. After customizing this variable, be sure to use `jde-complete-flush-classinfo-cache', any class information that was previously cache is not going to be affected by this setting" :group 'jde-complete :type 'boolean) (defcustom jde-complete-insert-method-signature t "If non nil it will insert the method signature when using `jde-complete-in-line' and `jde-complete-menu' i.e. v.get(int, java.lang.String). If `jde-complete-unique-method-names' is non nil methods with the same name will get the signature of the first one in the completion list. After customizing this variable, be sure to use `jde-complete-flush-classinfo-cache', any class information that was previously cache is not going to be affected by this setting" :group 'jde-complete :type 'boolean) (defcustom jde-complete-display-qualified-types t "If non nil use fully qualified types when displaying methods for selection, i.e. v.get(int, java.lang.String). If nil, use unqualified types, i.e. v.get(int, String). After customizing this variable, be sure to use `jde-complete-flush-classinfo-cache', any class information that was previously cache is not going to be affected by this setting." :group 'jde-complete :type 'boolean) (defcustom jde-complete-display-result-type t "If non nil include result type when displaying methods for selection. After customizing this variable, be sure to use `jde-complete-flush-classinfo-cache', any class information that was previously cache is not going to be affected by this setting." :group 'jde-complete :type 'boolean) (defcustom jde-complete-display-throws t "If non nil include thrown exceptions when displaying methods for selection. After customizing this variable, be sure to use `jde-complete-flush-classinfo-cache', any class information that was previously cache is not going to be affected by this setting." :group 'jde-complete :type 'boolean) ;; (makunbound 'jde-complete-function) (defcustom jde-complete-function 'jde-complete-menu "*Function that will be invoked by the `jde-complete-select' command. The `jde-complete-menu' function displays completions for the symbol at point in a popup menu. The `jde-complete-minibuf' function displays completions in the minibuffer. You may also specify a custom function to use. The custom function must be an interactive function that can be called by `call-interactively'." :group 'jde-project :type '(choice (function-item jde-complete-menu) (function-item jde-complete-minibuf) (function-item jde-complete-in-line) (function :format "%t %v" :tag "Custom:"))) (defvar jde-complete-current-list nil "The list of all the completion. Each element of the list is a list which car is the possible completion, and the cdr is an additional information about this completion.") (defvar jde-complete-current-list-index nil "An index to an element in jde-complete-current-list. This is used to cycle the list.") (defun jde-complete-valid-java-declaration-at (point varname) "Verify that a POINT starts a valid java declaration for the VARNAME variable." (save-excursion (goto-char point) (if (looking-at (concat "\\([A-Za-z0-9_.\177-\377]+\\)[ \t\n\r]+" (jde-complete-double-backquotes varname) "[ \t\n\r]*[;=]")) (match-string 1) nil))) (defun jde-complete-double-backquotes (varname) "Build a new string identical to VARNAME, except that every backquote `\' is doubled, so that it can be used in a regex expression" (let (result (idx 0) (len (length varname)) curcar) (while (< idx len) (setq curcar (elt varname idx)) (setq result (concat result (if (eq curcar ?\\) "\\\\" (make-string 1 curcar)))) (setq idx (1+ idx))) result)) (defun jde-complete-declared-type-of (name) "Find in the current buffer the java type of the variable NAME. The function returns a string containing the name of the class, or nil otherwise. This function does not give the fully-qualified java class name, it just returns the type as it is declared." (save-excursion (let (found res pos orgpt resname) (while (and (not found) (search-backward name nil t)) (setq pos (point)) (backward-word 1) (setq resname (jde-complete-valid-java-declaration-at (point) name)) (goto-char pos) (forward-char -1) (if resname (progn (setq res resname) (setq found t)))) res))) (defun jde-complete-filter-fqn (importlist) "Filter all the fully-qualified classnames in the import list. It uses the knowledge that those classnames are at the beginning of the list, so that it can stops at the first package import (with a star `*' at the end of the declaration)." (if importlist (if (string= "*" (car (cdr (car importlist)))) importlist (jde-complete-filter-fqn (cdr importlist))))) (defun jde-complete-guess-type-of (name) "Guess the fully qualified name of the class NAME, using the import list. It returns a string if the fqn was found, or a list of possible packages otherwise." (let ((importlist (jde-parse-import-list)) shortname fullname tmp result) (while (and importlist (null result)) (setq tmp (car importlist)) (setq shortname (car (cdr tmp))) (setq fullname (concat (car tmp) name)) (cond ((string= "*" shortname) (setq result importlist)) ((string= name shortname) (setq result fullname)) (t (setq importlist (cdr importlist))))) result)) (defvar jde-complete-classinfo-cache nil) (defcustom jde-complete-classinfo-cache-size 50 "The max size of completion's cache.") (defun jde-complete-flush-classinfo-cache () "Flushes all entries in the completion cache" (interactive) (setq jde-complete-classinfo-cache nil)) (defun jde-complete-flush-classes-in-cache (class-list) "Flushes all the classes in CLASS-LIST as entries of cache." (let ((temp (nth 0 jde-complete-classinfo-cache)) (index -1) (found nil) (class (car class-list))) (while class (while (and temp (not found)) (setq index (1+ index)) (setq temp (nth index jde-complete-classinfo-cache)) (if (string= (car temp) class) (setq found t))) (if found (setq jde-complete-classinfo-cache (nthcdr (1+ index) jde-complete-classinfo-cache))) (setq class-list (cdr class-list)) (setq class (car class-list)) (setq found nil)))) (defun jde-complete-add-to-classinfo-cache (name classinfo) (let (new-entry new-list) (if (nth jde-complete-classinfo-cache-size jde-complete-classinfo-cache) (progn (setq new-entry (list name classinfo)) (setq new-list (list new-entry nil)) (setcdr new-list (cdr jde-complete-classinfo-cache)) (setq jde-complete-classinfo-cache new-list) (message "cache is full")) ;;else (setq jde-complete-classinfo-cache (append jde-complete-classinfo-cache (list (list name classinfo))))))) (defun jde-complete-get-from-cache (name) (let ((temp (nth 0 jde-complete-classinfo-cache)) (index -1) (found nil)) (while (and temp (not found)) (setq index (1+ index)) (setq temp (nth index jde-complete-classinfo-cache)) (if (string= (car temp) name) (setq found t))) (if found (nth 1 temp) nil))) (defun jde-complete-get-classinfo (name &optional access-level) "Return the class info list for the class NAME and the ACCESS-LEVEL. Allowed values for access level are 0 for protected 1 for private. This function first checks to see if the class info is cached. If so, it returns the cached class info. Otherwise, it creates the class info list. Each element of the list returned by this function is itself a list whose car is a possible completion and whose cdr gives additional informations on the completion." ;;replacing double quotes by empty strings double quotes seems to break the ;;beanshell (while (string-match "\"" name) (setq name (replace-match "" nil nil name))) ;;replacing back slaches by empty strings backlashes causes beanshell problem (while (string-match "\\\\" name) (setq name (replace-match "" nil nil name))) ;;checking that access-level is not nil (if (not access-level) (setq access-level jde-complete-public)) (let ((class-info (jde-complete-get-from-cache name)) public-methods protected-methods private-methods package-methods) (when (not class-info) ;;Getting public class info (setq public-methods (jde-complete-invoke-get-class-info name jde-complete-public)) ;;Getting protected class info (setq protected-methods (jde-complete-invoke-get-class-info name jde-complete-protected)) ;;Getting package class info (setq package-methods (jde-complete-invoke-get-class-info name jde-complete-package)) ;;Getting private class info (setq private-methods (jde-complete-invoke-get-class-info name jde-complete-private)) (setq class-info (append public-methods protected-methods package-methods private-methods)) (if class-info (jde-complete-add-to-classinfo-cache name class-info))) ;;Getting the class info depending on the access level (setq class-info (jde-complete-get-accessible-info class-info access-level name)) (setq class-info (jde-complete-build-completion-list class-info)) ;;Removing duplicates (setq class-info (jde-complete-remove-duplicates class-info)) ;;Sorting the list (setq class-info (sort class-info 'jde-complete-sort-comparison)) class-info)) (defun jde-complete-remove-duplicates (class-list) "Removes duplicates from class-list" (let (answer temp) (while class-list (setq temp (car class-list)) (if (not (jde-complete-memberp (car temp) answer)) (setq answer (append answer (list temp)))) (setq class-list (cdr class-list))) answer)) (defun jde-complete-memberp (elt lst) "Returns t if elt is a memver of lst" (let (answer tmp) (while lst (setq tmp (caar lst)) (if (string= tmp elt) (progn (setq answer t) (setq lst nil)) (setq lst (cdr lst)))) answer)) (defun jde-complete-get-accessible-info (class-info access name) "Takes a list of class info in this format \(list \(list public info\) \(list protected info\) \(list package info\) \(list private info\)\). Each info list is in the format \(list \(list fields\) \(list constructors\) \(list methods\) \(list inner classes\)\). This method will return a list concatenating the fields, methods, and inner classes for the access level." (let* ((public (nth jde-complete-public class-info)) (protected (nth jde-complete-protected class-info)) (package (nth jde-complete-package class-info)) (private (nth jde-complete-private class-info)) (package-name (jde-parse-get-package-name)) (this (concat (if package-name (concat package-name "." )) (jde-parse-get-class-at-point))) answer fields constructors methods classes packagep) (if (null package-name) (setq package-name "")) (if package-name (setq packagep (string-match package-name name))) (setq fields (append (nth jde-complete-fields public) (if (>= access jde-complete-protected) (nth jde-complete-fields protected)) (if packagep (nth jde-complete-fields package)) (if (or (>= access jde-complete-private) (string= name this)) (nth jde-complete-fields private)))) (setq constructors (append (nth jde-complete-constructors public) (if (>= access jde-complete-protected) (nth jde-complete-constructors protected)) (if packagep (nth jde-complete-constructors package)) (if (or (>= access jde-complete-private) (string= name this)) (nth jde-complete-constructors private)))) (setq methods (append (nth jde-complete-methods public) (if (>= access jde-complete-protected) (nth jde-complete-methods protected)) (if packagep (nth jde-complete-methods package)) (if (or (>= access jde-complete-private) (string= name this)) (nth jde-complete-methods private)))) (setq classes (append (nth jde-complete-classes public) (if (>= access jde-complete-protected) (nth jde-complete-classes protected)) (if packagep (nth jde-complete-classes package)) (if (or (>= access jde-complete-private) (string= name this)) (nth jde-complete-classes private)))) (setq answer (list fields constructors methods classes)) answer)) (defun jde-complete-invoke-get-class-info (name access) "Invoke the method jde.util.Completion.getClassInfo(String, int)" (jde-jeval-r (format "jde.util.Completion.getClassInfo(\"%s\",%d);" name access))) (defun jde-complete-get-classinfo-javacode (name import access-level) "Return the java code that calls the jde.util.Completion.getClassInfo function with the short java class name NAME and the package list IMPORT where to look at." (save-excursion (concat "{ " "String[] lst = new String[" (number-to-string (length import)) "];\n" (let ((count -1)) (mapconcat (function (lambda (x) (setq count (+ 1 count)) (concat "lst[" (int-to-string count) "]=\"" (car (nth count import)) "\";\n"))) import " ")) "jde.util.Completion.getClassInfo(\"" name "\",lst," (number-to-string access-level) ");\n" "}"))) (defun jde-complete-sort-comparison (first second) (string< (car first) (car second))) (defun jde-complete-get-variables (variables) "Transform a list of the type (\"var\" \"java.lang.String\") into (\"var\" \"java.lang.String\ var\")" (let (result current prev) (if (null jde-complete-unique-method-names) (while variables (setq current (car (car variables))) (setq result (append (list (cons (concat current (if jde-complete-display-result-type (concat " : " (jde-complete-maybe-unqualify (nth 1 (car variables)))))) current)) result)) (setq variables (cdr variables))) (while variables (if (not (string= prev current)) (progn (setq prev current) (setq result (append (list (cons (concat current (if jde-complete-display-result-type (concat " : " (jde-complete-maybe-unqualify (nth 1 (car variables)))))) current)) result)))) (setq variables (cdr variables)))) result)) (defun jde-complete-build-completion-list (classinfo) "Build a completion list from the CLASSINFO list, as returned by the jde.util.Completion.getClassInfo function." (let (result tmp) ;; get the variable fields (setq tmp (nth jde-complete-fields classinfo)) (setq result (jde-complete-get-variables tmp)) ;;get the constructors (setq tmp (jde-complete-get-methods (nth jde-complete-constructors classinfo) t)) (if tmp (setq result (append tmp result))) ;; get the methods (setq tmp (jde-complete-get-methods (nth jde-complete-methods classinfo))) (if tmp (setq result (append tmp result))) ;; get inner classes (setq tmp (jde-complete-get-inner-classes (nth jde-complete-classes classinfo))) (if tmp (setq result (append tmp result))) result)) (defun jde-complete-get-methods (classinfo &optional constructor) (let ((end-paren (if (null jde-complete-add-space-after-method) "(" " (")) (end-parens (if (null jde-complete-insert-method-signature) (if (null jde-complete-add-space-after-method) "()" " ()") "")) prev tmp current) (while classinfo (let* ((type (car (cdr (car classinfo))));;method type i.e. boolean (exceptions (jde-get-exceptions (car (last (car classinfo))))) (method (jde-complete-build-information-for-completion (car classinfo) end-paren)) (display (jde-complete-build-display-for-completion (car classinfo) end-paren constructor))) (setq current (jde-parse-get-unqualified-name (car (car classinfo)))) (if (not (and jde-complete-unique-method-names (string= prev current))) (progn (setq prev current) (setq tmp (append (list (cons display (concat (if (null jde-complete-insert-method-signature) current method) end-parens))) tmp)))) (setq classinfo (cdr classinfo)))) tmp)) (defun jde-complete-get-inner-classes(class-info) "Takes as an argument a list of inner classes an return a string of them or nil" (let (tmp fullname pos name) (if class-info (while class-info (let* ((fullname (caar class-info)) (pos (string-match "\\$" fullname)) (name (substring fullname (+ 1 pos)))) (setq tmp (append (list (cons (concat name " : " fullname) name)) tmp))) (setq class-info (cdr class-info)))) tmp)) (defun jde-get-exceptions (exceptions) "Takes as an argument a list of EXCEPTIONS and return a string of them or nil" (if (and (listp exceptions) (car exceptions)) (let ((exs "")) (while exceptions (setq exs (concat exs (car exceptions))) (setq exceptions (cdr exceptions)) (if exceptions (setq exs (concat exs ", ")))) exs) nil)) (defun jde-complete-maybe-unqualify (type) (if jde-complete-display-qualified-types type (jde-parse-get-unqualified-name type))) (defun jde-complete-build-display-for-completion (lst end-parens &optional constructor) "Builds the string that describes a method in a menu for selecting a completion." (let ((result (concat (jde-parse-get-unqualified-name (car lst)) end-parens)) (rettype (car (cdr lst))) (exceptions (if (and (listp (last lst)) (car (last lst))) (car (last lst))))) (if constructor (setq lst (cdr lst)) (setq lst (cdr (cdr lst)))) (while (and lst (not (listp (car lst)))) (setq result (concat result (jde-complete-maybe-unqualify (car lst)))) (setq lst (cdr lst)) (if (and lst (not (listp (car lst)))) (setq result (concat result ", ")))) (setq result (concat result ")")) (concat result (if (or (and (not constructor) rettype jde-complete-display-result-type) (and exceptions jde-complete-display-throws)) " : ") (if (and (not constructor) rettype jde-complete-display-result-type) (jde-complete-maybe-unqualify rettype)) (if (and exceptions jde-complete-display-throws) (concat " throws " (jde-get-exceptions (mapcar 'jde-complete-maybe-unqualify exceptions))))))) (defun jde-complete-build-information-for-completion (lst end-parens &optional constructor) "Builds the text that is inserted in the code for a particular completion." (let ((result (concat (jde-parse-get-unqualified-name (car lst)) end-parens))) (if constructor (setq lst (cdr lst)) (setq lst (cdr (cdr lst)))) (while (and lst (not (listp (car lst)))) (setq result (concat result (car lst))) (setq lst (cdr lst)) (if (and lst (not (listp (car lst)))) (setq result (concat result ", ")))) (setq result (concat result ")")) result)) (defun jde-complete-complete-cycle () "Replace the previous completion by the next one in the list." (let (elem tmp) (setq jde-complete-current-list-index (1+ jde-complete-current-list-index)) (if (>= jde-complete-current-list-index (length jde-complete-current-list)) (setq jde-complete-current-list-index 0)) (setq elem (nth jde-complete-current-list-index jde-complete-current-list)) (setq tmp (cdr elem)) (if tmp (progn (delete-region jde-parse-current-beginning jde-parse-current-end) (insert tmp) (setq jde-complete-current-signature (car elem)) (jde-complete-place-cursor) (set-marker jde-parse-current-end (+ (marker-position jde-parse-current-beginning) (length tmp))) (jde-complete-display-current-signature));;displaying the signature (message (format "No completion at this point!(cycle)"))) ;; (goto-char (marker-position jde-complete-current-end)) )) (defun jde-complete-insert-completion (item) (if item (progn (delete-region jde-parse-current-beginning jde-parse-current-end) (insert item) (jde-complete-place-cursor) (jde-complete-display-current-signature) (set-marker jde-parse-current-end (+ (marker-position jde-parse-current-beginning) (length item)))))) (defun jde-complete-find-all-completions (pair lst &optional exact-match) (let* (tmp chop-pos (args (nth 2 pair)) (pat (nth 1 pair)) (result nil) (first-char (substring pat 0 1))) (if (null args) (setq exact-match nil) (setq pat (concat pat args))) (if (string= pat "$") (setq pat "\\$")) (while lst (setq tmp (car (car lst))) (setq chop-pos (string-match " : " tmp)) (setq tmp (substring tmp 0 chop-pos)) (if (if exact-match (string= pat tmp) (equal 0 (string-match pat tmp))) (setq result (append result (list (car lst))))) (setq lst (cdr lst))) result)) (defun jde-complete-find-completion-for-pair (pair &optional exact-completion access-level) (let ((type (jde-parse-eval-type-of (car pair)))) (if type (cond ((member type jde-parse-primitive-types) (error "Cannot complete primitive type: %s" type)) ((string= type "void") (error "Cannot complete return type of %s is void." (car pair))) (access-level (let ((classinfo (jde-complete-get-classinfo type access-level))) (if classinfo (if (and (string= (nth 1 pair) "") (not exact-completion)) (setq jde-complete-current-list classinfo) (setq jde-complete-current-list (jde-complete-find-all-completions pair classinfo exact-completion)))))) (t (let ((classinfo (jde-complete-get-classinfo type))) (if classinfo (if (and (string= (nth 1 pair) "") (not exact-completion)) (setq jde-complete-current-list classinfo) (setq jde-complete-current-list (jde-complete-find-all-completions pair classinfo exact-completion))))))) nil))) (defun jde-complete-in-line () "Completes the method or field name at point. Repeating the command cycles through all potential completions for the name. This function displays the signature of a method completion as specified by `jde-complete-display-current-signature' This command uses the Beanshell to run Java code that in turn uses Java reflection to determine the methods and fields defined by the class of the object at point. This command starts the Beanshell if necessary. Hence, you may experience a slight delay when using this command for the first time in a session or when completing a field or method of an object that has many methods and fields. See `jde-complete-menu' for a version of this command that lets you select the desired completion from a popup menu." (interactive) (if (and jde-complete-current-list (markerp jde-parse-current-beginning) (markerp jde-parse-current-end) (marker-position jde-parse-current-beginning) (marker-position jde-parse-current-end) (>= (point) (marker-position jde-parse-current-beginning)) (<= (point) (marker-position jde-parse-current-end)) (eq last-command this-command)) (jde-complete-complete-cycle) ;;else (jde-complete-generic "in-line"))) (defun jde-complete-choose-completion (&optional title initial-input use-menu) "Display completions for the object at point in a menu if USE-MENU is nonil, otherwise in the minibuffer. The display comprises all of the possible completions for the object it was invoked on. To automatically split large menus this function use `imenu--mouse-menu' to handle the popup menu. initial-input, whatever the user typed before invoking the completion" (let (index-alist pair name) (setq index-alist jde-complete-current-list) (setq pair (if (= (length index-alist) 1) ;; if only one item match, return it (car index-alist) (if use-menu ;; delegates menu handling to imenu :-) (imenu--mouse-menu index-alist ;; Popup window at text cursor (jde-cursor-posn-as-event) (or title "Completion")) (assoc (completing-read (or title "Completion: ") index-alist nil ;;predicate nil ;;required-match initial-input) ;;initial-input index-alist)))) (setq name (cdr pair)) (setq jde-complete-current-signature (car pair)) (jde-complete-insert-completion name))) (defun jde-cursor-posn-as-event() "Returns the text cursor position as an EVENT on Emacs and the mouse cursor position on XEmacs." (if jde-xemacsp (let* ((mouse-pos (mouse-pixel-position)) (x (car (cdr mouse-pos))) (y (cdr (cdr mouse-pos)))) (make-event 'button-press `(button 1 modifiers nil x ,x y ,y))) (let ((x (* (if jde-xemacsp (frame-width) (frame-char-width)) (if (and (boundp 'hscroll-mode) (fboundp 'hscroll-window-column)) (hscroll-window-column) (mod (current-column) (window-width))))) (y (* (if jde-xemacsp (frame-height) (frame-char-height)) (- (count-lines (point-min) (point)) (count-lines (point-min) (window-start))))) (window (get-buffer-window (current-buffer)))) (list (list x y) window)))) (defun jde-complete-menu () "Completes the method or field name at point. This command displays a popup menu listing the potential completions for the name at point. Selecting a completion causes the command to use the completion to complete the name at point. See `jde-complete-in-line' for a version of this command that lets you cycle throught the potential completions at point." (interactive) (jde-complete-generic t)) (defun jde-complete-minibuf () "Completes the method or field name at point. This command displays a popup menu listing the potential completions for the name at point. Selecting a completion causes the command to use the completion to complete the name at point. See `jde-complete-in-line' for a version of this command that lets you cycle throught the potential completions at point." (interactive) (jde-complete-generic nil)) (defun jde-complete-generic (completion-type) "Generic implementation for jde-complete methods" (let* ((pair (jde-parse-java-variable-at-point)) jde-parse-attempted-to-import) ;;resetting jde-complete-current-list (setq jde-complete-current-list nil) (if pair (condition-case err (jde-complete-pair (jde-complete-get-pair pair nil) completion-type) (error (condition-case err (jde-complete-pair (jde-complete-get-pair pair t) completion-type)) (error (message "%s" (error-message-string err))))) (message "No completion at this point")))) (defun jde-complete-pair (pair completion-type) (let ((access (jde-complete-get-access pair)) completion-list) (progn (if access (setq completion-list (jde-complete-find-completion-for-pair pair nil access)) (setq completion-list (jde-complete-find-completion-for-pair pair))) ;;if the completion list is nil check if the method is in the current ;;class(this) (if (null completion-list) (setq completion-list (jde-complete-find-completion-for-pair (list (concat "this." (car pair)) "") nil jde-complete-private))) ;;if completions is still null check if the method is in the ;;super class (if (null completion-list) (setq completion-list (jde-complete-find-completion-for-pair (list (concat "super." (car pair)) "") nil jde-complete-protected))) (if completion-list (let ((title (concat (car pair) "." (car (cdr pair)) "[...]"))) (if (null completion-type) (jde-complete-choose-completion title (car (cdr pair))) (if (string= completion-type "in-line") (progn (setq jde-complete-current-list-index -1) (jde-complete-complete-cycle)) (jde-complete-choose-completion title (car (cdr pair)) t)))) (error "No completion at this point"))))) (defun jde-complete-get-access (pair) (let (access) (if (string= (car pair) "this") (setq access jde-complete-private) (if (string= (car pair) "super") (setq access jde-complete-protected))) access)) (defun jde-complete-get-pair (pair op) (let ((tmp (list (car pair) (cadr pair)))) (if (and op (string= (car tmp) "" ) (not (string= (cadr tmp) ""))) (setcar tmp (cadr tmp))) (if (string= (car tmp) "" ) (setcar tmp "this")) tmp)) (defun jde-complete () "Displays completions for the Java symbol at point. This command delegates the task of displaying completions to the function specified by `jde-complete-function'. This allows you to select or specify the default method for displaying completions." (interactive) (call-interactively jde-complete-function)) (define-mode-overload-implementation semantic-idle-summary-current-symbol-info jde-mode () "Collect information on current symbol." (or jde-complete-display-signature (semantic-idle-summary-current-symbol-info-default))) (defun jde-complete-popup-message (message buffer-or-name) "Split up the current window horizontally, the new buffer is exactly 2 lines in height. Message is inserted in the new buffer. Succesive calls to this method with the same buffer-or-name will delete the text inside the buffer and replace it with message. Message should not be longer than a line." (interactive) (let* ((popup (get-buffer-window buffer-or-name));;last popup window (new (get-buffer-create buffer-or-name));;new popup window (current (current-buffer));;current buffer (min window-min-height);;current window-min-height (w (selected-window));;selected window (height (window-height w));;window height w2) (if popup (delete-window popup));;deleting previous windows (set-window-buffer w new);;set buffer of current window to be the new (erase-buffer) (insert message);;insert message (setq window-min-height 2);;set the minimum height (setq w2 (split-window w (- height 2)));;split windows (set-window-buffer w current);;restore the buffer in the current window (if (> (window-height w2) 2);;if the popup window is larger than 2 (enlarge-window (- (window-height w2) 2)));;resize it to 2 (setq window-min-height min);; restore window-min-height (run-at-time jde-complete-signature-display-time nil 'delete-window w2)));;set timer to delete popup window (defun jde-complete-display-current-signature() "Displays the current signature: `jde-complete-current-signature'. The display mode will depend on the variable `jde-complete-signature-display'" (interactive) (if jde-complete-current-signature (let ((display (car jde-complete-signature-display))) (cond ((string= display "Eldoc") ;;eldoc (setq jde-complete-display-signature t) (run-at-time jde-complete-signature-display-time nil `set-variable `jde-complete-display-signature nil) (eldoc-message jde-complete-current-signature)) ;;use buffer ((string= display "Buffer") (jde-complete-popup-message jde-complete-current-signature jde-complete-signature-buffer)) (t));; do nothing ))) (defun jde-complete-place-cursor () "Places the cursor in between the parenthesis after a completion. This is only done for methods that contain parameters, for all the other completions the cursor is place at the end." (let ((end-paren (string-match ")" jde-complete-current-signature)) (start-paren (string-match "(" jde-complete-current-signature))) (if (and end-paren start-paren (not (= start-paren (- end-paren 1)))) (goto-char (- (point) 1))))) (provide 'jde-complete) ;; $Log: jde-complete.el,v $ ;; Revision 1.84 2004/08/23 12:13:00 paulk ;; Change name of jde-complete-invoke-getClassInfo to ;; jde-complete-invoke-get-class-info to conform to Lisp naming ;; conventions. Have this function invoke jde-jeval-r instead of ;; bsh-eval-r to ensure that class info is updated when starting ;; Beanshell. ;; ;; Revision 1.83 2004/06/13 05:02:14 paulk ;; Update to use semantic-idle-summary mode to display signature of symbol at point. TThanks to David Ponce. ;; ;; Revision 1.82 2004/06/06 04:55:47 paulk ;; Remove references to obsolete (i.e., pre-cedet 1.0) semantic symbols in eldoc ;; display code. These are no longer necessary because the JDEE now ;; requires cedet and they cause distracting compile warnings. ;; ;; Revision 1.81 2004/04/29 02:55:53 paulk ;; Patch from David Ponce to to make eldoc-like stuff compatible with Semantic version 2 (cedet 1.0). ;; ;; Revision 1.80 2003/06/07 04:03:01 paulk ;; Prevent jde-mode from disabling the eldoc facility as ;; soon as eldoc delays display of a message, which made semantic eldoc ;; extensions practically unusable in Java buffers. Thanks to David Ponce. ;; ;; Revision 1.79 2003/02/25 05:32:32 paulk ;; Fixes defadvice bug that disables eldoc-print-current-symbol-info ;; in all modes but emacs-lisp-mode and jde-mode. Thanks to ABE Yasushi. ;; ;; Revision 1.78 2003/02/18 02:09:40 jslopez ;; Fixes regression bugs. ;; ;; Revision 1.77 2002/10/04 02:06:47 jslopez ;; Fixes logic building completion. ;; ;; Revision 1.76 2002/09/11 03:26:49 paulk ;; Unconditionalize requires for eldoc and senator so the using completion does not require compiling completion.el. ;; ;; Revision 1.75 2002/08/07 06:36:19 paulk ;; Removed code intended to eliminate spurious warnings when byte-compiling the JDEE. The ;; removed code now resides in a separate package, jde-compat.el. Thanks to Andy Piper ;; for suggesting this restructuring. Also fixed a number of compiler warnings caused ;; by genuine bugs. ;; ;; Revision 1.74 2002/07/13 16:53:35 jslopez ;; Fixes bug that was causing completion to fail. ;; ;; Revision 1.73 2002/07/13 16:36:21 jslopez ;; Fixes bug in completion that was not appending all exceptions to the ;; possible completions. ;; ;; Revision 1.72 2002/06/26 04:54:58 paulk ;; Reimplemented jde-complete-function as a choice widget so that it's value can ;; be a function symbol rather than a function symbol wrapped in a list. ;; ;; Revision 1.71 2002/06/23 17:29:35 jslopez ;; Integrates all the common functionality from jde-complete-in-line, ;; jde-complete-minibuffer, and jde-complete-menu into a method ;; jde-complete-generic. ;; ;; Fixes regresion bug. Caused when trying to complete member within this. ;; i.e. get (trying to complete methods starting with get was failing) ;; ;; Revision 1.70 2002/05/31 18:57:42 mnl ;; The possible completions displayed in popup menus can get quite long ;; (fully qualified parameter types, return type and throw ;; clauses). Added new customization variables jde-display-... that allow ;; omitting some of the details. ;; ;; Revision 1.69 2002/05/25 19:54:24 jslopez ;; Fixes bug displaying constructor information. ;; ;; Revision 1.68 2002/05/25 19:42:11 jslopez ;; Fixes bug in completion. ;; Now you can complete String ;; ^ ;; And the possible constructor will show up. ;; ;; Revision 1.67 2002/05/25 19:00:49 jslopez ;; Removes package information from the constructor completion. ;; ;; Revision 1.66 2002/05/25 18:37:40 jslopez ;; Adds constructors to completion. ;; ;; Revision 1.65 2002/05/25 18:26:37 jslopez ;; Moves routine to get the methods from ;; jde-complete-build-completion-list into jde-complete-get-methods. ;; ;; Revision 1.64 2002/05/25 18:03:37 jslopez ;; Moves routing to get the inner classes from ;; jde-complete-build-completion-list into jde-complete-get-inner-classes. ;; ;; Revision 1.63 2002/05/25 17:54:38 jslopez ;; More formatting changes. ;; ;; Revision 1.62 2002/05/25 17:18:42 jslopez ;; Formatting changes. ;; ;; Revision 1.61 2002/05/12 06:32:42 paulk ;; - Removed jde-complete-find-type (never used). ;; - Moved jde-complete-convert-args-to-types to jde-parse package as ;; jde-parse-convert-args-to-types ;; - Moved jde-complete-get-name-of-this-class to jde-parse package as ;; jde-parse-get-buffer-class ;; ;; Revision 1.60 2002/04/16 03:17:07 jslopez ;; Integrates jde-open-source. ;; ;; Revision 1.59 2002/04/15 15:45:36 jslopez ;; Fixes method signature for jde-complete-find-completion-for-pair. ;; ;; Revision 1.58 2002/04/14 13:59:24 paulk ;; Minor improvements to jde-complete-function customization prompts. ;; ;; Revision 1.57 2002/04/11 04:31:11 paulk ;; Turns out the JDEE already supported completion in the minibuffer. ;; Seems Stuart Popejoy was using an old version of the JDEE and used ;; an outdated version of jde-complete-menu as the based for his ;; version of jde-complete-minbuf. I've now replaced his version ;; with a new version based on Javier's latest version. I've ;; also eliminated the variable jde-complete-use-menu as ;; jde-complete-function provides the same functionality. ;; ;; Revision 1.56 2002/04/02 06:43:20 paulk ;; Minor docstring tweak. ;; ;; Revision 1.55 2002/04/02 06:41:11 paulk ;; - Added jde-complete-minibuf command contributed by Stuart Popejoy . ;; - Added jde-complete-select command and jde-complete-select-function variable. ;; ;; Revision 1.54 2002/02/14 01:52:50 jslopez ;; Fixes bug handling brackets i.e. array[ ;; ^ ;; ;; Revision 1.53 2001/11/30 18:09:44 jslopez ;; Updates constant jde-emacs21-p to jde-emacs21p ;; ;; Revision 1.52 2001/11/30 04:11:01 jslopez ;; Fixes compiler warnings. ;; ;; Revision 1.51 2001/11/30 02:50:02 jslopez ;; Improves error handling in the methods ;; jde-complete-at-point and jde-complete-at-point-menu ;; ;; Revision 1.50 2001/11/29 01:44:39 jslopez ;; Fixes bug in completion. Private members were not ;; being included when trying to complete an instance ;; of this in the same class. ;; ;; Revision 1.49 2001/10/28 16:07:49 jslopez ;; Updated documentation for jde-complete-use-menu. ;; ;; Revision 1.48 2001/10/24 03:12:28 jslopez ;; Modified the jde-complete-popup-completion-menu to add ;; the initial user input when using the buffer for completion. ;; Fixed bug that did not complete methods for variables names ;; that start with the '$' character. ;; ;; Revision 1.47 2001/10/21 14:32:05 jslopez ;; Added customization group jde-complete ;; and moved all the jde-complete customization under ;; that group. ;; ;; Revision 1.46 2001/10/21 13:12:25 jslopez ;; Adds customization variable jde-complete-use-menu. Modifies ;; jde-complete-popup-completion-menu to use the emacs completion buffer when ;; jde-complete-use-menu is nil. ;; ;; Revision 1.45 2001/10/08 13:08:21 paulk ;; Fixed bug that caused the in-buffer verion of completion to throw ;; an error for the first completion. Thanks to Jason Rumney. ;; ;; Revision 1.44 2001/10/03 13:43:25 jslopez ;; Fixed function `jde-complete-invoke-getClassInfo\' which failed on getting the class info from Completion.java because the class variable was not inside double quotes\. ;; ;; Revision 1.43 2001/10/03 05:26:14 paulk ;; Fixed function `jde-complete-invoke-getClassInfo' which failed on concat because ACCESS variable is an integer. Thanks to David Ponce. ;; ;; Revision 1.42 2001/10/02 23:27:44 jslopez ;; Updating the access level. It was still using 1 for private, and 0 for ;; protected. Now 3 is for private, 2 for default, 1 for protected, and 0 for ;; public. ;; ;; Revision 1.41 2001/09/28 21:18:13 jslopez ;; Updated my email address. ;; Fix bug that produce duplicates in the completion list ;; when class A has a protected method x and class be that ;; extends B overwrites method x and makes it public. ;; ;; Revision 1.40 2001/09/28 13:43:59 jslopez ;; Fixing typo. ;; ;; Revision 1.39 2001/09/28 13:31:36 jslopez ;; The completion code now has a notion of package protection. ;; Completion cache modified, now it store the public, protected, ;; package and private methods for each class. ;; ;; Revision 1.38 2001/09/07 14:07:21 jslopez ;; Added method jde-complete-place-cursor. This method is call after any ;; completion insertion to place the cursor in the right spot. i.e. in between ;; the parenthesis when the method have parameters and at the end in all other ;; cases. ;; ;; Revision 1.37 2001/09/02 03:38:23 jslopez ;; Added support to show exceptions when using the completion methods. ;; ;; Revision 1.36 2001/08/30 03:01:35 jslopez ;; Fixing typo. ;; ;; Revision 1.35 2001/08/30 02:34:50 jslopez ;; Added customization variable jde-complete-signature-display. ;; Changed the jde-complete-signature-display-time from 60 to 5. ;; Added methods jde-complete-popup-message and ;; jde-complete-display-current-signagute. ;; Modified the completion method to show the completion signature by using ;; jde-complete-display-current-signature. ;; ;; Revision 1.34 2001/08/15 05:15:38 paulk ;; Fixed bug that cause completion to return the wrong name when the class does not have ;; a package, e.g., .Test for Test.java. Thanks to Javier Lopez. ;; ;; Revision 1.33 2001/07/31 05:09:03 paulk ;; Fixes bug that prevented completion of variables in inner classes. Thanks to Javier Lopez. ;; ;; Revision 1.32 2001/07/21 03:49:17 paulk ;; Fixed missing first line. Fix for handling static inner classes was contributed by Javier Lopez. ;; ;; Revision 1.31 2001/07/21 03:46:12 paulk ;; Now handles completion for methods and fields of static inner classes. ;; ;; Revision 1.30 2001/07/18 01:45:38 paulk ;; Fixes bug in completion of inner classes. Thanks to Javier Lopez. ;; ;; Revision 1.29 2001/07/17 05:37:46 paulk ;; Bug fixes from Javier Lopez. ;; ;; Revision 1.28 2001/07/13 04:59:06 paulk ;; Bug fixes from Javier Lopez. ;; ;; Revision 1.27 2001/07/06 02:09:33 paulk ;; Many improvements. Thanks to Javier Lopez. ;; ;; Revision 1.26 2001/06/12 07:22:05 paulk ;; Completion now works for instances of inner classes in many cases. ;; The inner class must be declared before it is referenced for completion to work. ;; Future releases will fix this and other deficiencies in inner class handling. ;; Thanks to "Evan Easton" . ;; ;; Revision 1.25 2001/06/12 06:03:14 paulk ;; * Completion list now includes private methods of the current class and protected methods of ;; the super class. ;; ;; * Now sorts the possible completions in alphabetical order. ;; ;; * Fixes bug that prevented completion of super class methods, i.e. super.ge... ;; ;; Thanks to Javier Lopez ;; ;; Revision 1.24 2001/05/31 05:14:39 paulk ;; Provide support for per-project caching of class data in the Beanshell. Thanks to Matt Conway. ;; ;; Revision 1.23 2001/04/02 02:45:04 paulk ;; Add some comments. ;; ;; Revision 1.22 2001/03/15 19:47:07 paulk ;; Now pops up the completion menu at the text cursor position on Emacs and at the mouse cursor position on XEmacs. I will change this to popup at the text cursor position on XEmacs as well as soon as I can figure out how to do it. Thanks to Matt_Conway@i2.com for providing the Emacs version of this enhancement. ;; ;; Revision 1.21 2001/01/25 04:31:01 paulk ;; Completion now asks user whether to import a class that it cannot find. Thanks to Phillip Lord. ;; ;; Revision 1.20 2000/12/19 04:33:34 paulk ;; Fixed popup completion menu to work on XEmacs. Thanks to David Ponce for providing this fix. ;; ;; Revision 1.19 2000/10/25 02:52:16 paulk ;; Fixed bug where the completion function was completing symbols that it could not find with the results of the previous completion. ;; ;; Revision 1.18 2000/10/20 04:02:10 paulk ;; Now uses semantic for some functions. Thanks to David Ponce. ;; ;; Revision 1.17 2000/10/08 12:55:39 paulk ;; *** empty log message *** ;; ;; Revision 1.16 2000/09/30 17:00:20 paulk ;; Use imenu to display completion choices. Thanks to David Ponce. ;; ;; Revision 1.15 2000/08/19 07:07:05 paulk ;; Flushes cache at end of compilation. ;; ;; Revision 1.14 2000/08/11 05:15:05 paulk ;; Now flushes the classinfo cache at the end of a compilation. ;; ;; Revision 1.13 2000/08/10 08:48:49 paulk ;; Now handles primitive arrays correctly. ;; ;; Revision 1.12 2000/08/09 02:04:26 paulk ;; Adds support for completion of array instances. Thanks to Steff. ;; ;; Revision 1.11 2000/08/01 07:37:40 paulk ;; Now caches methods and fields for each class referenced in a session. Now completes private and protected methods and fields. Thanks to Stephane . ;; ;; Revision 1.10 2000/07/30 20:06:12 paulk ;; Updated doc for jde-complete-at-point and jde-complete-at-point-menu commands. ;; ;; Revision 1.9 2000/07/27 04:54:00 paulk ;; Now completes object fields to any depth and completes variables declared in method argument lists. Thanks to Stephane Nicolas . ;; ;; Revision 1.8 2000/07/26 14:42:23 paulk ;; Adds support for static fields and methods and completion of fields and methods of this ;; and super objects. Thanks to Stephane Nicolas for this enhancement. ;; ;; Revision 1.7 2000/06/01 05:52:25 paulk ;; Completion menu now works on XEmacs. ;; ;; Revision 1.6 2000/05/16 04:41:28 paulk ;; *** empty log message *** ;; ;; end of jde-complete.el