This repository has been archived by the owner on Aug 21, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathspiral-mode.el
598 lines (509 loc) · 24.7 KB
/
spiral-mode.el
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
;;; spiral-mode.el --- SPIRAL minor mode for interactions with Socket Server -*- lexical-binding: t; -*-
;;
;; Filename: spiral-mode.el
;; Description:
;; Author: Daniel Barreto <[email protected]>
;; Maintainer: Daniel Barreto <[email protected]>
;; Copyright (C) 2017 Daniel Barreto
;; Created: Sun Nov 12 12:25:44 2017 (+0100)
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;;; Commentary:
;;
;; SPIRAL minor mode for interactions with Socket Server
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; This program 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 3 of the License, or (at
;; your option) any later version.
;;
;; This program 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.
;;
;; You should have received a copy of the GNU General Public License
;; along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;;; Code:
(require 'spiral-ast)
(require 'spiral-project)
(require 'spiral-overlay)
(require 'spiral-util)
(defcustom spiral-ask-for-connection t
"Automatically ask for host:port when trying to interact with UNREPL in an unconnected buffer."
:type 'boolean
:group 'spiral)
(defcustom spiral-display-repl-in-current-window nil
"Whether to display the REPL in the current window."
:type 'boolean
:group 'spiral-repl)
(defcustom spiral-mode-line-show-connection t
"Whether to have mode-line show the connection id or not."
:type 'boolean
:group 'spiral)
(defcustom spiral-mode-line
'(:eval (format " 🌀[%s]" (or spiral-conn-id "-")))
"Mode line lighter for `spiral-mode'."
:type 'sexp
:risky t
:group 'spiral)
(defcustom spiral-eval-result-display 'both
"Whether to display evaluation results with overlays, in the echo area, or both."
:type '(choice (const :tag "End of line" overlay)
(const :tag "Echo area" echo)
(const :tag "Both" both))
:group 'spiral)
(defcustom spiral-automatic-ns-sync 'do-it-and-notify
"Whether to automatically sync ns in interactive evaluations."
:type '(choice (const :tag "Do sync and notify in the REPL buffer" do-it-and-notify)
(const :tag "Do sync and skip notification" do-it-without-notify)
(const :tag "Do not sync and notify in the REPL buffer" avoid-and-notify)
(const :tag "Do not sync and skip notification" avoid-without-notify))
:group 'spiral)
(defcustom spiral-auto-mode t
"Whether or not to automatically enable `spiral-mode' for all Clojure buffer."
:type 'boolean
:group 'spiral)
(defvar spiral-completion-last-context nil
"Last compliment context used.")
(defvar-local spiral-conn-id nil
"Port number used when creating a new Socket REPL.")
(defvar-local spiral-latest-eval nil
"Latest evaluation for the current buffer.")
;; Helpers
;; -------------------------------------------------------------------
(defun spiral-mode--find-project-by-file-name (&optional allow-nil)
"Check the current buffer file name and try to find a matching SPIRAL project.
If ALLOW-NIL is non-nil, allows returning projects with nil directory.
If there is more than one project for this buffers file name, return the
most recently created."
(let ((project-dir (spiral-clojure-dir)))
(when (or project-dir allow-nil)
(spiral-projects-get-by-dir project-dir))))
(defun spiral-mode-enable-auto ()
"Automatically enable SPIRAL's minor mode in every new Clojure buffer.
Setup a `clojure-mode-hook' that checks for a possible project connection
each time a new Clojure buffer gets opened."
(add-hook 'clojure-mode-hook #'spiral-mode-conditionally-turn-on))
(defun spiral-mode--turn-on (project)
"Turn on `spiral-mode' in current buffer and associate PROJECT to it."
(setq-local spiral-conn-id (spiral-project-id project))
(spiral-mode t)
project)
(defun spiral-mode-turn-on (conn-id-or-project &optional buffer)
"Turn on `spiral-mode' in BUFFER and associate CONN-ID-OR-PROJECT to it.
If BUFFER is nil, use current buffer.
Return the connected project."
(with-current-buffer (or buffer (current-buffer))
(let ((project (if (symbolp conn-id-or-project)
(spiral-projects-get conn-id-or-project))))
(spiral-mode--turn-on project))))
(defun spiral-mode-conditionally-turn-on ()
"Turn on `spiral-mode' only if variable `buffer-file-name' belongs to an existing project."
(when-let (project (spiral-mode--find-project-by-file-name))
(spiral-mode--turn-on project)))
(declare-function spiral--connection-prompt "spiral")
(defun spiral-ensure-connected! ()
"Make sure a `spiral-conn-id' exists for current buffer.
If this local variable is not already set, tries to find a good candidate
by looking at the buffer's file path and comparing to existing
`spiral-projects'.
If that fails and `spiral-ask-for-connection' is non-nil, asks the user for
an *existing* host:port connection to connect to. If everything else
fails, raise an error.
Return a SPIRAL project"
(if-let (project (and spiral-conn-id
(spiral-projects-get spiral-conn-id)))
project
(if-let (project (spiral-mode-conditionally-turn-on))
project
(if spiral-ask-for-connection
(seq-let [project _] (spiral--connection-prompt (spiral-clojure-dir))
(spiral-mode--turn-on project))
(error "Could not find a SPIRAL connection for this buffer")))))
(declare-function spiral-repl-disconnect-message "unrepl-repl")
(defun spiral-disconnect (conn-id &optional message)
"Quit a project for CONN-ID, and possibly kill its REPL buffer.
If MESSAGE is non-nil, it will be displayed at the end of the REPL buffer,
which won't be automatically killed."
(let* ((project (spiral-projects-get conn-id))
(repl-buffer (spiral-project-repl-buffer project)))
(if message
(progn
(spiral-repl-disconnect-message conn-id message)
(spiral-project-quit project))
(when repl-buffer
(kill-buffer repl-buffer)))))
(defmacro with-current-project (&rest body)
"Ensure the current buffer is connected and put its project in BODY's local scope."
`(let ((project (spiral-ensure-connected!)))
,@body))
(declare-function spiral-aux-sync-request "spiral-loop")
(defun spiral-update-print-settings (project context coll-length nesting-depth string-length
&optional async)
"Adjust UNREPL print settings for PROJECT's CONTEXT.
This function acts synchronously, and returns the previous print limits,
ASYNC overrides this behavior.
COLL-LENGTH, NESTING-DEPTH, and STRING-LENGTH are UNREPL's update-able
settings.
If any of them is the symbol `max', the symbol `Long/MAX_VALUE' will be
sent instead.
If any of them is the symbol `same', the value will be unaltered in
UNREPL."
(let* ((request-fn (if async #'spiral-aux-send #'spiral-aux-sync-request))
(print-settings (spiral-project-print-settings project))
(context-print-settings (map-elt print-settings context)) ;; TODO: exposing AST structure here, avoid
(parse-value (lambda (key val)
(cl-case val
('max 'Long/MAX_VALUE)
('same (map-elt context-print-settings key))
(t val)))))
(funcall request-fn
(spiral-project-templated-action
project :print-settings
:unrepl.print/context context
:unrepl.print/coll-length (funcall parse-value :coll-length coll-length)
:unrepl.print/nesting-depth (funcall parse-value :nesting-depth nesting-depth)
:unrepl.print/string-length (funcall parse-value :string-length string-length)))))
(defmacro spiral-binding-print-limits (coll-length nesting-depth string-length &rest body)
"Edit UNREPL `:eval' print settings, exec BODY and revert limits back.
COLL-LENGTH, NESTING-DEPTH, and STRING-LENGTH are UNREPL's update-able
settings."
(declare (indent 3))
`(with-current-project
(let ((ast-limits (lambda (bindings-node)
(mapcar
(lambda (key)
(parseclj-ast-value (spiral-ast-map-elt bindings-node key)))
'(:unrepl.print/coll-length
:unrepl.print/nesting-depth
:unrepl.print/string-length)))))
(let ((previous-limits (spiral-update-print-settings
project :eval
,coll-length ,nesting-depth ,string-length)))
(prog1 (progn ,@body)
(apply #'spiral-update-print-settings
project :eval
(funcall ast-limits previous-limits)))))))
(declare-function spiral-client-sync-request "spiral-loop")
(defun spiral--switch-ns (ns)
"Switch client connection's namespace to NS."
(spiral-client-sync-request (format "(unrepl/do (ns %s))" ns)))
(defun spiral-list-ns ()
"Synchronously list namespaces loaded in `:client' connection."
;; TODO: Make this into a session action, list only loaded namespaces and put
;; current one first, if any.
(spiral-binding-print-limits 'max 'same 'same
(thread-first "(all-ns)"
(spiral-aux-sync-request)
(spiral-ast-to-elisp `((unrepl/ns . ,#'parseclj-ast-value))))))
;; Evaluation
;; -------------------------------------------------------------------
(declare-function spiral-client-send "spiral-loop")
(declare-function spiral-repl-notify "spiral-repl")
(defun spiral-eval (form eval-callback &optional stdout-callback)
"Send FORM to UNREPL Socket Server for evaluation.
FORM can either be a string or a list tuple of buffer start, end positions.
This function sends everything through the `:client' connection, and
dispatches the evaluation payload (as an AST node) to EVAL-CALLBACK, which
can expect it as its only argument. STDOUT-CALLBACK is also a function
that expects just one argument, any STDOUT belonging to this evaluation.
Depending on the value of `spiral-automatic-ns-sync', this function
will make sure that the current file's namespace is in sync with the
REPL's/project's namespace. For more info, see
`spiral-automatic-ns-sync' and `spiral-aux-switch-ns'."
(when (listp form)
(remove-overlays (car form) (cadr form) 'temporary t))
;; Check for namespace synchronization
(when-let (file-ns-str (clojure-find-ns))
(let ((repl-ns (thread-first spiral-conn-id
(spiral-projects-get)
(spiral-project-namespace)))
(file-ns (intern file-ns-str)))
(unless (eql repl-ns file-ns)
(cl-case spiral-automatic-ns-sync
('do-it-and-notify (progn
(spiral--switch-ns file-ns)
(spiral-repl-notify
'info
(concat "Switched to namespace `"
(propertize (symbol-name file-ns)
'font-lock-face 'font-lock-keyword-face)
"'")
(concat "SPIRAL automatically switches to your file's namespace when doing\n"
"interactive evaluations.")
(concat "This behavior can be customize by changing the\n"
" `spiral-automatic-ns-sync' custom variable.\n\n"
"Invoke:\n\n"
"- `M-x customize-variable [RET] spiral-automatic-ns-sync [RET]'\n\n"
"to see all the options.\n\n"
"If you want to learn more about namespaces, check out:\n"
"- https://clojure.org/reference/namespaces\n"
"- https://www.braveclojure.com/organization/"))))
('do-it-without-notify (spiral--switch-ns file-ns))
('avoid-and-notify (spiral-repl-notify
'warn
(concat "REPL namespace differs from `"
(propertize (symbol-name file-ns)
'font-lock-face 'font-lock-keyword-face)
"'")
"Beware that some of your file's definitions may not be found."
(concat "SPIRAL can automatically sync your namespace to whichever buffer you are\n"
"evaluating. Please check the customization option\n"
"`spiral-automatic-ns-sync' by issuing:\n\n"
"- `M-x customize-variable [RET] spiral-automatic-ns-sync [RET]'\n\n"
"And if you want SPIRAL to help you with your namespaces synchronization,\n"
"select: 'Do sync and notify in the REPL buffer'.\n\n"
"If you want to learn more about namespaces, check out:\n"
"- https://clojure.org/reference/namespaces\n"
"- https://www.braveclojure.com/organization/")))
('avoid-without-notify 'noop)
(t (when spiral-debug
(error "Unrecognized `spiral-automatic-ns-sync' value: %s"
spiral-automatic-ns-sync)))))))
;; Evaluate the form
(let ((form (if (consp form)
(apply #'buffer-substring-no-properties form)
form)))
(spiral-client-send form eval-callback stdout-callback (current-buffer))))
(defun spiral-mode--interactive-eval-display-callback (eval-payload &optional bounds)
"Display evaluation result EVAL-PAYLOAD as a string.
This function will put a string version of EVAL-PAYLOAD in the echo area,
font-locked as Clojure.
If BOUNDS is non-nil and `spiral-eval-result-display' is something else
than 'echo, VALUE will also be displayed in an overlay starting at the end
bound."
(let ((value (spiral-ast-unparse-to-string eval-payload))
(point (cadr bounds)))
(when (and point (not (eql spiral-eval-result-display 'echo)))
(spiral--make-result-overlay value point))
(message "%s%s" spiral-eval-result-prefix value)))
(defun spiral-mode--interactive-eval-replace-callback (eval-payload bounds)
"Replace whatever it is in BOUNDS with the evaluation result EVAL-PAYLOAD.
This function will delete whatever it is between BOUNDS in BUFFER, and
replace it with a string version of EVAL-PAYLOAD."
(with-current-buffer (marker-buffer (car bounds))
(apply #'delete-region bounds)
(goto-char (car bounds))
(spiral-ast-unparse eval-payload)))
(defun spiral-eval-last-sexp (&optional prefix)
"Evaluate the expression preceding point.
If invoked with PREFIX, replace the evaluated for with its result in
current buffer."
(interactive "P")
(spiral-ensure-connected!)
(let ((bounds (spiral-last-sexp 'marker-bounds))
(callback (if prefix
#'spiral-mode--interactive-eval-replace-callback
#'spiral-mode--interactive-eval-display-callback)))
(spiral-eval bounds
(lambda (eval-payload)
(funcall callback eval-payload bounds)))))
(defun spiral-eval-top-level-form ()
"Evaluate the \"top-level\" form containing point."
(interactive)
(spiral-ensure-connected!)
(let ((bounds (spiral-top-level-form-at-point 'marker-bounds)))
(spiral-eval bounds
(lambda (eval-payload)
(spiral-mode--interactive-eval-display-callback eval-payload bounds)))))
;; Interactive Commands
;; -------------------------------------------------------------------
(defun spiral-switch-to-repl-buffer ()
"Switch to the REPL buffer for `spiral-conn-id'."
(interactive)
(with-current-project
(let ((repl-buffer (spiral-project-repl-buffer project)))
(if spiral-display-repl-in-current-window
(pop-to-buffer-same-window repl-buffer t)
(pop-to-buffer repl-buffer nil t)))))
(defun spiral-switch-ns (&optional prefix)
"Switch REPL namespace.
This command's default behavior is to switch to the current buffer's
namespace. When there's no namespace in the current buffer, or PREFIX is
non-nil, this command will prompt for a namespace to switch to, providing a
completion list."
(interactive "P")
(let ((file-ns (clojure-find-ns)))
(spiral--switch-ns
(if (or prefix
(null file-ns))
(completing-read "Namespace: " (spiral-list-ns) nil 'confirm)
file-ns))))
(declare-function spiral-repl-insert-phantom-input "spiral-repl")
(defun spiral-inspect-last-eval ()
"Replicate last evaluation in REPL buffer for further inspection."
(interactive)
(spiral-repl-insert-phantom-input spiral-latest-eval nil 'switch))
(declare-function spiral-aux-send "spiral-loop")
(defun spiral-eval-interrupt ()
"Interrupt pending evaluation."
(interactive)
(with-current-project
(let ((conn-id (spiral-project-id project))
(interrupt (lambda (pe)
(let* ((actions (spiral-pending-eval-entry-actions pe))
(interrupt-templ (spiral-ast-map-elt actions :interrupt)))
(spiral-aux-send (spiral-command-template interrupt-templ)
(lambda (_) (message "Evaluation interrupted!")))))))
(if-let (pending-eval (or (spiral-pending-eval :client conn-id)
(spiral-pending-eval :aux conn-id)))
(funcall interrupt pending-eval)
(message "No evaluations pending...")))))
(defun spiral-eval-buffer (&optional buffer)
"Eval BUFFER's file in UNREPL.
If no buffer is provided the command acts on the current buffer."
(interactive)
(check-parens)
(with-current-project
(with-current-buffer (or buffer (current-buffer))
(unless buffer-file-name
(user-error "Buffer `%s' is not associated with a file" (current-buffer)))
(when (and (buffer-modified-p)
(y-or-n-p (format "Save file %s? " buffer-file-name)))
(save-buffer))
(remove-overlays nil nil 'temporary t)
(let ((filename (buffer-file-name buffer))
;; (ns-form (cider-ns-form))
)
(spiral-aux-send
(spiral-project-templated-action project :spiral/load-file
:spiral/file (spiral-file-string filename)
:spiral/file-name (funcall spiral-filename-function filename)
:spiral/file-path (file-name-nondirectory filename))
(lambda (payload)
(message "%s" (spiral-ast-unparse-to-string payload 'mute-ui))))
(message "Loading %s..." filename)))))
(defun spiral-quit (&optional just-do-it conn-id)
"Quit connection to CONN-ID or current `spiral-conn-id'.
If JUST-DO-IT is non-nil, don't ask for confirmation."
(interactive "P")
(let ((conn-id (or conn-id
spiral-conn-id)))
(if-let (project (spiral-projects-get conn-id))
(when (or just-do-it
(y-or-n-p (format "Are you sure you want to quit connection to %s? " conn-id)))
(spiral-disconnect conn-id)
(message "UNREPL connection to %s terminated" conn-id))
(error "Connection %s could not be found" conn-id))))
;; Completion
;; -------------------------------------------------------------------
(defun spiral-complete--symbol-start-pos ()
"Find the starting position of the symbol at point, unless inside a string.
BORROWED FROM CIDER."
(let ((sap (symbol-at-point)))
(when (and sap (not (nth 3 (syntax-ppss))))
(car (bounds-of-thing-at-point 'symbol)))))
(defun spiral-complete--get-context-at-point ()
"Extract the context at point.
If point is not inside the list, returns nil; otherwise return \"top-level\"
form, with symbol at point replaced by __prefix__.
BORROWED FROM CIDER."
(when (save-excursion
(condition-case _
(progn
(up-list)
(check-parens)
t)
(scan-error nil)
(user-error nil)))
(save-excursion
(let* ((pref-end (point))
(pref-start (spiral-complete--symbol-start-pos))
(context (spiral-top-level-form-at-point))
(expr-start (progn
(beginning-of-defun)
(point))))
(concat (when pref-start (substring context 0 (- pref-start expr-start)))
"__prefix__"
(substring context (- pref-end expr-start)))))))
(defun spiral-complete--get-context ()
"Extract compliment's context."
(let ((context (when (derived-mode-p 'clojure-mode)
;; Important because `beginning-of-defun' and
;; `ending-of-defun' work incorrectly in the REPL
;; buffer, so context extraction fails there.
(spiral-complete--get-context-at-point))))
(if (string= spiral-completion-last-context context)
:same
(setq spiral-completion-last-context context)
context)))
(defun spiral-complete--candidates (str &optional ns)
"Find completion candidates for STR.
NS is an optional namespace symbol."
(when-let (project (and spiral-conn-id
(spiral-projects-get spiral-conn-id)))
(let* ((context (spiral-complete--get-context))
(candidates (spiral-aux-sync-request
(spiral-project-templated-action project :spiral/complete
:spiral/prefix str
:spiral/context context
:spiral/ns ns))))
(mapcar
(lambda (candidate-node)
(let* ((node-get (lambda (key) (thread-first candidate-node
(spiral-ast-map-elt key)
(parseclj-ast-value))))
(candidate (funcall node-get :candidate))
(type (funcall node-get :type))
(ns (funcall node-get :ns)))
(when candidate
(put-text-property 0 1 'type type candidate)
(put-text-property 0 1 'ns ns candidate))
candidate))
(parseclj-ast-children candidates)))))
(defun spiral-complete--annotate-symbol (symbol)
"Return a string suitable for annotating SYMBOL.
Takes properties `type' and `ns' from SYMBOL, if any, and concat them into
a single annotation. `ns' is only used when the symbol to be completed is
not fully qualified."
(let ((type (get-text-property 0 'type symbol))
(ns (unless (spiral-namespace-qualified-p symbol)
(get-text-property 0 'ns symbol))))
(concat (when ns (format " (%s) " ns))
(when type (format " <%s> " (spiral-keyword-name type))))))
(defun spiral-complete-at-point ()
"Complete the symbol at point.
Used as a `completion-at-point-functions' function.
BORROWED FROM CIDER."
(when (not (or (spiral-in-string-p) (spiral-in-comment-p)))
(when-let (bounds (bounds-of-thing-at-point 'symbol))
(list (car bounds) (cdr bounds)
(completion-table-dynamic #'spiral-complete--candidates)
:annotation-function #'spiral-complete--annotate-symbol))))
;; Setup
;; -------------------------------------------------------------------
(defconst spiral-mode-map
(let ((map (make-sparse-keymap)))
(define-key map (kbd "C-TAB") #'complete-symbol)
(define-key map (kbd "C-c C-z") #'spiral-switch-to-repl-buffer)
(define-key map (kbd "C-c M-n") #'spiral-switch-ns)
(define-key map (kbd "C-x C-e") #'spiral-eval-last-sexp)
(define-key map (kbd "C-c C-c") #'spiral-eval-top-level-form)
(define-key map (kbd "C-c C-r") #'spiral-inspect-last-eval)
(define-key map (kbd "C-c C-b") #'spiral-eval-buffer)
(define-key map (kbd "C-c C-g") #'spiral-eval-interrupt)
(define-key map (kbd "C-c q") #'spiral-quit)
(define-key map (kbd "C-c C-q") #'spiral-quit)
map))
(define-minor-mode spiral-mode
"Minor mode for SPIRAL.
\\{spiral-mode-map\}"
nil
spiral-mode-line
spiral-mode-map
(if spiral-mode
(progn
(make-local-variable 'completion-at-point-functions)
(add-to-list 'completion-at-point-functions
#'spiral-complete-at-point))
(mapc #'kill-local-variable '(spiral-conn-id
completion-at-point-functions))))
(provide 'spiral-mode)
;;; spiral-mode.el ends here