-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpd-writer.lisp
145 lines (133 loc) · 4.87 KB
/
pd-writer.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
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
(defpackage :pd-writer
(:nicknames :pdx)
(:use :common-lisp
:pd-structs
:pd-ranking)
(:export :with-patch
:port
:connect
:color/live
:color/file))
(in-package :pd-writer)
;; --------------------------------------------------------------------------------
;; data
;;
(defparameter *patch* "")
(defparameter *nodes* nil)
(defparameter *connections* nil)
;; --------------------------------------------------------------------------------
;; pd
;;
(defun write-connection (c)
(let ((out (find (connection-out-id c) *nodes* :key #'node-id))
(in (find (connection-in-id c) *nodes* :key #'node-id)))
(cond ((or (null out)
(null in))
(error "wow, no node found!")))
(concatenate 'string
"#X connect "
(to-string (node-index out)) " "
(to-string (connection-out-port c)) " "
(to-string (node-index in)) " "
(to-string (connection-in-port c))
";"
(string #\newline))))
(defun write-patch (&key
(width 512)
(height 512)
graph-on-parent
view-width
view-height
hide-object-name)
(with-open-file (f *patch* :direction :output :if-exists :supersede)
(format f "#N canvas 0 0 ~d ~d 10;~%" width height) ; patch init
;; ... nodes
(mapcar (lambda (n)
(princ (write-node n) f))
(reverse (rank *nodes*
*connections*
view-height)))
;; ... connections
(mapcar (lambda (c)
(princ (write-connection c) f))
*connections*)
;; graph-on-parent
(when graph-on-parent
(format f "#X coords 0 -1 1 1 ~d ~d ~d 0 0;~%"
view-width
view-height
(if hide-object-name 2 1)))
t))
(defun add-connection (out-id out-port in-id in-port)
(let ((c (make-connection :out-id out-id
:out-port out-port
:in-id in-id
:in-port in-port)))
(push c *connections*)))
(defun add-node (n)
;; this could incorporate the duplication between node definitions
(setf (node-index n) (length *nodes*))
(push n *nodes*))
;; --------------------------------------------------------------------------------
;; user-facing
;;
(defmacro with-patch (patch-config &rest form)
`(progn
(setq *patch* ,(first patch-config))
(setq *nodes* nil)
(setq *connections* nil)
,@form
(write-patch ,@(rest patch-config))))
(defun port (number node) ; TODO rename to "outlet", flip argument order
(make-port :number number
:node node))
(defun connect (n &rest args)
(labels
((recur (n port args
&optional (inc-port-p t))
(if (not (null args))
(let ((arg (first args))
(next-args (rest args))
(next-port (if inc-port-p
(1+ port)
port)))
(cond
;; connect node
((node-p arg)
(add-connection (node-id arg) 0 ; default outlet
(node-id n) port) ; argument position
(recur n next-port next-args inc-port-p))
;; connect specific outlet of a node
((port-p arg)
(add-connection (node-id (port-node arg)) (port-number arg)
(node-id n) port)
(recur n next-port next-args inc-port-p))
;; all elements of a list are connected to the same inlet
((listp arg)
(recur n port arg nil)
(recur n next-port next-args inc-port-p))
;; literals are added as init-args
((not (null arg))
(setf (node-init-args n)
(concatenate 'string
(node-init-args n) " "
(to-string arg)))
(recur n port next-args inc-port-p))
;; an argument of `nil` explicitly increments the input port counter
((null arg)
(recur n next-port next-args inc-port-p))))
n)))
(recur n 0 args t)))
(defun color/live (r g b)
"used to assign colors at runtime, e.g. when adressing ui-nodes by pd messages"
(1- (+ (* r -65536)
(* g -256)
(* b -1))))
(defun color/file (r g b)
"used to assign colors at compile-time, e.g. when writing them into a patch file"
(destructuring-bind (r2 g2 b2)
(mapcar (lambda (x) (floor (/ x 4)))
(list r g b))
(1- (+ (* r2 -4096)
(* g2 -64)
(* b2 -1)))))