1 ;;;; This file is part of LilyPond, the GNU music typesetter.
3 ;;;; Copyright (C) 2009--2015 Carl Sorensen <c_sorensen@byu.edu>
5 ;;;; LilyPond is free software: you can redistribute it and/or modify
6 ;;;; it under the terms of the GNU General Public License as published by
7 ;;;; the Free Software Foundation, either version 3 of the License, or
8 ;;;; (at your option) any later version.
10 ;;;; LilyPond is distributed in the hope that it will be useful,
11 ;;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
12 ;;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 ;;;; GNU General Public License for more details.
15 ;;;; You should have received a copy of the GNU General Public License
16 ;;;; along with LilyPond. If not, see <http://www.gnu.org/licenses/>.
18 ;;; specify time signature default settings
23 ;;; (time-signature . default-properties) entries.
25 ;;; where default-properties is an alist containing information about the
26 ;;; time signature. Each default-properties set can contain the
27 ;;; following entries:
29 ;;; (baseMoment . (/ numerator denominator))
30 ;;; (beatStructure . structure-list)
31 ;;; (beamExceptions . (alist of beam exceptions that don't follow beats))
33 ;;; The alist of beam exceptions has the following entries:
35 ;;; (end . grouping-rules)
36 ;;; (subdivide . grouping-rules) (not yet implemented, reserved for future use)
38 ;;; grouping-rules is an alist containing (beam-type . grouping-list) entries
40 ;;; beam-type is the length as a rational number
41 ;;; grouping-list is a list that specifies the
42 ;;; number of stems of the given duration that are grouped in a beamed unit.
43 ;;; For an exception, the duration used is beam-type. For measureBeats,
44 ;;; the duration used is baseMoment.
46 ;;; If an exception is specified for a given beam-type, it will apply to all
47 ;;; beams of shorter durations that don't have an individual exception, so
49 ;;; will cause all primary beams to be broken at 3/8, 6/8, and 8/8.
51 ;;; (1/32 . (16 8 4 4))
52 ;;; will cause all 1/32, 1/64, and 1/128 beams to be broken at 1/2, 3/4,
55 ;;; Tuplets are referenced using their actual (scaled) length, so
56 ;;; a 3/2 tuplet of the 1/8 kind would get exceptions looked up
59 ;;; If no values are given for baseMoment and measureBeats, default values
61 ;;; baseMoment gets the value (/ time-signature-denominator)
62 ;;; beatStructure gets a list of (3 3 3 ...), where the number of entries is the
63 ;;; number of beats, each containing 3 base-moments, if the time
64 ;;; signature numerator is greater than 3 and divisible by 3, and
65 ;;; a list of (1 1 1 ...), where the number of entries is the
66 ;;; number of base moments in a measure otherwise.
68 (define-public default-time-signature-settings
71 ;; use defaults, but end beams with 32nd notes each 1 4 beat
73 ((beamExceptions . ((end . ((1/32 . (8 8 8 8))))))))
75 ;; in 2/4, 2/8 and 2/16 time:
76 ;; use defaults, so no entries are necessary
79 ;; use defaults, but end beams with 32nd notes and higher each 1 4 beat
82 ((beamExceptions . ((end . ((1/32 . (8 8 8 8 8 8))))))))
85 ;; use defaults, but combine all beats into a unit if possible
87 ;; set all beams to end on beats, but 1 8 to beam entire measure
88 ;; in order to avoid beaming every beam type for the entire measure, we set
89 ;; triplets back to every beat.
91 ((beamExceptions . ((end . ((1/8 . (6)) ;1/8 note whole measure
92 (1/12 . (3 3 3)))))))) ;Anything shorter by beat
95 ;; beam entire measure together
96 ((3 . 8) . ((beamExceptions . ((end . ((1/8 . (3))))))))
99 ;; use defaults -- no entries necessary
102 ;; use defaults, but end beams with 16th notes or finer each 1 4 beat
104 ((beamExceptions . ((end . ((1/16 . (4 4 4 4 4 4 4 4))))))))
106 ;; in 4 4 (common) time:
107 ;; use defaults, but combine beats 1,2 and 3,4 if only 8th notes
108 ;; NOTE: Any changes here need to be duplicated in
109 ;; ly/engraver-init.ly where the default time signature is set
112 ((beamExceptions . ((end . ((1/8 . (4 4)) ; 1/8 notes half measure
113 (1/12 . (3 3 3 3)))))))) ;Anything shorter by beat
116 ;; combine beats 1 and 2, so beam in 2
117 ((4 . 8) . ((beatStructure . (2 2))))
120 ;; use defaults, so no entries necessary
123 ;; use defaults, but end beams with 32nd or finer each 1/4 beat
125 ((beamExceptions . ((end . ((1/16 . (4 4 4 4 4 4))))))))
128 ;; use defaults, so no entries necessary
131 ;; use defaults, so no entries necessary
134 ;; use defaults, but end beams with 32nd or finer each 1 4 beat
136 ((beamExceptions . ((end . ((1/32 . (8 8 8 8 8 8 8 8))))))))
139 ;; use defaults, so no entries necessary
142 ;; use defaults, so no entries necessary
145 ;; use defaults, but end beams with 32nd or finer notes each 1 4 beat
147 ((beamExceptions . ((end . ((1/32 . (8 8 8 8 8 8 8 8 8 8 8 8))))))))
150 ;; use defaults, so no entries necessary
153 ;; use defaults, so no entries necessary
156 ;; default: group (3 2)
158 ((beatStructure . (3 2))))
161 ;; default: group (3 3 2)
163 ((beatStructure . (3 3 2))))
165 )) ; end of alist definition
168 ;;; Accessor and constructor functions
171 (define (get-setting my-symbol time-signature time-signature-settings)
172 "Get setting @code{my-symbol} for @code{time-signature} from
173 @code{time-signature-settings}."
174 (let ((my-time-signature-settings
175 (assoc-get time-signature time-signature-settings '())))
176 (assoc-get my-symbol my-time-signature-settings '())))
178 (define-public (make-setting base-fraction
182 (cons 'baseMoment (if (pair? base-fraction)
183 (/ (car base-fraction) (cdr base-fraction))
185 (cons 'beatStructure beat-structure)
186 (cons 'beamExceptions beam-exceptions)))
188 (define-public (base-length time-signature time-signature-settings)
189 "Get @code{baseMoment} rational value for @var{time-signature} from
190 @var{time-signature-settings}."
191 (let ((return-value (get-setting 'baseMoment
193 time-signature-settings)))
194 (if (null? return-value)
195 (/ (cdr time-signature))
198 (define-public (beat-structure base-length time-signature time-signature-settings)
199 "Get @code{beatStructure} value in @var{base-length} units
200 for @var{time-signature} from @var{time-signature-settings}."
202 (let ((return-value (get-setting 'beatStructure
204 time-signature-settings)))
205 (if (null? return-value)
206 ;; calculate default beatStructure
207 (let* ((numerator (car time-signature))
208 (group-size (if (and (> numerator 3)
209 (zero? (remainder numerator 3)))
212 (beat-count (/ (car time-signature)
216 (if (integer? beat-count)
217 (make-list beat-count group-size)
219 ;; use value obtained from time-signature-settings
222 (define-public (beam-exceptions time-signature time-signature-settings)
223 "Get @code{beamExceptions} value for @var{time-signature} from
224 @var{time-signature-settings}."
225 (get-setting 'beamExceptions time-signature time-signature-settings))
228 ;;; Functions for overriding time-signature settings
231 (define (override-property-setting context property setting value)
232 "Like the C++ code that executes \\override, but without type
235 (ly:context-set-property!
238 (cons (cons setting value) (ly:context-property context property)))))
240 (define (revert-property-setting context property setting)
241 "Like the C++ code that executes \revert, but without type
244 (define (entry-count alist entry-key)
245 "Count the number of entries in alist with a key of
249 ((equal? (caar alist) entry-key)
250 (+ 1 (entry-count (cdr alist) entry-key)))
251 (else (entry-count (cdr alist) entry-key))))
253 (define (revert-member alist entry-key)
254 "Return ALIST, with the first entry having a key of
255 ENTRY-KEY removed. ALIST is not modified, instead
256 a fresh copy of the list-head is made."
259 ((equal? (caar alist) entry-key) (cdr alist))
260 (else (cons (car alist)
261 (revert-member (cdr alist) entry-key)))))
263 ;; body of revert-property-setting
264 (let ((current-value (ly:context-property context property)))
265 (if (> (entry-count current-value setting) 0)
266 (ly:context-set-property!
269 (revert-member current-value setting)))))
271 (define-public (override-time-signature-setting time-signature setting)
272 "Override the time signature settings for the context in
273 @var{time-signature}, with the new setting alist @var{setting}."
276 (lambda (c) (override-property-setting
278 'timeSignatureSettings
283 (define-public (revert-time-signature-setting time-signature)
287 (revert-property-setting
289 'timeSignatureSettings
296 ;;;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
297 ;;; Formatting of complex/compound time signatures
299 ; There ought to be a \join-line sep {...} command
300 (define (insert-markups l m)
301 (let ((ll (reverse l)))
302 (let join-markups ((markups (list (car ll)))
303 (remaining (cdr ll)))
304 (if (pair? remaining)
305 (join-markups (cons (car remaining) (cons m markups)) (cdr remaining))
308 ;;; Use a centered-column inside a left-column, because the centered column
309 ;;; moves its reference point to the center, which the left-column undoes.
310 (define (format-time-fraction time-sig-fraction)
311 (let* ((revargs (reverse (map number->string time-sig-fraction)))
313 (nums (reverse (cdr revargs))))
314 (make-override-markup '(baseline-skip . 0)
315 (make-left-column-markup
316 (list (make-center-column-markup
317 (list (make-line-markup (insert-markups nums "+"))
320 (define (format-time-numerator time-sig)
321 (make-vcenter-markup (number->string (car time-sig))))
323 (define (format-time-element time-sig)
324 (cond ((number-pair? time-sig)
325 (format-time-fraction (list (car time-sig) (cdr time-sig))))
326 ((pair? (cdr time-sig))
327 (format-time-fraction time-sig))
329 (format-time-numerator time-sig))))
331 (define (format-time-list time-sig)
332 (make-override-markup '(baseline-skip . 0)
334 (insert-markups (map format-time-element time-sig)
335 (make-vcenter-markup "+")))))
337 (define (format-compound-time time-sig)
340 ((number? time-sig) (format-time-element (list time-sig)))
341 ((number-pair? time-sig)
342 (format-time-element (list (car time-sig) (cdr time-sig))))
343 ((pair? (car time-sig)) (format-time-list time-sig))
344 (else (format-time-element time-sig)))))
346 (define-markup-command (compound-meter layout props time-sig)
349 "Draw a numeric time signature.
351 @lilypond[verbatim,quote]
354 \\line { Single number: \\compound-meter #3 }
355 \\line { Conventional: \\compound-meter #'(4 . 4)
356 or \\compound-meter #'(4 4) }
357 \\line { Compound: \\compound-meter #'(2 3 8) }
358 \\line { Single-number compound: \\compound-meter #'((2) (3)) }
359 \\line { Complex compound: \\compound-meter #'((2 3 8) (3 4)) }
364 (interpret-markup layout props (format-compound-time time-sig)))
366 (add-simple-time-signature-style 'numbered make-compound-meter-markup)
368 (add-simple-time-signature-style 'single-digit
369 (lambda (fraction) (make-compound-meter-markup (car fraction))))
371 ;;;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
372 ;;; Formatting of symbolic time signatures
374 (define-public (make-glyph-time-signature-markup style fraction)
375 "Make markup for a symbolic time signature. If the music font does not have a glyph for the requested style and fraction, issue a warning and make a numbered time signature instead."
376 (make-first-visible-markup
377 (list (make-musicglyph-markup (string-append
379 (symbol->string style)
380 (number->string (car fraction))
381 (number->string (cdr fraction))))
382 (make-compound-meter-markup fraction))))
384 (define-public (make-c-time-signature-markup fraction)
385 "Make markup for the `C' time signature style."
386 (let ((n (car fraction))
388 ; check specific fractions to avoid warnings when no glyph exists
389 (if (or (and (= n 2) (= d 2))
390 (and (= n 4) (= d 4)))
391 (make-glyph-time-signature-markup 'C fraction)
392 (make-compound-meter-markup fraction))))
394 (add-simple-time-signature-style 'C make-c-time-signature-markup)
395 (add-simple-time-signature-style 'default make-c-time-signature-markup)
397 ;;;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
398 ;;; Measure length calculation of (possibly complex) compound time signatures
400 (define (calculate-time-fraction time-sig-fraction)
401 (let* ((revargs (reverse time-sig-fraction))
403 (num (apply + (cdr revargs))))
404 (ly:make-moment num den)))
406 (define (calculate-complex-compound-time time-sig)
407 (let add-moment ((moment ZERO-MOMENT)
408 (remaining (map calculate-time-fraction time-sig)))
409 (if (pair? remaining)
410 (add-moment (ly:moment-add moment (car remaining)) (cdr remaining))
413 (define-public (calculate-compound-measure-length time-sig)
415 ((not (pair? time-sig)) (ly:make-moment 4 4))
416 ((pair? (car time-sig)) (calculate-complex-compound-time time-sig))
417 (else (calculate-time-fraction time-sig))))
420 ;;;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
421 ;;; Base beat length: Use the smallest denominator from all fraction
423 (define (calculate-compound-base-beat-full time-sig)
424 (apply max (map last time-sig)))
426 (define-public (calculate-compound-base-beat time-sig)
429 ((not (pair? time-sig)) 4)
430 ((pair? (car time-sig)) (calculate-compound-base-beat-full time-sig))
431 (else (calculate-compound-base-beat-full (list time-sig))))))
434 ;;;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
437 (define (normalize-fraction frac beat)
438 (let* ((thisbeat (car (reverse frac)))
439 (factor (/ beat thisbeat)))
440 (map (lambda (f) (* factor f)) frac)))
442 (define (beat-grouping-internal time-sig)
443 ;; Normalize to given beat, extract the beats and join them to one list
444 (let* ((beat (calculate-compound-base-beat-full time-sig))
445 (normalized (map (lambda (f) (normalize-fraction f beat)) time-sig))
446 (beats (map (lambda (f) (drop-right f 1)) normalized)))
447 (concatenate beats)))
449 (define-public (calculate-compound-beat-grouping time-sig)
451 ((not (pair? time-sig)) '(2 . 2))
452 ((pair? (car time-sig)) (beat-grouping-internal time-sig))
453 (else (beat-grouping-internal (list time-sig)))))