forked from ruricolist/serapeum
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathexporting.lisp
85 lines (68 loc) · 3.14 KB
/
exporting.lisp
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
(in-package :serapeum)
(uiop:define-package :serapeum.exporting
(:use)
(:export
:defclass :define-values
;; Prevent implicit exports from being unexported on redefinition.
. #.(and (find-package :serapeum.exporting)
(mapcar #'make-keyword
(serapeum:package-exports :serapeum.exporting)))))
(defmacro serapeum.exporting:defclass (name supers &body (slots . options))
"Like `defclass', but implicitly export the name of the class and
the names of all accessors (including readers and writers).
You can specify `:export nil' in the definition of a slot to prevent
its accessors from being exported."
(let ((slots (mapcar #'ensure-list slots))
accessors)
(dolist (slot slots)
(destructuring-bind (&key accessor reader writer (export t)
&allow-other-keys)
(rest slot)
(when export
(when accessor (push accessor accessors))
(when reader (push reader accessors))
(when writer (push writer accessors)))))
(setf accessors (remove-duplicates accessors))
`(progn
(export-always ',(cons name accessors))
(defclass ,name ,supers
,(mapcar (lambda (slot)
(cons (first slot)
(remove-from-plist (rest slot) :export)))
slots)
,@options))))
(defmacro serapeum.exporting:define-values (values &body (expr))
"Like `define-values', with implicit export of VALUES."
`(progn
(export-always ',values)
(define-values ,values ,expr)))
(defmacro define-simple-exporter (macro-name lambda-list)
(with-unique-names (whole)
(let ((name-sym (first lambda-list))
(exporter-name (intern (string macro-name)
(find-package :serapeum.exporting))))
`(progn
(export-always
(list (intern ,(string macro-name)
(find-package :serapeum.exporting)))
(find-package :serapeum.exporting))
(defmacro ,exporter-name (&whole ,whole ,@lambda-list)
,(fmt "Like `~(~a~)', with implicit export of ~:@(~a~)."
macro-name name-sym)
(declare (ignore ,@(set-difference (flatten (rest lambda-list))
lambda-list-keywords)))
(list 'progn
(list 'export-always (list 'quote ,name-sym))
(cons ',macro-name (rest ,whole))))))))
(define-simple-exporter def (var &body (&optional val docs)))
(define-simple-exporter define-symbol-macro (symbol expansion))
(define-simple-exporter deftype (name lamda-list &body body))
(define-simple-exporter defconst (symbol init &optional docstring))
(define-simple-exporter defconstant (name value &optional doc))
(define-simple-exporter defvar (var &optional val doc))
(define-simple-exporter defparameter (var val &optional doc))
(define-simple-exporter defun (name lambda-list &body body))
(define-simple-exporter defalias (name &body body))
(define-simple-exporter defmacro (name &body body))
(define-simple-exporter defgeneric (name lambda-list &body options))
(define-simple-exporter defmethod (name &body args))