BREAKING CHANGE: This deprecates the IS-(MAC|WINDOWS|LINUX|BSD) family of global constants in favor of a native `featurep` check: IS-MAC -> (featurep :system 'macos) IS-WINDOWS -> (featurep :system 'windows) IS-LINUX -> (featurep :system 'linux) IS-BSD -> (featurep :system 'bsd) The constants will stick around until the v3 release so folks can still use it -- and there are still some modules that use it, but I'll phase those uses out gradually. Fix: #7479
443 lines
18 KiB
EmacsLisp
443 lines
18 KiB
EmacsLisp
;;; tools/lookup/autoload/lookup.el -*- lexical-binding: t; -*-
|
|
|
|
;;;###autodef
|
|
(defun set-lookup-handlers! (modes &rest plist)
|
|
"Define jump handlers for major or minor MODES.
|
|
|
|
A handler is either an interactive command that changes the current buffer
|
|
and/or location of the cursor, or a function that takes one argument: the
|
|
identifier being looked up, and returns either nil (failed to find it), t
|
|
(succeeded at changing the buffer/moving the cursor), or 'deferred (assume this
|
|
handler has succeeded, but expect changes not to be visible yet).
|
|
|
|
There are several kinds of handlers, which can be defined with the following
|
|
properties:
|
|
|
|
:definition FN
|
|
Run when jumping to a symbol's definition. Used by `+lookup/definition'.
|
|
:implementations FN
|
|
Run when looking for implementations of a symbol in the current project. Used
|
|
by `+lookup/implementations'.
|
|
:type-definition FN
|
|
Run when jumping to a symbol's type definition. Used by
|
|
`+lookup/type-definition'.
|
|
:references FN
|
|
Run when looking for usage references of a symbol in the current project. Used
|
|
by `+lookup/references'.
|
|
:documentation FN
|
|
Run when looking up documentation for a symbol. Used by
|
|
`+lookup/documentation'.
|
|
:file FN
|
|
Run when looking up the file for a symbol/string. Typically a file path. Used
|
|
by `+lookup/file'.
|
|
:xref-backend FN
|
|
Defines an xref backend for a major-mode. A :definition and :references
|
|
handler isn't necessary with a :xref-backend, but will have higher precedence
|
|
if they exist.
|
|
:async BOOL
|
|
Indicates that *all* supplied FNs are asynchronous. Note: lookups will not try
|
|
any handlers after async ones, due to their nature. To get around this, you
|
|
must write a specialized wrapper to await the async response, or use a
|
|
different heuristic to determine, ahead of time, whether the async call will
|
|
succeed or not.
|
|
|
|
If you only want to specify one FN is async, declare it inline instead:
|
|
|
|
(set-lookup-handlers! 'rust-mode
|
|
:definition '(racer-find-definition :async t))
|
|
|
|
Handlers can either be interactive or non-interactive. Non-interactive handlers
|
|
must take one argument: the identifier being looked up. This function must
|
|
change the current buffer or window or return non-nil when it succeeds.
|
|
|
|
If it doesn't change the current buffer, or it returns nil, the lookup module
|
|
will fall back to the next handler in `+lookup-definition-functions',
|
|
`+lookup-implementations-functions', `+lookup-type-definition-functions',
|
|
`+lookup-references-functions', `+lookup-file-functions' or
|
|
`+lookup-documentation-functions'.
|
|
|
|
Consecutive `set-lookup-handlers!' calls will overwrite previously defined
|
|
handlers for MODES. If used on minor modes, they are stacked onto handlers
|
|
defined for other minor modes or the major mode it's activated in.
|
|
|
|
This can be passed nil as its second argument to unset handlers for MODES. e.g.
|
|
|
|
(set-lookup-handlers! 'python-mode nil)
|
|
|
|
\(fn MODES &key DEFINITION IMPLEMENTATIONS TYPE-DEFINITION REFERENCES DOCUMENTATION FILE XREF-BACKEND ASYNC)"
|
|
(declare (indent defun))
|
|
(dolist (mode (ensure-list modes))
|
|
(let ((hook (intern (format "%s-hook" mode)))
|
|
(fn (intern (format "+lookup--init-%s-handlers-h" mode))))
|
|
(if (null (car plist))
|
|
(progn
|
|
(remove-hook hook fn)
|
|
(unintern fn nil))
|
|
(fset
|
|
fn
|
|
(lambda ()
|
|
(cl-destructuring-bind (&key definition implementations type-definition references documentation file xref-backend async)
|
|
plist
|
|
(cl-mapc #'+lookup--set-handler
|
|
(list definition
|
|
implementations
|
|
type-definition
|
|
references
|
|
documentation
|
|
file
|
|
xref-backend)
|
|
(list '+lookup-definition-functions
|
|
'+lookup-implementations-functions
|
|
'+lookup-type-definition-functions
|
|
'+lookup-references-functions
|
|
'+lookup-documentation-functions
|
|
'+lookup-file-functions
|
|
'xref-backend-functions)
|
|
(make-list 5 async)
|
|
(make-list 5 (or (eq major-mode mode)
|
|
(and (boundp mode)
|
|
(symbol-value mode))))))))
|
|
(add-hook hook fn)))))
|
|
|
|
|
|
;;
|
|
;;; Helpers
|
|
|
|
(defun +lookup--set-handler (spec functions-var &optional async enable)
|
|
(when spec
|
|
(cl-destructuring-bind (fn . plist)
|
|
(ensure-list spec)
|
|
(if (not enable)
|
|
(remove-hook functions-var fn 'local)
|
|
(put fn '+lookup-async (or (plist-get plist :async) async))
|
|
(add-hook functions-var fn nil 'local)))))
|
|
|
|
(defun +lookup--run-handler (handler identifier)
|
|
(if (commandp handler)
|
|
(call-interactively handler)
|
|
(funcall handler identifier)))
|
|
|
|
(defun +lookup--run-handlers (handler identifier origin)
|
|
(doom-log "Looking up '%s' with '%s'" identifier handler)
|
|
(condition-case-unless-debug e
|
|
(let ((wconf (current-window-configuration))
|
|
(result (condition-case-unless-debug e
|
|
(+lookup--run-handler handler identifier)
|
|
(error
|
|
(doom-log "Lookup handler %S threw an error: %s" handler e)
|
|
'fail))))
|
|
(cond ((eq result 'fail)
|
|
(set-window-configuration wconf)
|
|
nil)
|
|
((or (get handler '+lookup-async)
|
|
(eq result 'deferred)))
|
|
((or result
|
|
(null origin)
|
|
(/= (point-marker) origin))
|
|
(prog1 (point-marker)
|
|
(set-window-configuration wconf)))))
|
|
((error user-error)
|
|
(message "Lookup handler %S: %s" handler e)
|
|
nil)))
|
|
|
|
(defun +lookup--jump-to (prop identifier &optional display-fn arg)
|
|
(let* ((origin (point-marker))
|
|
(handlers
|
|
(plist-get (list :definition '+lookup-definition-functions
|
|
:implementations '+lookup-implementations-functions
|
|
:type-definition '+lookup-type-definition-functions
|
|
:references '+lookup-references-functions
|
|
:documentation '+lookup-documentation-functions
|
|
:file '+lookup-file-functions)
|
|
prop))
|
|
(result
|
|
(if arg
|
|
(if-let
|
|
(handler
|
|
(intern-soft
|
|
(completing-read "Select lookup handler: "
|
|
(delete-dups
|
|
(remq t (append (symbol-value handlers)
|
|
(default-value handlers))))
|
|
nil t)))
|
|
(+lookup--run-handlers handler identifier origin)
|
|
(user-error "No lookup handler selected"))
|
|
(run-hook-wrapped handlers #'+lookup--run-handlers identifier origin))))
|
|
(unwind-protect
|
|
(when (cond ((null result)
|
|
(message "No lookup handler could find %S" identifier)
|
|
nil)
|
|
((markerp result)
|
|
(funcall (or display-fn #'switch-to-buffer)
|
|
(marker-buffer result))
|
|
(goto-char result)
|
|
result)
|
|
(result))
|
|
(with-current-buffer (marker-buffer origin)
|
|
(better-jumper-set-jump (marker-position origin)))
|
|
result)
|
|
(set-marker origin nil))))
|
|
|
|
|
|
;;
|
|
;;; Lookup backends
|
|
|
|
(autoload 'xref--show-defs "xref")
|
|
(defun +lookup--xref-show (fn identifier &optional show-fn)
|
|
(let ((xrefs (funcall fn
|
|
(xref-find-backend)
|
|
identifier)))
|
|
(when xrefs
|
|
(let* ((jumped nil)
|
|
(xref-after-jump-hook
|
|
(cons (lambda () (setq jumped t))
|
|
xref-after-jump-hook)))
|
|
(funcall (or show-fn #'xref--show-defs)
|
|
(lambda () xrefs)
|
|
nil)
|
|
(if (cdr xrefs)
|
|
'deferred
|
|
jumped)))))
|
|
|
|
(defun +lookup-dictionary-definition-backend-fn (identifier)
|
|
"Look up dictionary definition for IDENTIFIER."
|
|
(when (derived-mode-p 'text-mode)
|
|
(+lookup/dictionary-definition identifier)
|
|
'deferred))
|
|
|
|
(defun +lookup-thesaurus-definition-backend-fn (identifier)
|
|
"Look up synonyms for IDENTIFIER."
|
|
(when (derived-mode-p 'text-mode)
|
|
(+lookup/synonyms identifier)
|
|
'deferred))
|
|
|
|
(defun +lookup-xref-definitions-backend-fn (identifier)
|
|
"Non-interactive wrapper for `xref-find-definitions'"
|
|
(condition-case _
|
|
(+lookup--xref-show 'xref-backend-definitions identifier #'xref--show-defs)
|
|
(cl-no-applicable-method nil)))
|
|
|
|
(defun +lookup-xref-references-backend-fn (identifier)
|
|
"Non-interactive wrapper for `xref-find-references'"
|
|
(condition-case _
|
|
(+lookup--xref-show 'xref-backend-references identifier #'xref--show-xrefs)
|
|
(cl-no-applicable-method nil)))
|
|
|
|
(defun +lookup-dumb-jump-backend-fn (_identifier)
|
|
"Look up the symbol at point (or selection) with `dumb-jump', which conducts a
|
|
project search with ag, rg, pt, or git-grep, combined with extra heuristics to
|
|
reduce false positives.
|
|
|
|
This backend prefers \"just working\" over accuracy."
|
|
(and (require 'dumb-jump nil t)
|
|
(dumb-jump-go)))
|
|
|
|
(defun +lookup-project-search-backend-fn (identifier)
|
|
"Conducts a simple project text search for IDENTIFIER.
|
|
|
|
Uses and requires `+ivy-file-search', `+helm-file-search', or `+vertico-file-search'.
|
|
Will return nil if neither is available. These require ripgrep to be installed."
|
|
(unless identifier
|
|
(let ((query (rxt-quote-pcre identifier)))
|
|
(ignore-errors
|
|
(cond ((modulep! :completion ivy)
|
|
(+ivy-file-search :query query)
|
|
t)
|
|
((modulep! :completion helm)
|
|
(+helm-file-search :query query)
|
|
t)
|
|
((modulep! :completion vertico)
|
|
(+vertico-file-search :query query)
|
|
t))))))
|
|
|
|
(defun +lookup-evil-goto-definition-backend-fn (_identifier)
|
|
"Uses `evil-goto-definition' to conduct a text search for IDENTIFIER in the
|
|
current buffer."
|
|
(when (fboundp 'evil-goto-definition)
|
|
(ignore-errors
|
|
(cl-destructuring-bind (beg . end)
|
|
(bounds-of-thing-at-point 'symbol)
|
|
(evil-goto-definition)
|
|
(let ((pt (point)))
|
|
(not (and (>= pt beg)
|
|
(< pt end))))))))
|
|
|
|
(defun +lookup-ffap-backend-fn (identifier)
|
|
"Tries to locate the file at point (or in active selection).
|
|
Uses find-in-project functionality (provided by ivy, helm, or project),
|
|
otherwise falling back to ffap.el (find-file-at-point)."
|
|
(let ((guess
|
|
(cond (identifier)
|
|
((doom-region-active-p)
|
|
(buffer-substring-no-properties
|
|
(doom-region-beginning)
|
|
(doom-region-end)))
|
|
((if (require 'ffap) (ffap-guesser)))
|
|
((thing-at-point 'filename t)))))
|
|
(cond ((and (stringp guess)
|
|
(or (file-exists-p guess)
|
|
(ffap-url-p guess)))
|
|
(find-file-at-point guess))
|
|
((and (modulep! :completion ivy)
|
|
(doom-project-p))
|
|
(counsel-file-jump guess (doom-project-root)))
|
|
((and (modulep! :completion vertico)
|
|
(doom-project-p))
|
|
(+vertico/consult-fd-or-find (doom-project-root) guess))
|
|
((find-file-at-point (ffap-prompter guess))))
|
|
t))
|
|
|
|
(defun +lookup-bug-reference-backend-fn (_identifier)
|
|
"Searches for a bug reference in user/repo#123 or #123 format and opens it in
|
|
the browser."
|
|
(require 'bug-reference)
|
|
(when (fboundp 'bug-reference-try-setup-from-vc)
|
|
(let ((old-bug-reference-mode bug-reference-mode)
|
|
(old-bug-reference-prog-mode bug-reference-prog-mode)
|
|
(bug-reference-url-format bug-reference-url-format)
|
|
(bug-reference-bug-regexp bug-reference-bug-regexp))
|
|
(bug-reference-try-setup-from-vc)
|
|
(unwind-protect
|
|
(let ((bug-reference-mode t)
|
|
(bug-reference-prog-mode nil))
|
|
(catch 'found
|
|
(bug-reference-fontify (line-beginning-position) (line-end-position))
|
|
(dolist (o (overlays-at (point)))
|
|
;; It should only be possible to have one URL overlay.
|
|
(when-let (url (overlay-get o 'bug-reference-url))
|
|
(browse-url url)
|
|
|
|
(throw 'found t)))))
|
|
;; Restore any messed up fontification as a result of this.
|
|
(bug-reference-unfontify (line-beginning-position) (line-end-position))
|
|
(if (or old-bug-reference-mode
|
|
old-bug-reference-prog-mode)
|
|
(bug-reference-fontify (line-beginning-position) (line-end-position)))))))
|
|
|
|
|
|
;;
|
|
;;; Main commands
|
|
|
|
;;;###autoload
|
|
(defun +lookup/definition (identifier &optional arg)
|
|
"Jump to the definition of IDENTIFIER (defaults to the symbol at point).
|
|
|
|
Each function in `+lookup-definition-functions' is tried until one changes the
|
|
point or current buffer. Falls back to dumb-jump, naive
|
|
ripgrep/the_silver_searcher text search, then `evil-goto-definition' if
|
|
evil-mode is active."
|
|
(interactive (list (doom-thing-at-point-or-region)
|
|
current-prefix-arg))
|
|
(cond ((null identifier) (user-error "Nothing under point"))
|
|
((+lookup--jump-to :definition identifier nil arg))
|
|
((user-error "Couldn't find the definition of %S" (substring-no-properties identifier)))))
|
|
|
|
;;;###autoload
|
|
(defun +lookup/implementations (identifier &optional arg)
|
|
"Jump to the implementations of IDENTIFIER (defaults to the symbol at point).
|
|
|
|
Each function in `+lookup-implementations-functions' is tried until one changes
|
|
the point or current buffer."
|
|
(interactive (list (doom-thing-at-point-or-region)
|
|
current-prefix-arg))
|
|
(cond ((null identifier) (user-error "Nothing under point"))
|
|
((+lookup--jump-to :implementations identifier nil arg))
|
|
((user-error "Couldn't find the implementations of %S" (substring-no-properties identifier)))))
|
|
|
|
;;;###autoload
|
|
(defun +lookup/type-definition (identifier &optional arg)
|
|
"Jump to the type definition of IDENTIFIER (defaults to the symbol at point).
|
|
|
|
Each function in `+lookup-type-definition-functions' is tried until one changes
|
|
the point or current buffer."
|
|
(interactive (list (doom-thing-at-point-or-region)
|
|
current-prefix-arg))
|
|
(cond ((null identifier) (user-error "Nothing under point"))
|
|
((+lookup--jump-to :type-definition identifier nil arg))
|
|
((user-error "Couldn't find the definition of %S" (substring-no-properties identifier)))))
|
|
|
|
;;;###autoload
|
|
(defun +lookup/references (identifier &optional arg)
|
|
"Show a list of usages of IDENTIFIER (defaults to the symbol at point)
|
|
|
|
Tries each function in `+lookup-references-functions' until one changes the
|
|
point and/or current buffer. Falls back to a naive ripgrep/the_silver_searcher
|
|
search otherwise."
|
|
(interactive (list (doom-thing-at-point-or-region)
|
|
current-prefix-arg))
|
|
(cond ((null identifier) (user-error "Nothing under point"))
|
|
((+lookup--jump-to :references identifier nil arg))
|
|
((user-error "Couldn't find references of %S" (substring-no-properties identifier)))))
|
|
|
|
;;;###autoload
|
|
(defun +lookup/documentation (identifier &optional arg)
|
|
"Show documentation for IDENTIFIER (defaults to symbol at point or selection.
|
|
|
|
First attempts the :documentation handler specified with `set-lookup-handlers!'
|
|
for the current mode/buffer (if any), then falls back to the backends in
|
|
`+lookup-documentation-functions'."
|
|
(interactive (list (doom-thing-at-point-or-region)
|
|
current-prefix-arg))
|
|
(cond ((+lookup--jump-to :documentation identifier #'pop-to-buffer arg))
|
|
((user-error "Couldn't find documentation for %S" (substring-no-properties identifier)))))
|
|
|
|
;;;###autoload
|
|
(defun +lookup/file (&optional path)
|
|
"Figure out PATH from whatever is at point and open it.
|
|
|
|
Each function in `+lookup-file-functions' is tried until one changes the point
|
|
or the current buffer.
|
|
|
|
Otherwise, falls back on `find-file-at-point'."
|
|
(interactive)
|
|
(cond ((and path
|
|
buffer-file-name
|
|
(file-equal-p path buffer-file-name)
|
|
(user-error "Already here")))
|
|
|
|
((+lookup--jump-to :file path))
|
|
|
|
((user-error "Couldn't find any files here"))))
|
|
|
|
|
|
;;
|
|
;;; Dictionary
|
|
|
|
;;;###autoload
|
|
(defun +lookup/dictionary-definition (identifier &optional arg)
|
|
"Look up the definition of the word at point (or selection)."
|
|
(interactive
|
|
(list (or (doom-thing-at-point-or-region 'word)
|
|
(if (equal major-mode 'pdf-view-mode)
|
|
(car (pdf-view-active-region-text)))
|
|
(read-string "Look up in dictionary: "))
|
|
current-prefix-arg))
|
|
(message "Looking up dictionary definition for %S" identifier)
|
|
(cond ((and (featurep :system 'macos) (require 'osx-dictionary nil t))
|
|
(osx-dictionary--view-result identifier))
|
|
((and +lookup-dictionary-prefer-offline
|
|
(require 'wordnut nil t))
|
|
(unless (executable-find wordnut-cmd)
|
|
(user-error "Couldn't find %S installed on your system"
|
|
wordnut-cmd))
|
|
(wordnut-search identifier))
|
|
((require 'define-word nil t)
|
|
(define-word identifier nil arg))
|
|
((user-error "No dictionary backend is available"))))
|
|
|
|
;;;###autoload
|
|
(defun +lookup/synonyms (identifier &optional _arg)
|
|
"Look up and insert a synonym for the word at point (or selection)."
|
|
(interactive
|
|
(list (doom-thing-at-point-or-region 'word) ; TODO actually use this
|
|
current-prefix-arg))
|
|
(message "Looking up synonyms for %S" identifier)
|
|
(cond ((and +lookup-dictionary-prefer-offline
|
|
(require 'synosaurus-wordnet nil t))
|
|
(unless (executable-find synosaurus-wordnet--command)
|
|
(user-error "Couldn't find %S installed on your system"
|
|
synosaurus-wordnet--command))
|
|
(synosaurus-choose-and-replace))
|
|
((require 'powerthesaurus nil t)
|
|
(powerthesaurus-lookup-word-dwim))
|
|
((user-error "No thesaurus backend is available"))))
|