summaryrefslogtreecommitdiff
path: root/lisp/emacs-lisp/find-func.el
diff options
context:
space:
mode:
Diffstat (limited to 'lisp/emacs-lisp/find-func.el')
-rw-r--r--lisp/emacs-lisp/find-func.el117
1 files changed, 80 insertions, 37 deletions
diff --git a/lisp/emacs-lisp/find-func.el b/lisp/emacs-lisp/find-func.el
index 82132f27ff6..9b98f05ae81 100644
--- a/lisp/emacs-lisp/find-func.el
+++ b/lisp/emacs-lisp/find-func.el
@@ -43,6 +43,8 @@
;;; Code:
+(eval-when-compile (require 'cl-lib))
+
;;; User variables:
(defgroup find-function nil
@@ -182,15 +184,15 @@ See the functions `find-function' and `find-variable'."
LIBRARY should be a string (the name of the library)."
;; If the library is byte-compiled, try to find a source library by
;; the same name.
- (if (string-match "\\.el\\(c\\(\\..*\\)?\\)\\'" library)
- (setq library (replace-match "" t t library)))
+ (when (string-match "\\.el\\(c\\(\\..*\\)?\\)\\'" library)
+ (setq library (replace-match "" t t library)))
(or
(locate-file library
- (or find-function-source-path load-path)
- (find-library-suffixes))
+ (or find-function-source-path load-path)
+ (find-library-suffixes))
(locate-file library
- (or find-function-source-path load-path)
- load-file-rep-suffixes)
+ (or find-function-source-path load-path)
+ load-file-rep-suffixes)
(when (file-name-absolute-p library)
(let ((rel (find-library--load-name library)))
(when rel
@@ -201,8 +203,22 @@ LIBRARY should be a string (the name of the library)."
(locate-file rel
(or find-function-source-path load-path)
load-file-rep-suffixes)))))
+ (find-library--from-load-history library)
(error "Can't find library %s" library)))
+(defun find-library--from-load-history (library)
+ ;; In `load-history', the file may be ".elc", ".el", ".el.gz", and
+ ;; LIBRARY may be "foo.el" or "foo".
+ (let ((load-re
+ (concat "\\(" (regexp-quote (file-name-sans-extension library)) "\\)"
+ (regexp-opt (get-load-suffixes)) "\\'")))
+ (cl-loop
+ for (file . _) in load-history thereis
+ (and (stringp file) (string-match load-re file)
+ (let ((dir (substring file 0 (match-beginning 1)))
+ (basename (match-string 1 file)))
+ (locate-file basename (list dir) (find-library-suffixes)))))))
+
(defvar find-function-C-source-directory
(let ((dir (expand-file-name "src" source-directory)))
(if (file-accessible-directory-p dir) dir))
@@ -257,37 +273,64 @@ TYPE should be nil to find a function, or `defvar' to find a variable."
;;;###autoload
(defun find-library (library)
"Find the Emacs Lisp source of LIBRARY.
-LIBRARY should be a string (the name of the library)."
- (interactive
- (let* ((dirs (or find-function-source-path load-path))
- (suffixes (find-library-suffixes))
- (table (apply-partially 'locate-file-completion-table
- dirs suffixes))
- (def (if (eq (function-called-at-point) 'require)
- ;; `function-called-at-point' may return 'require
- ;; with `point' anywhere on this line. So wrap the
- ;; `save-excursion' below in a `condition-case' to
- ;; avoid reporting a scan-error here.
- (condition-case nil
- (save-excursion
- (backward-up-list)
- (forward-char)
- (forward-sexp 2)
- (thing-at-point 'symbol))
- (error nil))
- (thing-at-point 'symbol))))
- (when (and def (not (test-completion def table)))
- (setq def nil))
- (list
- (completing-read (if def (format "Library name (default %s): " def)
- "Library name: ")
- table nil nil nil nil def))))
- (let ((buf (find-file-noselect (find-library-name library))))
- (condition-case nil
- (prog1
- (switch-to-buffer buf)
- (run-hooks 'find-function-after-hook))
- (error (pop-to-buffer buf)))))
+
+Interactively, prompt for LIBRARY using the one at or near point."
+ (interactive (list (read-library-name)))
+ (prog1
+ (switch-to-buffer (find-file-noselect (find-library-name library)))
+ (run-hooks 'find-function-after-hook)))
+
+(defun read-library-name ()
+ "Read and return a library name, defaulting to the one near point.
+
+A library name is the filename of an Emacs Lisp library located
+in a directory under `load-path' (or `find-function-source-path',
+if non-nil)."
+ (let* ((dirs (or find-function-source-path load-path))
+ (suffixes (find-library-suffixes))
+ (table (apply-partially 'locate-file-completion-table
+ dirs suffixes))
+ (def (if (eq (function-called-at-point) 'require)
+ ;; `function-called-at-point' may return 'require
+ ;; with `point' anywhere on this line. So wrap the
+ ;; `save-excursion' below in a `condition-case' to
+ ;; avoid reporting a scan-error here.
+ (condition-case nil
+ (save-excursion
+ (backward-up-list)
+ (forward-char)
+ (forward-sexp 2)
+ (thing-at-point 'symbol))
+ (error nil))
+ (thing-at-point 'symbol))))
+ (when (and def (not (test-completion def table)))
+ (setq def nil))
+ (completing-read (if def
+ (format "Library name (default %s): " def)
+ "Library name: ")
+ table nil nil nil nil def)))
+
+;;;###autoload
+(defun find-library-other-window (library)
+ "Find the Emacs Lisp source of LIBRARY in another window.
+
+See `find-library' for more details."
+ (interactive (list (read-library-name)))
+ (prog1
+ (switch-to-buffer-other-window (find-file-noselect
+ (find-library-name library)))
+ (run-hooks 'find-function-after-hook)))
+
+;;;###autoload
+(defun find-library-other-frame (library)
+ "Find the Emacs Lisp source of LIBRARY in another frame.
+
+See `find-library' for more details."
+ (interactive (list (read-library-name)))
+ (prog1
+ (switch-to-buffer-other-frame (find-file-noselect
+ (find-library-name library)))
+ (run-hooks 'find-function-after-hook)))
;;;###autoload
(defun find-function-search-for-symbol (symbol type library)