-
-
Notifications
You must be signed in to change notification settings - Fork 32
/
Copy pathenvironment.lisp
234 lines (193 loc) · 9.77 KB
/
environment.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
(in-package #:org.shirakumo.fraf.kandria)
(defvar *environments* (make-hash-table :test 'eql))
(defclass audible-entity (listener entity)
((voice :initarg :voice :accessor voice)))
(defmethod stage :after ((entity audible-entity) (area staging-area))
(when (typep (voice entity) 'placeholder-resource)
(let ((generator (generator (voice entity))))
(setf (voice entity) (apply #'generate-resources generator (input* generator) :resource NIL (trial::generation-arguments generator)))))
(stage (voice entity) area))
(defmethod active-p ((entity audible-entity)) T)
(defmethod handle :after ((ev tick) (entity audible-entity))
(when (active-p entity)
(let* ((voice (voice entity))
(max (expt (mixed:max-distance voice) 2))
(dist (vsqrdistance (location entity) (location (node 'player +world+)))))
(cond ((< dist (+ max 32)) ; Start a bit before the entrance to ensure we have a smooth ramp up.
(harmony:play voice :location (location entity)))
((< (+ max 128) dist) ; Stop even farther out to ensure we don't accidentally flip-flop and overburden the sound server.
(harmony:stop voice))))))
(defclass switch-music-state (event)
((area :initarg :area :initform NIL :reader area)
(state :initarg :state :initform NIL :reader state)))
(defun (setf music-state) (state area)
(if (eql T area)
(dolist (area (delete-duplicates (mapcar #'area (list-environments))))
(issue +world+ 'switch-music-state :area area :state state))
(issue +world+ 'switch-music-state :area area :state state)))
(defclass environment () ()) ; Early def
(defclass environment-controller (entity listener)
((name :initform 'environment)
(environment :initarg :environment :initform NIL :accessor environment)
(area-states :initarg :area-states :initform NIL :accessor area-states)
(override :initarg :override :initform NIL :accessor override)))
(defmethod stage :after ((controller environment-controller) (area staging-area))
(when (environment controller) (stage (environment controller) area))
(when (override controller) (stage (override controller) area)))
(defmethod reset ((controller environment-controller))
(setf (area-states controller) ())
(when (override controller)
(harmony:transition (override controller) 0.0 :in 0.1)
(setf (slot-value controller 'override) NIL))
(let ((env (environment controller)))
(when env
(when (music env)
(harmony:transition (music env) 0.0 :in 0.1))
(when (ambience env)
(harmony:transition (ambience env) 0.0 :in 0.1))
(setf (environment controller) NIL))))
(defmethod handle ((ev switch-chunk) (controller environment-controller))
(switch-environment controller (environment (chunk ev))))
(defmethod handle ((ev switch-music-state) (controller environment-controller))
(let* ((env (environment controller))
(area (or (area ev) (area env)))
(state (state ev))
(cons (assoc area (area-states controller))))
(v:info :kandria.harmony "Switching music state of ~s to ~s." area state)
(if cons
(setf (cdr cons) state)
(push (cons area state) (area-states controller)))
(when (and env
(eql area (area env))
(null (override controller)))
(when (music env)
(harmony:transition (music env) state :in 5.0))
(when (ambience env)
(harmony:transition (ambience env) state :in 3.0 :error NIL)))))
(defmethod switch-environment ((controller environment-controller) (name symbol))
(cond (name
(switch-environment controller (environment name)))
(T
(switch-environment (environment controller) NIL)
(setf (environment controller) NIL))))
(defmethod switch-environment ((controller environment-controller) (environment environment))
(unless (override controller)
(when (allocated-p environment)
(switch-environment (environment controller) environment)))
(setf (environment controller) environment))
(defmethod (setf override) :before ((override null) (controller environment-controller))
(when (override controller)
(harmony:transition (override controller) 0.0 :in 5.0))
(switch-environment NIL (environment controller)))
(defmethod (setf override) :before ((override (eql 'null)) (controller environment-controller))
(when (override controller)
(harmony:transition (override controller) 0.0 :in 5.0))
(switch-environment (environment controller) NIL))
(defmethod (setf override) :before ((override resource) (controller environment-controller))
(when (allocated-p override)
(unless (eq override (override controller))
(if (override controller)
(harmony:transition (override controller) 0.0 :in 3.0)
(switch-environment (environment controller) NIL))
(harmony:transition override 1.0 :in 3.0))))
(defmethod harmony:transition ((controller environment-controller) (to real) &key (in 1.0))
(cond ((override controller)
(harmony:transition (override controller) to :in in))
((environment controller)
(harmony:transition (environment controller) to :in in))))
(defmethod handle ((event load-complete) (controller environment-controller))
(let ((override (override controller))
(environment (environment controller)))
(setf (slot-value controller 'environment) NIL)
(setf (slot-value controller 'override) NIL)
(setf (override controller) override)
(switch-environment controller environment)))
(defmethod harmony:transition ((nothing (eql 'null)) to &key in)
(declare (ignore nothing to in)))
(defmethod harmony:transition ((thing placeholder-resource) to &rest args &key &allow-other-keys)
(trial:commit thing (loader +main+) :unload NIL)
(apply #'harmony:transition thing to args))
(defun override-music (track)
(setf (override (node 'environment +world+)) (when track (// 'music track))))
(defclass environment ()
((name :initarg :name :initform NIL :accessor name)
(music :initarg :music :initform NIL :accessor music)
(ambience :initarg :ambience :initform NIL :accessor ambience)
(area :initarg :area :initform NIL :accessor area)))
(defmethod print-object ((environment environment) stream)
(print-unreadable-object (environment stream :type T)
(format stream "~s" (name environment))))
(defmethod allocated-p ((environment environment))
(let ((music (music environment))
(ambience (ambience environment)))
(and (or (null ambience) (allocated-p ambience))
(or (null music) (allocated-p music)))))
(defmethod stage ((environment environment) (area staging-area))
(when (music environment) (stage (music environment) area))
(when (ambience environment) (stage (ambience environment) area)))
(defmethod state ((environment environment))
(let* ((controller (node 'environment +world+))
(cons (assoc (area environment) (area-states controller))))
(if cons
(cdr cons)
:normal)))
(defmethod (setf state) (state (environment environment))
(issue +world+ 'switch-area-state :area (area environment) :state state))
(defmethod quest:activate ((environment environment))
(let ((controller (node 'environment +world+)))
(when (not (eq environment (environment controller)))
(switch-environment (environment controller) environment))))
(defmethod switch-environment ((a null) (b null)))
(defmethod switch-environment ((none null) (environment environment))
(let ((state (state environment)))
(when (music environment)
(harmony:transition (music environment) state :in 5.0 :error NIL))
(when (ambience environment)
(harmony:transition (ambience environment) state :in 3.0 :error NIL))))
(defmethod switch-environment ((environment environment) (none null))
(when (music environment)
(harmony:transition (music environment) NIL :in 3.0))
(when (ambience environment)
(harmony:transition (ambience environment) NIL :in 5.0)))
(defmethod switch-environment ((from environment) (to environment))
(let ((state (state to)))
(flet ((tx (a b in &optional (state state))
(unless (eq a b)
(when a
(harmony:transition a NIL :in in)))
(when b
(harmony:transition b state :in in :error NIL))))
(tx (music from) (music to) 5.0)
(tx (ambience from) (ambience to) 3.0 :normal))))
(defmethod harmony:transition ((environment environment) (to real) &key (in 1.0))
(when (music environment)
(harmony:transition (music environment) to :in in))
(when (ambience environment)
(harmony:transition (ambience environment) to :in in)))
(defmethod environment ((name symbol))
(gethash name *environments*))
(defmethod (setf environment) ((value environment) (name symbol))
(setf (gethash name *environments*) value))
(defun remove-environment (name)
(remhash name *environments*))
(defun list-environments ()
(loop for env being the hash-values of *environments*
collect env))
(defmacro define-environment ((area name) &body initargs)
(let ((music (getf initargs :music))
(ambience (getf initargs :ambience))
(initargs (remf* initargs :music :ambience))
(name (trial::mksym *package* area '/ name)))
`(setf (environment ',name)
(ensure-instance (environment ',name) 'environment
:area ',area
:name ',name
:music ,(when music `(// 'music ,music))
:ambience ,(when ambience `(// 'music ,ambience))
,@initargs))))
(defmethod harmony:transition :before ((voice harmony::voice) (to real) &key)
(v:info :kandria.harmony "Transitioning ~a to ~a" voice to))
(trial-alloy:define-set-representation environment/combo
:represents environment
:item-text (string (when alloy:value (name alloy:value)))
(list-environments))