Source code for abjadext.nauert.attackpointoptimizers

import abc

import abjad


[docs] class AttackPointOptimizer(abc.ABC): """ Abstract attack-point optimizer. Attack-point optimizers may alter the number, order, and individual durations of leaves in a logical tie, but may not alter the overall duration of that logical tie. They effectively "clean up" notation, post-quantization. """ ### CLASS VARIABLES ### __slots__ = () ### SPECIAL METHODS ###
[docs] @abc.abstractmethod def __call__(self, argument): """Calls attack-point optimizer.""" raise NotImplementedError
[docs] class MeasurewiseAttackPointOptimizer(AttackPointOptimizer): r""" Measurewise attack-point optimizer. Attempts to optimize attack points in an expression with regard to the effective time signature of that expression. Only acts on measures. .. container:: example >>> staff = abjad.Staff("c'8 d'8 e'8 f'8 g'8 a'8 b'8 c''8") >>> abjad.show(staff) # doctest: +SKIP >>> source_tempo = abjad.MetronomeMark(abjad.Duration(1, 4), 60) >>> q_events = nauert.QEventSequence.from_tempo_scaled_leaves( ... staff[:], ... tempo=source_tempo, ... ) >>> target_tempo = abjad.MetronomeMark(abjad.Duration(1, 4), 54) >>> q_schema = nauert.MeasurewiseQSchema( ... tempo=target_tempo, ... ) .. container:: example Without the measure-wise attack-point optimizer: >>> result = nauert.quantize(q_events, q_schema=q_schema) >>> abjad.show(result) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(result) >>> print(string) \new Voice { { %%% \time 4/4 %%% \tempo 4=54 c'16.. d'64 ~ \tuplet 5/4 { d'8 e'32 ~ } \tuplet 7/4 { e'8 ~ e'32 f'16 ~ } \tuplet 5/4 { f'16. g'16 ~ } g'16 a'16 ~ \tuplet 5/4 { a'16 b'16. ~ } \tuplet 7/4 { b'16 c''8 ~ c''32 ~ } \tuplet 5/4 { c''32 r32 r32 r32 r32 } } } .. container:: example With the measure-wise attack-point optimizer: >>> optimizer = nauert.MeasurewiseAttackPointOptimizer() >>> result = nauert.quantize( ... q_events, ... attack_point_optimizer=optimizer, ... q_schema=q_schema, ... ) >>> abjad.show(result) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(result) >>> print(string) \new Voice { { %%% \time 4/4 %%% \tempo 4=54 c'16.. d'64 ~ \tuplet 5/4 { d'16. ~ d'32 e'32 ~ } \tuplet 7/4 { e'16. ~ e'16 f'16 ~ } \tuplet 5/4 { f'16. g'16 ~ } g'16 a'16 ~ \tuplet 5/4 { a'16 b'32 ~ b'16 ~ } \tuplet 7/4 { b'16 c''32 ~ c''8 ~ } \tuplet 5/4 { c''32 r16 r16 } } } """ ### CLASS VARIABLES ### __slots__ = () ### SPECIAL METHODS ###
[docs] def __call__( self, argument: abjad.Container, time_signature: abjad.TimeSignature | None = None, ) -> None: """ Calls measurewise attack-point optimizer. """ assert isinstance(argument, abjad.Container) leaf = abjad.get.leaf(argument, 0) time_signature = time_signature or abjad.get.indicator( leaf, abjad.TimeSignature ) assert time_signature is not None, repr(time_signature) all_annotations = self._get_attachment_annotations_of_logical_ties(argument[:]) abjad.Meter.rewrite_meter(argument[:], time_signature, boundary_depth=1) assert len(all_annotations) == len( list(abjad.iterate.logical_ties(argument[:], pitched=True)) ) self._reannotate_logical_ties(argument[:], all_annotations)
@staticmethod def _get_attachment_annotations_of_logical_ties(components): all_annotations = [] for logical_tie in abjad.iterate.logical_ties(components, pitched=True): first_leaf = abjad.get.leaf(logical_tie, 0) annotation = abjad.get.annotation(first_leaf, "q_event_attachments") all_annotations.append(annotation) return all_annotations @staticmethod def _reannotate_logical_ties(components, all_annotations): for logical_tie, annotation in zip( abjad.iterate.logical_ties(components, pitched=True), all_annotations ): first_leaf = abjad.get.leaf(logical_tie, 0) abjad.annotate(first_leaf, "q_event_attachments", annotation)
[docs] class NaiveAttackPointOptimizer(AttackPointOptimizer): r""" Naive attack-point optimizer. (The default attack-point optimizer) Optimizes attack points by fusing tie leaves within logical ties with leaf durations decreasing monotonically. Logical ties will be partitioned into sub-logical-ties if leaves are found with metronome marks attached. .. container:: example >>> staff = abjad.Staff("c'8 d'8 e'8 f'8 g'8 a'8 b'8 c''8") >>> abjad.show(staff) # doctest: +SKIP >>> source_tempo = abjad.MetronomeMark(abjad.Duration(1, 4), 60) >>> q_events = nauert.QEventSequence.from_tempo_scaled_leaves( ... staff[:], ... tempo=source_tempo, ... ) >>> target_tempo = abjad.MetronomeMark(abjad.Duration(1, 4), 54) >>> q_schema = nauert.MeasurewiseQSchema( ... tempo=target_tempo, ... ) .. container:: example >>> optimizer = nauert.NaiveAttackPointOptimizer() >>> result = nauert.quantize( ... q_events, ... attack_point_optimizer=optimizer, ... q_schema=q_schema, ... ) >>> abjad.show(result) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(result) >>> print(string) \new Voice { { %%% \time 4/4 %%% \tempo 4=54 c'16.. d'64 ~ \tuplet 5/4 { d'8 e'32 ~ } \tuplet 7/4 { e'8 ~ e'32 f'16 ~ } \tuplet 5/4 { f'16. g'16 ~ } g'16 a'16 ~ \tuplet 5/4 { a'16 b'16. ~ } \tuplet 7/4 { b'16 c''8 ~ c''32 ~ } \tuplet 5/4 { c''32 r32 r32 r32 r32 } } } """ ### CLASS VARIABLES ### __slots__ = () ### SPECIAL METHODS ###
[docs] def __call__(self, argument): """ Calls naive attack-point optimizer. """ for logical_tie in abjad.iterate.logical_ties( argument, grace=False, reverse=True ): sub_logical_ties = [] current_sub_logical_tie = [] for leaf in logical_tie: tempos = leaf._get_indicators(abjad.MetronomeMark) if tempos: if current_sub_logical_tie: current_sub_logical_tie = abjad.LogicalTie( current_sub_logical_tie ) sub_logical_ties.append(current_sub_logical_tie) current_sub_logical_tie = [] current_sub_logical_tie.append(leaf) if current_sub_logical_tie: current_sub_logical_tie = abjad.LogicalTie(current_sub_logical_tie) sub_logical_ties.append(current_sub_logical_tie) for sub_logical_tie in sub_logical_ties: abjad.mutate._fuse_leaves_by_immediate_parent(sub_logical_tie)
[docs] class NullAttackPointOptimizer(AttackPointOptimizer): r""" Null attack-point optimizer. Performs no attack point optimization. .. container:: example >>> staff = abjad.Staff("c'8 d'8 e'8 f'8 g'8 a'8 b'8 c''8") >>> abjad.show(staff) # doctest: +SKIP >>> source_tempo = abjad.MetronomeMark(abjad.Duration(1, 4), 60) >>> q_events = nauert.QEventSequence.from_tempo_scaled_leaves( ... staff[:], ... tempo=source_tempo, ... ) >>> target_tempo = abjad.MetronomeMark(abjad.Duration(1, 4), 54) >>> q_schema = nauert.MeasurewiseQSchema( ... tempo=target_tempo, ... ) .. container:: example >>> optimizer = nauert.NullAttackPointOptimizer() >>> result = nauert.quantize( ... q_events, ... attack_point_optimizer=optimizer, ... q_schema=q_schema, ... ) >>> abjad.show(result) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(result) >>> print(string) \new Voice { { %%% \time 4/4 %%% \tempo 4=54 c'16 ~ c'32 ~ c'64 d'64 ~ \tuplet 5/4 { d'32 ~ d'32 ~ d'32 ~ d'32 e'32 ~ } \tuplet 7/4 { e'32 ~ e'32 ~ e'32 ~ e'32 ~ e'32 f'32 ~ f'32 ~ } \tuplet 5/4 { f'32 ~ f'32 ~ f'32 g'32 ~ g'32 ~ } g'16 a'16 ~ \tuplet 5/4 { a'32 ~ a'32 b'32 ~ b'32 ~ b'32 ~ } \tuplet 7/4 { b'32 ~ b'32 c''32 ~ c''32 ~ c''32 ~ c''32 ~ c''32 ~ } \tuplet 5/4 { c''32 r32 r32 r32 r32 } } } """ ### CLASS VARIABLES ### __slots__ = () ### SPECIAL METHODS ###
[docs] def __call__(self, argument): """ Calls null attack-point optimizer. """ pass