emacs.d/elpa/racket-mode-20200329.1841/racket-edit.el
2020-03-30 12:12:00 +02:00

405 lines
14 KiB
EmacsLisp

;;; racket-edit.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.
;; racket-mode per se, i.e. the mode for .rkt file buffers
(require 'cl-lib)
(require 'cl-macs)
(require 'comint)
(require 'racket-custom)
(require 'racket-cmd)
(require 'racket-common)
(require 'racket-complete)
(require 'racket-repl)
(require 'racket-util)
(require 'hideshow)
(defun racket-racket ()
"Do \"racket <file>\" in a shell buffer."
(interactive)
(racket--shell (concat (shell-quote-argument racket-program)
" "
(shell-quote-argument (racket--buffer-file-name)))))
(defun racket-raco-test ()
"Do \"raco test -x <file>\" in a shell buffer to run the \"test\" submodule."
(interactive)
(racket--shell (concat (shell-quote-argument racket-program)
" -l raco test -x "
(shell-quote-argument (racket--buffer-file-name)))))
(defun racket--shell (cmd)
(racket--save-if-changed)
(let ((w (selected-window)))
(pcase (get-buffer-window "*shell*" t)
(`() (other-window -1))
(win (select-window win)))
(with-temp-message cmd
(shell)
(pop-to-buffer-same-window "*shell*")
(comint-send-string "*shell*" (concat cmd "\n"))
(select-window w)
(sit-for 3))))
;;; code folding
;;;###autoload
(add-to-list 'hs-special-modes-alist
'(racket-mode "(" ")" ";" nil nil))
(defun racket--for-all-tests (verb f)
(save-excursion
(goto-char (point-min))
(let ((n 0))
(while (re-search-forward "^(module[+*]? test" (point-max) t)
(funcall f)
(cl-incf n)
(goto-char (match-end 0)))
(message "%s %d test submodules" verb n))))
(defun racket-fold-all-tests ()
"Fold (hide) all test submodules."
(interactive)
(racket--for-all-tests "Folded" 'hs-hide-block))
(defun racket-unfold-all-tests ()
"Unfold (show) all test submodules."
(interactive)
(racket--for-all-tests "Unfolded" 'hs-show-block))
;;; requires
(defun racket-tidy-requires ()
"Make a single top-level \"require\" form, modules sorted, one per line.
All top-level require forms are combined into a single form.
Within that form:
- A single subform is used for each phase level, sorted in this
order: for-syntax, for-template, for-label, for-meta, and
plain (phase 0).
- Within each level subform, the modules are sorted:
- Collection path modules -- sorted alphabetically.
- Subforms such as only-in.
- Quoted relative requires -- sorted alphabetically.
At most one module is listed per line.
Note: This only works for requires at the top level of a source
file using #lang. It does NOT work for require forms inside
module forms.
See also: `racket-trim-requires' and `racket-base-requires'."
(interactive)
(unless (eq major-mode 'racket-mode)
(user-error "Current buffer is not a racket-mode buffer"))
(pcase (racket--top-level-requires 'find)
(`nil (user-error "The file module has no requires; nothing to do"))
(reqs (racket--cmd/async
nil
`(requires/tidy ,reqs)
(lambda (result)
(pcase result
("" nil)
(new (goto-char (racket--top-level-requires 'kill))
(insert (concat new "\n")))))))))
(defun racket-trim-requires ()
"Like `racket-tidy-requires' but also deletes unnecessary requires.
Note: This only works when the source file can be fully expanded
with no errors.
Note: This only works for requires at the top level of a source
file using #lang. It does NOT work for require forms inside
module forms. Furthermore, it is not smart about module+ or
module* forms -- it might delete top level requires that are
actually needed by such submodules.
See also: `racket-base-requires'."
(interactive)
(unless (eq major-mode 'racket-mode)
(user-error "Current buffer is not a racket-mode buffer"))
(when (racket--ok-with-module+*)
(racket--save-if-changed)
(pcase (racket--top-level-requires 'find)
(`nil (user-error "The file module has no requires; nothing to do"))
(reqs (racket--cmd/async
nil
`(requires/trim
,(racket--buffer-file-name)
,reqs)
(lambda (result)
(pcase result
(`nil (user-error "Syntax error in source file"))
("" (goto-char (racket--top-level-requires 'kill)))
(new (goto-char (racket--top-level-requires 'kill))
(insert (concat new "\n"))))))))))
(defun racket-base-requires ()
"Change from \"#lang racket\" to \"#lang racket/base\".
Adds explicit requires for imports that are provided by
\"racket\" but not by \"racket/base\".
This is a recommended optimization for Racket applications.
Avoiding loading all of \"racket\" can reduce load time and
memory footprint.
Also, as does `racket-trim-requires', this removes unneeded
modules and tidies everything into a single, sorted require form.
Note: This only works when the source file can be fully expanded
with no errors.
Note: This only works for requires at the top level of a source
file using #lang. It does NOT work for require forms inside
module forms. Furthermore, it is not smart about module+ or
module* forms -- it might delete top level requires that are
actually needed by such submodules.
Note: Currently this only helps change \"#lang racket\" to
\"#lang racket/base\". It does not help with other similar
conversions, such as changing \"#lang typed/racket\" to \"#lang
typed/racket/base\"."
(interactive)
(unless (eq major-mode 'racket-mode)
(user-error "Current buffer is not a racket-mode buffer"))
(when (racket--buffer-start-re "^#lang.*? racket/base$")
(user-error "Already using #lang racket/base. Nothing to change."))
(unless (racket--buffer-start-re "^#lang.*? racket$")
(user-error "File does not use use #lang racket. Cannot change."))
(when (racket--ok-with-module+*)
(racket--save-if-changed)
(let ((reqs (racket--top-level-requires 'find)))
(racket--cmd/async
nil
`(requires/base
,(racket--buffer-file-name)
,reqs)
(lambda (result)
(pcase result
(`nil (user-error "Syntax error in source file"))
(new (goto-char (point-min))
(re-search-forward "^#lang.*? racket$")
(insert "/base")
(goto-char (or (racket--top-level-requires 'kill)
(progn (insert "\n\n") (point))))
(unless (string= "" new)
(insert (concat new "\n"))))))))))
(defun racket--ok-with-module+* ()
(save-excursion
(goto-char (point-min))
(or (not (re-search-forward (rx ?\( "module" (or "+" "*")) nil t))
(prog1
(y-or-n-p "Analysis will be unreliable due to module+ or module* forms -- proceed anyway? ")
(message "")))))
(defun racket--buffer-start-re (re)
(save-excursion
(ignore-errors
(goto-char (point-min))
(re-search-forward re)
t)))
(defun racket--top-level-requires (what)
"Identify all top-level requires and do WHAT.
When WHAT is 'find, returns the top-level require forms.
When WHAT is 'kill, kill the top-level requires, returning the
location of the first one."
(save-excursion
(goto-char (point-min))
(let ((first-beg nil)
(requires nil))
(while (re-search-forward "^(require " nil t)
(let* ((beg (progn (up-list -1) (point)))
(end (progn (forward-sexp) (point)))
(str (buffer-substring-no-properties beg end))
(sexpr (read str)))
(unless first-beg (setq first-beg beg))
(setq requires (cons sexpr requires))
(when (eq 'kill what)
(kill-sexp -1)
(delete-blank-lines))))
(if (eq 'kill what) first-beg requires))))
;;; align
(defun racket-align ()
"Align values in the same column.
Useful for binding forms like \"let\" and \"parameterize\",
conditionals like \"cond\" and \"match\", association lists, and
any series of couples like the arguments to \"hash\".
Before choosing this command, put point on the first of a series
of \"couples\". A couple is:
- A list of two or more sexprs: \"[sexpr val sexpr ...]\".
- Two sexprs: \"sexpr val\".
Each \"val\" moves to the same column and is
`prog-indent-sexp'-ed (in case it is a multi-line form).
For example with point on the \"[\" before \"a\":
#+BEGIN_SRC racket
Before After
(let ([a 12] (let ([a 12]
[bar 23]) [bar 23])
....) ....)
'([a . 12] '([a . 12]
[bar . 23]) [bar . 23])
(cond [a? #t] (cond [a? #t]
[b? (f x [b? (f x
y)] y)]
[else #f]) [else #f])
#+END_SRC
Or with point on the quote before \"a\":
#+BEGIN_SRC racket
(list 'a 12 (list 'a 12
'bar 23) 'bar 23)
#+END_SRC
If more than one couple is on the same line, none are aligned,
because it is unclear where the value column should be. For
example the following form will not change; `racket-align' will
display an error message:
#+BEGIN_SRC racket
(let ([a 0][b 1]
[c 2]) error; unchanged
....)
#+END_SRC
When a couple's sexprs start on different lines, that couple is
ignored. Other, single-line couples in the series are aligned as
usual. For example:
#+BEGIN_SRC racket
(let ([foo (let ([foo
0] 0]
[bar 1] [bar 1]
[x 2]) [x 2])
....) ....)
#+END_SRC
See also: `racket-unalign'."
(interactive)
(save-excursion
(let ((listp (eq ?\( (char-syntax (char-after))))
(prev-line 0)
(max-col 0))
(racket--for-each-couple listp
(lambda ()
(setq max-col (max max-col (current-column)))
(let ((this-line (line-number-at-pos)))
(when (= prev-line this-line)
(user-error
"Can't align if any couples are on same line"))
(setq prev-line this-line))))
(racket--for-each-couple listp
(lambda ()
(indent-to max-col)
(prog-indent-sexp))))))
(defun racket-unalign ()
"The opposite of `racket-align'.
Effectively does M-x `just-one-space' and `prog-indent-sexp' for
each couple's value."
(interactive)
(save-excursion
(let ((listp (eq ?\( (char-syntax (char-after)))))
(racket--for-each-couple listp
(lambda ()
(just-one-space)
(prog-indent-sexp))))))
(defun racket--for-each-couple (listp f)
"Move point to each value sexp of a couple, and `funcall' F.
Only call F when the couple's sexprs are on the same line.
When LISTP is true, expects couples to be `[id val]`, else `id val`."
(save-excursion
(condition-case ()
(while t
(when listp
(down-list))
(forward-sexp)
(let ((line (line-number-at-pos)))
(forward-sexp)
(backward-sexp)
(when (= line (line-number-at-pos))
;; Defensive: Backup over any prefix or punctuation
;; chars just in case backward-sexp didn't (although it
;; should have if our syntax table is correct).
(while (memq (char-syntax (char-before)) '(?\' ?\.))
(goto-char (1- (point))))
(funcall f)))
;; On to the next couple...
(if listp
(up-list)
(forward-sexp)))
(scan-error nil))))
;;; Completion
(defvar racket--completion-candidates (list racket-type-list
racket-keywords
racket-builtins-1-of-2
racket-builtins-2-of-2))
(defun racket--completion-candidates-for-prefix (prefix)
(cl-reduce (lambda (results strs)
(append results (all-completions prefix strs)))
racket--completion-candidates
:initial-value ()))
(defun racket-complete-at-point ()
"A value for the variable `completion-at-point-functions'.
Completion candidates are drawn from the same symbols used for
font-lock. This is a static list. If you want dynamic, smarter
completion candidates, enable the minor mode `racket-xp-mode'."
(racket--call-with-completion-prefix-positions
(lambda (beg end)
(list beg
end
(completion-table-dynamic
#'racket--completion-candidates-for-prefix)
:predicate #'identity
:exclusive 'no))))
(provide 'racket-edit)
;; racket-edit.el ends here