emacs.d/elpa/racket-mode-20200412.1611/racket-describe.el

123 lines
4.6 KiB
EmacsLisp
Raw Normal View History

2020-03-24 18:20:37 +01:00
;;; racket-describe.el -*- lexical-binding: t -*-
;; Copyright (c) 2013-2020 by Greg Hendershott.
;; Portions Copyright (C) 1985-1986, 1999-2013 Free Software Foundation, Inc.
;; Author: Greg Hendershott
;; URL: https://github.com/greghendershott/racket-mode
;; License:
;; This 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. This 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. See
;; http://www.gnu.org/licenses/ for details.
(require 'racket-cmd)
(require 'racket-util)
(require 'racket-visit)
(require 'shr)
(defun racket--do-describe (how
repl-session-id
str
&optional
display-and-pop-to-p
visit-thunk
doc-thunk)
"Create a `racket-describe-mode' buffer.
HOW is supplied as the first argument to the back-end
\"describe\" command. See it for details that we don't need to
know or care about in this function.
STR is the string form of an identifier that is to be described.
DISPLAY-AND-POP-TO-P should be t for use by direct user commands
like `racket-xp-describe' and `racket-repl-describe' -- in which
the buffer is displayed -- and nil for use as
a :company-doc-buffer function.
VISIT-THUNK and DOC-THUNK are, when not nil, used to insert
\"Visit Definition\" and \"Documentation in Browser\" buttons.
Returns the buffer in which the description was written."
;; Work around what seems to be a bug with `shr-insert-document' --
;; elements are out of order when an existing Racket Describe buffer
;; hasn't had a `quit-window' -- by re-creating the buffer.
(with-current-buffer (racket--get-buffer-recreate "*Racket Describe*")
(let* ((html (racket--cmd/await repl-session-id
`(describe ,how ,str)))
;; Because shr removes leading &nbsp; from <td> elements --
;; which messes up the indentation of s-expressions
;; including contracts -- replace &nbsp with `spc' in the
;; source HTML. Below we'll replace `spc' with " " in the
;; result of `shr-insert-document'.
(spc (string #x2020)) ;unlikely character (hopefully)
(dom (with-temp-buffer
(insert html)
(goto-char (point-min))
(while (re-search-forward "&nbsp;" nil t)
(replace-match spc t t))
(libxml-parse-html-region (point-min) (point-max)))))
(racket-describe-mode)
(read-only-mode -1)
(let ((shr-use-fonts nil))
(shr-insert-document dom))
(goto-char (point-min))
(while (re-search-forward spc nil t)
(replace-match " " t t))
(when display-and-pop-to-p
(goto-char (point-max))
(insert "\n")
(when visit-thunk
(insert-text-button "Visit Definition"
'follow-link t
'action
(lambda (_btn) (funcall visit-thunk)))
(insert " "))
(when doc-thunk
(insert-text-button "Documentation in Browser"
'follow-link t
'action
(lambda (_btn) (funcall doc-thunk))))
(insert " [q]uit"))
(read-only-mode 1)
(goto-char (point-min))
(when display-and-pop-to-p
(display-buffer (current-buffer) t)
(pop-to-buffer (current-buffer))
(message "Type TAB to move to links, 'q' to restore previous window"))
(current-buffer))))
(defvar racket-describe-mode-map
(let ((m (make-sparse-keymap)))
(set-keymap-parent m special-mode-map)
(mapc (lambda (x)
(define-key m (kbd (car x)) (cadr x)))
'(("<tab>" racket-describe--next-button)
("S-<tab>" racket-describe--prev-button)))
m)
"Keymap for Racket Describe mode.")
(define-derived-mode racket-describe-mode special-mode
"RacketDescribe"
"Major mode for describing Racket functions.
\\{racket-describe-mode-map}"
(setq show-trailing-whitespace nil))
(defun racket-describe--next-button ()
(interactive)
(forward-button 1 t t))
(defun racket-describe--prev-button ()
(interactive)
(forward-button -1 t t))
(provide 'racket-describe)
;; racket-describe.el ends here