-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path3d-viewer.lisp
359 lines (325 loc) · 11.8 KB
/
3d-viewer.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
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
;; 3d-viewer.lisp
;;
;; Copyright (c) 2022 Jeremiah LaRocco <[email protected]>
(in-package #:simple-gl)
(defclass 3d-viewer (viewer)
((pos
:initform (vec3 0 0 -5)
:initarg :pos
:type vec3
:documentation "Camera position.")
(quat
:initform (vec4 0 0 0 1.0)
:initarg :pos
:type vec4
:documentation "Orientation quaternion (x,y,z,w) w:real xyz: imaginary")
(rotation-enabled
:initform t
:type (or t nil)
:initarg :rotation-enabled
:documentation "nil if view rotation is disabled. non-nil if view rotation is enabled.")
(zoom-enabled
:initform t
:type (or t nil)
:initarg :zoom-enabled
:documentation "nil if zooming is disabled. non-nil if zooming is enabled.")
(field-of-view
:initform 80.0
:initarg :field-of-view
:type real
:documentation "The view field of view for perspective viewing.")
(near
:initform 0.1
:initarg :near
:type real
:documentation "The view's near distance.")
(far
:initform 100.0
:initarg :far
:type real
:documentation "The view's far distance.")
)
(:documentation "A viewer with keyboard and 3d mouse camera rotation around a target point."))
(defun mat4-quat (quat)
(declare (optimize (speed 3) (space 0) (safety 0) (debug 0))
(type vec4 quat))
(let* ((xsq2 (* 2 (vx quat) (vx quat)))
(ysq2 (* 2 (vy quat) (vy quat)))
(zsq2 (* 2 (vz quat) (vz quat)))
(sx (- 1 ysq2 zsq2))
(sy (- 1 xsq2 zsq2))
(sz (- 1 xsq2 ysq2)))
(declare (type real xsq2 ysq2 zsq2 sx sy sz))
(with-vec4 (x y z w) quat
(mat4 `(,sx ;; 0
,(* 2 (+ (* x y) (* w z))) ;; 1
,(* 2 (+ (* z x) (* -1 w y))) ;; 2
0 ;; 3
,(* 2 (+ (* x y) (* -1 w z))) ;; 4
,sy ;; 5
,(* 2 (+ (* y z) (* w x))) ;; 6
0 ;;7
,(* 2 (+ (* z x) (* w y))) ;; 8
,(* 2 (+ (* y z) (* -1 w x))) ;; 9
,sz ;; 10
0 ;; 11
0 ;; 12
0 ;; 13
0 ;; 14
1.0) ;; 15
))))
(defmethod view-matrix ((viewer 3d-viewer))
(with-slots (field-of-view aspect-ratio near far quat pos) viewer
(m*
(mperspective field-of-view aspect-ratio near far)
(3d-matrices:mtranslation pos)
(mat4-quat quat))))
(defmethod reset-view-safe ((viewer 3d-viewer))
(with-slots (view-changed objects pos quat) viewer
(setf pos (vec3 0 0 -5))
(setf quat (vec4 0 0 0.0 1))
(loop
:with view-xform :of-type mat4 = (view-matrix viewer)
:for (nil . object) :in objects
:do
(ensure-initialized object)
(set-uniform object "view_transform" view-xform :mat4))
(setf view-changed t)))
(defmethod handle-key ((viewer 3d-viewer) window key scancode action mod-keys)
(declare (ignorable window scancode mod-keys)
(optimize (speed 3) (safety 0) (debug 0) (space 0)))
(with-viewer-lock (viewer)
(with-slots (aspect-ratio rotation-enabled zoom-enabled pos quat view-changed) viewer
(let* (
(x 0)
(y 0)
(z 0)
(rx 0)
(ry 0)
(rz 0))
(cond
((and (eq key :r) (eq action :press))
(reset-view-safe viewer)
(setf view-changed t)
t)
((and (eq key :x)
(find action '(:repeat :press)))
(incf rx)
(setf view-changed t)
t)
((and (eq key :x)
(find :shift mod-keys)
(find action '(:repeat :press)))
(decf rx)
(setf view-changed t)
t)
((and (eq key :y) (find action '(:repeat :press)))
(incf ry)
(setf view-changed t)
t)
((and (eq key :y)
(find :shift mod-keys)
(find action '(:repeat :press)))
(decf ry)
(setf view-changed t)
t)
((and (eq key :z) (find action '(:repeat :press)))
(incf rz)
(setf view-changed t)
t)
((and (eq key :z) (find action '(:repeat :press)))
(incf rz)
(setf view-changed t)
t)
((and (eq key :x)
(find :control mod-keys)
(find action '(:repeat :press)))
(incf x)
(setf view-changed t)
t)
((and (eq key :x)
(find :shift mod-keys)
(find :control mod-keys)
(find action '(:repeat :press)))
(decf x)
(setf view-changed t)
t)
((and (eq key :y)
(find :control mod-keys)
(find action '(:repeat :press)))
(incf y)
(setf view-changed t)
t)
((and (eq key :y)
(find :shift mod-keys)
(find :control mod-keys)
(find action '(:repeat :press)))
(decf y)
(setf view-changed t)
t)
((and (eq key :z)
(find :control mod-keys)
(find action '(:repeat :press)))
(incf z)
(setf view-changed t)
t)
((and (eq key :z)
(find :shift mod-keys)
(find :control mod-keys)
(find action '(:repeat :press)))
(decf z)
(setf view-changed t)
t)
(t nil))
(when view-changed
(let ((len (sqrt
(+ (* rx rx)
(* ry ry)
(* rz rz)))))
(when (and rotation-enabled
(not (zerop len)))
(setf quat (quaternion-rotate quat
(* len 0.01)
(vec3 (/ rx len -1)
(/ ry len -1)
(/ rz len 1)))))
(when zoom-enabled
(setf pos (v+ pos
(vec3-qrot (vec3 (- x)
(- y)
z)
quat)))))
t))))
;; TODO: Figure out a way to avoid locking the mutex a second time here,
;; preferably without using a recursive lock.
(call-next-method))
(defun quaternion-rotate (quaternion angle xyz)
(declare (optimize (speed 3) (safety 0) (debug 0) (space 0))
(type #.3d-vectors::*float-type* angle)
(type vec3 xyz))
(let ((half (* 0.5 angle))
(sin-half (sin (* 0.5 angle))))
(quat-mul quaternion
(vec4 (* (vx xyz) sin-half)
(* (vy xyz) sin-half)
(* (vz xyz) sin-half)
(cos half))
)))
(defun quat-mul (a b)
(declare (optimize (speed 3) (safety 0) (debug 0) (space 0))
(type vec4 a b))
(declare (optimize (speed 3) (safety 0) (debug 0) (space 0)))
(let ((dot (v. (vxyz a) (vxyz b)))
(cross (vc (vxyz a) (vxyz b))))
(declare (type #.3d-vectors::*float-type* dot)
(type vec3 cross))
(with-vec4 (ax ay az aw) a
(with-vec4 (bx by bz bw) b
(with-vec3 (cx cy cz) cross
(vec4 (+ (* aw bx)
(* bw ax)
cx)
(+ (* aw by)
(* bw ay)
cy)
(+ (* aw bz)
(* bw az)
cz)
(- (* aw bw)
dot)))))))
(defun quat-invert (quat)
(declare (optimize (speed 3) (safety 0) (debug 0) (space 0))
(type vec4 quat))
(let* ((len-sq (vsqrlength quat))
(scale (if (> len-sq 0.01)
(/ 1.0 len-sq)
1.0)))
(vec4 (* scale (- (vx quat)))
(* scale (- (vy quat)))
(* scale (- (vz quat)))
(* scale (vw quat)))))
(defun vec3-qrot (vec quat)
(let ((inv-q (quat-invert quat))
(vq (vxyz_ vec)))
(vxyz (quat-mul
(quat-mul vq quat)
inv-q))))
#+spacenav
(defmethod handle-3d-mouse-event ((viewer 3d-viewer) (event sn:motion-event))
(with-viewer-lock (viewer)
(with-slots (pos
quat
zoom-enabled
rotation-enabled
view-changed) viewer
(setf view-changed t)
(with-slots (sn:x sn:y sn:z
sn:rx sn:ry sn:rz) event
(let ((len (sqrt (+ (* sn:rx sn:rx)
(* sn:ry sn:ry)
(* sn:rz sn:rz)))))
(when (and rotation-enabled
(not (zerop len)))
(setf quat (quaternion-rotate quat
(* len 0.03)
(vec3 (/ sn:rx len -1.0)
(/ sn:ry len -1.0)
(/ sn:rz len 1.0)))))
(when zoom-enabled
(setf pos (v+ pos (vec3-qrot (vec3 (* sn:x -0.08)
(* sn:y -0.08)
(* sn:z 0.08))
quat)))))))))
(defmethod handle-joystick ((viewer 3d-viewer))
(with-viewer-lock (viewer)
(with-slots (pos
quat
zoom-enabled
joysticks
rotation-enabled
view-changed) viewer
(when (null joysticks)
(return-from handle-joystick nil))
(let* ((joystick (etypecase joysticks
(list (first joysticks))
(fixnum joysticks)))
(buttons (make-array (length #1=(the list (glfw:get-joystick-buttons joystick))) :element-type 'fixnum :initial-contents #1#))
(axes (make-array (length #2= (the list (glfw:get-joystick-axes joystick))) :element-type 'single-float :initial-contents #2#)))
(declare (ignorable buttons))
;; (format t "buttons: ~a axes: ~a~%" buttons axes)
(setf view-changed t)
(let ((len (sqrt (+ (* (aref axes 0) (aref axes 0))
(* (aref axes 1) (aref axes 1))
(* (aref axes 2) (aref axes 2))))))
(declare (ignorable len))
(when zoom-enabled
(setf pos
(v+ pos
(vec3-qrot (vec3
(* (if (> (abs (aref axes 0)) 0.02)
(aref axes 0)
0.0)
0.2)
(* (if (> (abs (aref axes 2)) 0.02)
(aref axes 2)
0.0)
0.1)
(* (if (> (abs (aref axes 1)) 0.02)
(aref axes 1)
0.0)
-0.5))
quat)))))))))
#+spacenav
(defmethod handle-3d-mouse-event ((viewer 3d-viewer) (event sn:button-event))
(sgl:with-viewer-lock (viewer)
(with-slots (sgl:view-changed
rotation-enabled
window
sgl:objects) viewer
(cond
((sn:button-press-p event :joystick)
(format t ":joystick pressed!~%")
(setf rotation-enabled (not rotation-enabled))
t)
(t
(call-next-method))))))