commit | author | age
|
c38e61
|
1 |
|
71cec5
|
2 |
;====================================== |
c38e61
|
3 |
;OM-Lead, 2022-2024 |
71cec5
|
4 |
; |
MS |
5 |
;Library for Rule-based Voice-Leading |
|
6 |
;Author: Lorenz Lehmann |
|
7 |
;Supervision: Marlon Schumacher |
|
8 |
;====================================== |
|
9 |
|
c38e61
|
10 |
|
71cec5
|
11 |
(in-package :omlead) |
MS |
12 |
|
|
13 |
|
|
14 |
|
c38e61
|
15 |
;;;;;;GEN-HARMONIES;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; |
71cec5
|
16 |
|
c38e61
|
17 |
(om::defmethod! gen-harmonies (pitch-sets trans-scale ranges expand-harmonies filter-harmonies) |
LL |
18 |
:icon 030719978 |
|
19 |
|
|
20 |
;++++++++++++++++++++++ transposition ++++++++++++++++++++++++++++++++++++++++++++++++ |
|
21 |
(let* ((scale |
|
22 |
|
|
23 |
(cond |
|
24 |
|
|
25 |
((numberp (car trans-scale )) |
|
26 |
trans-scale) |
|
27 |
|
|
28 |
|
|
29 |
(t (let ((fundamental (second trans-scale)) |
|
30 |
(negativ (third trans-scale)) |
|
31 |
(positiv (fourth trans-scale)) |
|
32 |
(mode (fifth trans-scale)) |
|
33 |
(range-min (list-min_oml (flat_oml ranges))) |
|
34 |
(range-max (list-max_oml (flat_oml ranges)))) |
|
35 |
|
|
36 |
(append |
|
37 |
|
|
38 |
|
|
39 |
;;;;;;;;;;;;;;;;;;negativ |
|
40 |
(reverse (loop for i in (if (or (eq 'hold mode) (eq 'hold-mirror mode)) |
|
41 |
(hold_oml negativ) (circular_oml negativ)) ;;endless down-list |
|
42 |
sum i into total |
|
43 |
until (> range-min (- fundamental total)) |
|
44 |
collect (- fundamental total) |
|
45 |
)) |
|
46 |
|
|
47 |
(list fundamental) |
|
48 |
|
|
49 |
;;;;;;;;;;;;;;;;;;positiv |
|
50 |
|
|
51 |
(loop for i in (if (or (eq 'hold mode) (eq 'hold-mirror mode)) |
|
52 |
(hold_oml positiv) (circular_oml positiv)) ;;endless up-list |
|
53 |
sum i into total |
|
54 |
until (< range-max (+ fundamental total)) |
|
55 |
collect (+ fundamental total))))))) |
71cec5
|
56 |
|
c38e61
|
57 |
(harmonies (loop for pitch-set in pitch-sets |
LL |
58 |
append (loop for pitch in scale |
|
59 |
collect (mapcar (lambda (x) (+ x pitch)) pitch-set)))) |
71cec5
|
60 |
|
c38e61
|
61 |
;++++++++++++++++ adapt-collection-to-voice-ranges ++++++++++++++++++++++++++++++++++++++++++++++++++++ |
71cec5
|
62 |
|
MS |
63 |
|
c38e61
|
64 |
(harmonies (remove nil (loop for harmony in (remove-duplicates harmonies :test #'equal) |
LL |
65 |
collect (if (find -1 (mapcar (lambda (pitch range) |
|
66 |
(if (and (<= pitch (list-max_oml range)) (>= pitch (list-min_oml range))) pitch -1)) |
|
67 |
harmony ranges)) nil harmony)))) |
2b28da
|
68 |
|
c38e61
|
69 |
|
LL |
70 |
;++++++++++++++++ expand-harmonies ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ |
|
71 |
|
|
72 |
(expand-harmonies (if (atom (caar expand-harmonies)) (list expand-harmonies) expand-harmonies)) |
|
73 |
|
|
74 |
(harmonies |
|
75 |
(if (car expand-harmonies) |
|
76 |
(let ((harmony-list (list harmonies))) |
|
77 |
|
|
78 |
|
|
79 |
(loop for expand-box in expand-harmonies |
|
80 |
do (loop for rule in (second expand-box) |
|
81 |
do (loop for harmony in (car harmony-list) |
|
82 |
if (tester_oml |
|
83 |
(lambda (x) (and (<= (list-min_oml (third expand-box)) x) (>= (list-max_oml (third expand-box)) x))) |
|
84 |
harmony) |
|
85 |
append |
|
86 |
|
|
87 |
(mapcar (lambda (y) |
|
88 |
(if (tester_oml (lambda (x) (and (<= (list-min_oml (flat_oml ranges)) x) (>= (list-max_oml (flat_oml ranges)) x))) y) |
|
89 |
y nil)) |
|
90 |
(funcall rule (first expand-box) harmony)) into temp-harmonies |
|
91 |
else append harmony into temp-harmonies |
|
92 |
finally (fill harmony-list (remove nil temp-harmonies))))) |
|
93 |
(car harmony-list)) |
|
94 |
harmonies)) |
|
95 |
|
|
96 |
|
|
97 |
;++++++++++++++++ filter-harmonies ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ |
|
98 |
|
|
99 |
(filter-harmonies (if (atom (caar filter-harmonies)) (list filter-harmonies) filter-harmonies)) |
|
100 |
|
|
101 |
(harmonies |
|
102 |
(if (car filter-harmonies) |
|
103 |
(let ((filter-list harmonies)) |
|
104 |
|
|
105 |
|
|
106 |
(loop for filter-box in filter-harmonies |
|
107 |
do (loop for rule in (second filter-box) |
|
108 |
|
|
109 |
do (loop for harmony in filter-list |
|
110 |
do (if (tester_oml |
|
111 |
(lambda (x) (and (<= (list-min_oml (third filter-box)) x) (>= (list-max_oml (third filter-box)) x))) |
|
112 |
harmony) |
|
113 |
(if (not (funcall rule (first filter-box) harmony)) |
|
114 |
(delete harmony filter-list)))))) |
|
115 |
filter-list) |
|
116 |
harmonies)) |
|
117 |
|
|
118 |
;++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ |
|
119 |
|
|
120 |
) |
|
121 |
|
|
122 |
|
|
123 |
harmonies)) |
|
124 |
|
|
125 |
|
|
126 |
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; |
2b28da
|
127 |
|
LL |
128 |
|
|
129 |
|
71cec5
|
130 |
|
MS |
131 |
|
|
132 |
|
c38e61
|
133 |
;%%%%%%%%%%%%%%%TRANS-SCALE%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
LL |
134 |
|
|
135 |
|
|
136 |
(om::defmethod! trans-scale (fundamental scale (mode symbol)) |
|
137 |
:icon 0307199714 |
|
138 |
:initvals '(nil nil 'absolute) |
|
139 |
:menuins '((2 (("absolute" 'absolute) ("circular" 'circular) ("hold" 'hold) |
|
140 |
("circular-mirror" 'circular-mirror) ("hold-mirror" 'hold-mirror)))) |
|
141 |
|
|
142 |
(cond |
|
143 |
((null scale) |
|
144 |
(list fundamental)) |
|
145 |
|
|
146 |
((eq 'absolute mode) |
|
147 |
(append (list fundamental) (loop for i in scale |
|
148 |
sum i into total |
|
149 |
collect (+ fundamental total)))) |
|
150 |
|
|
151 |
((eq 'circular mode) |
|
152 |
(list nil fundamental (reverse scale) scale mode)) |
|
153 |
|
|
154 |
((eq 'hold mode) |
|
155 |
(list nil fundamental (reverse scale) scale mode)) |
|
156 |
|
|
157 |
((eq 'circular-mirror mode) |
|
158 |
(list nil fundamental (reverse (reverse scale)) scale mode)) |
|
159 |
|
|
160 |
((eq 'hold-mirror mode) |
|
161 |
(list nil fundamental (reverse (reverse scale)) scale mode)))) |
|
162 |
|
|
163 |
|
|
164 |
;#LL: double reverse in is the only way to avoid an endless loop in the main-function |
|
165 |
|
|
166 |
|
|
167 |
;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
|
168 |
|
|
169 |
|
|
170 |
|
|
171 |
|
|
172 |
;--------------------------RULES------------------------------------------------------------------------------------------------------------------------------- |
|
173 |
|
|
174 |
|
|
175 |
(om::defmethod! filter-doubles (ids harmony) |
|
176 |
:icon 030719977 |
|
177 |
(if (= (length (remove-duplicates harmony)) (length harmony)) t) |
|
178 |
) |
|
179 |
|
|
180 |
(om::defmethod! permutations (ids harmony) |
|
181 |
:icon 030719975 |
|
182 |
(let ((ids (list! ids))) |
|
183 |
(mapcar (lambda (x) (subs-posn harmony ids x)) |
|
184 |
(permutations_oml (mapcar (lambda (x) (nth x harmony)) ids)) |
|
185 |
))) |
|
186 |
|
|
187 |
(om::defmethod! registrations (ids harmony register-list) |
|
188 |
:icon 030719975 |
|
189 |
|
|
190 |
(let ((voices (mapcar (lambda (x) (nth x harmony)) ids))) |
|
191 |
|
|
192 |
(mapcar (lambda (x) (subs-posn harmony ids x)) |
|
193 |
|
|
194 |
(loop for registration in (variations_oml register-list (length ids)) |
|
195 |
collect (mapcar (lambda (x y) (+ x y)) voices registration))))) |
|
196 |
|
|
197 |
|
|
198 |
;------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
|
199 |
|
|
200 |
|
|
201 |
|
|
202 |
|
|
203 |
|
|
204 |
|
|
205 |
;:::::::::::::::::::::::::::APPLY-BOXES::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: |
|
206 |
|
|
207 |
(om::defmethod! filter-harmonies (ids rules rule-range) |
|
208 |
:icon 030719976 |
|
209 |
(list (list! ids) (list! rules) rule-range) |
|
210 |
) |
|
211 |
|
|
212 |
(om::defmethod! expand-harmonies (ids rules rule-range) |
|
213 |
:icon 030719974 |
|
214 |
(list (list! ids) (list! rules) rule-range) |
|
215 |
) |
|
216 |
|
|
217 |
;::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: |
|
218 |
|
|
219 |
|
|
220 |
|
|
221 |
|
|
222 |
|
|
223 |
|
|
224 |
|
|
225 |
;;;;;;;;HELPFUNCTIONS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; |
|
226 |
|
|
227 |
(defun list-max_oml (list) |
|
228 |
(reduce #'max list)) |
|
229 |
|
|
230 |
|
|
231 |
|
|
232 |
(defun list-min_oml (list) |
|
233 |
(reduce #'min list)) |
|
234 |
|
|
235 |
|
|
236 |
|
|
237 |
|
|
238 |
(defun sum_oml (liste &optional (start 0)) |
|
239 |
(cond ((null liste) start) |
|
240 |
(t (sum_oml (cdr liste) (+ start (car liste)))))) |
|
241 |
|
|
242 |
|
|
243 |
|
|
244 |
|
71cec5
|
245 |
|
a53ccf
|
246 |
(defun circular_oml (items) |
LL |
247 |
(setf (cdr (last items)) items) |
|
248 |
items) |
|
249 |
|
c38e61
|
250 |
|
LL |
251 |
|
|
252 |
|
a53ccf
|
253 |
|
LL |
254 |
(defun flat_oml (liste) |
|
255 |
(cond |
|
256 |
((null (cdr liste)) |
|
257 |
(if (atom (car liste)) |
|
258 |
liste |
|
259 |
(flat_oml (car liste)))) |
|
260 |
|
|
261 |
((and (listp (car liste)) (not (listp (cadr liste)))) |
|
262 |
(append (car liste) |
|
263 |
(flat_oml (cdr liste)))) |
|
264 |
|
|
265 |
((and (not (listp (car liste))) (not (listp (cadr liste)))) |
|
266 |
(append (list (car liste)) |
|
267 |
(flat_oml (cdr liste)))) |
|
268 |
|
|
269 |
((and (listp (car liste)) (listp (cadr liste))) |
|
270 |
(append (car liste) |
|
271 |
(flat_oml (cdr liste)))) |
|
272 |
|
|
273 |
((and (not (listp (car liste))) (listp (cadr liste))) |
|
274 |
(append (list (car liste)) |
|
275 |
(flat_oml (cdr liste)))))) |
|
276 |
|
c38e61
|
277 |
|
LL |
278 |
|
|
279 |
|
|
280 |
|
|
281 |
|
|
282 |
|
a53ccf
|
283 |
(defun get-pos_oml (positions seq) |
LL |
284 |
(let ((positions (if (atom positions) (list positions) positions))) |
|
285 |
(mapcar #'(lambda (it) (nth it seq)) seq))) |
|
286 |
|
|
287 |
|
c38e61
|
288 |
|
LL |
289 |
|
a53ccf
|
290 |
(defun permutations_oml (bag) |
LL |
291 |
|
|
292 |
(if (null bag) |
|
293 |
'(()) |
|
294 |
(mapcan #'(lambda (e) |
|
295 |
(mapcar #' (lambda (p) (cons e p)) |
|
296 |
(permutations_oml (remove e bag :count 1 :test #'eq)))) |
|
297 |
bag))) |
|
298 |
|
|
299 |
|
|
300 |
|
|
301 |
|
|
302 |
(defun modulo_oml (pos liste) |
|
303 |
(append |
|
304 |
(nthcdr pos liste) |
|
305 |
(butlast liste (- (length liste) pos)))) |
|
306 |
|
|
307 |
|
c38e61
|
308 |
|
LL |
309 |
|
|
310 |
|
a53ccf
|
311 |
(defun register-permut_oml (register-list) |
LL |
312 |
(remove-duplicates |
|
313 |
(om::flat |
|
314 |
(loop for r in register-list |
|
315 |
collect |
|
316 |
(mapcar #'(lambda (l) (permutations_oml |
|
317 |
(append (make-list l :initial-element r) |
|
318 |
(make-list (- (length register-list) l) :initial-element 0)))) |
|
319 |
(loop for x from 1 to (length register-list) |
|
320 |
collect x))) 2) :test 'equal)) |
|
321 |
|
|
322 |
|
c38e61
|
323 |
|
LL |
324 |
|
a53ccf
|
325 |
(defun find-pos_oml (item seq) |
LL |
326 |
(remove nil (loop for s in seq |
|
327 |
for x |
|
328 |
collect (if (equal item s) x)))) |
c38e61
|
329 |
|
LL |
330 |
|
a53ccf
|
331 |
|
LL |
332 |
|
|
333 |
(defun sum_oml (liste) |
|
334 |
(if (null (cdr liste)) |
|
335 |
(car liste) |
|
336 |
(+ (car liste) |
|
337 |
(sum_oml (cdr liste))))) |
|
338 |
|
c38e61
|
339 |
|
LL |
340 |
|
|
341 |
|
a53ccf
|
342 |
(defun variations_oml (liste pcl) |
LL |
343 |
(let* ((liste (remove-duplicates liste)) |
|
344 |
(liste (make-list pcl :initial-element liste))) |
|
345 |
(apply #'alexandria:map-product #'list liste))) |
c38e61
|
346 |
|
LL |
347 |
|
|
348 |
|
a53ccf
|
349 |
|
2b28da
|
350 |
|
LL |
351 |
(defun find-dups_oml (lst) |
|
352 |
(cond ((null lst) '()) |
|
353 |
((member (car lst) (cdr lst)) (cons (car lst) (find-dups_oml (cdr lst)))) |
|
354 |
(t (find-dups_oml (cdr lst))))) |
|
355 |
|
71cec5
|
356 |
|
MS |
357 |
|
|
358 |
|
|
359 |
|
c38e61
|
360 |
(defun tester_oml (test list) |
71cec5
|
361 |
|
c38e61
|
362 |
(cond |
LL |
363 |
((null list) t) |
|
364 |
((funcall test (car list)) (tester_oml test (cdr list))) |
|
365 |
(t nil) |
|
366 |
|
|
367 |
)) |
|
368 |
|
|
369 |
|
|
370 |
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; |
71cec5
|
371 |
|