Source code for abjad.indicators

"""
Indicators.
"""

import collections
import dataclasses
import fractions
import functools
import math
import typing

from . import contributions as _contributions
from . import duration as _duration
from . import enums as _enums
from . import math as _math
from . import overrides as _overrides
from . import pitch as _pitch
from . import sequence as _sequence
from . import string as _string

_EMPTY_CHORD = "<>"


[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class Arpeggio: r""" Arpeggio. .. container:: example Without direction arrow: >>> chord = abjad.Chord("<c' e' g' c''>4") >>> arpeggio = abjad.Arpeggio() >>> abjad.attach(arpeggio, chord) >>> staff = abjad.Staff([chord]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { <c' e' g' c''>4 \arpeggio } .. container:: example With direction arrow: >>> chord = abjad.Chord("<c' e' g' c''>4") >>> arpeggio = abjad.Arpeggio(direction=abjad.DOWN) >>> abjad.attach(arpeggio, chord) >>> staff = abjad.Staff([chord]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \arpeggioArrowDown <c' e' g' c''>4 \arpeggio } .. container:: example Tweaks: >>> chord = abjad.Chord("<c' e' g' c''>4") >>> arpeggio = abjad.Arpeggio() >>> bundle = abjad.bundle(arpeggio, r"- \tweak color #blue") >>> abjad.attach(bundle, chord) >>> staff = abjad.Staff([chord]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { <c' e' g' c''>4 - \tweak color #blue \arpeggio } """ direction: _enums.Vertical | None = None post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after"
[docs] def __post_init__(self): if self.direction is not None: assert isinstance(self.direction, _enums.Vertical), repr(self.direction)
def _get_lilypond_format(self): return r"\arpeggio" def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) site.articulations.append(r"\arpeggio") if self.direction in (_enums.UP, _enums.DOWN): if self.direction is _enums.UP: command = r"\arpeggioArrowUp" else: command = r"\arpeggioArrowDown" contributions.before.commands.append(command) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class Articulation: r""" Articulation. .. container:: example >>> abjad.Articulation("staccato") Articulation(name='staccato') >>> abjad.Articulation(".") Articulation(name='.') Use ``direction=abjad.UP`` like this: >>> note = abjad.Note("c'4") >>> articulation = abjad.Articulation("staccato") >>> abjad.attach(articulation, note, direction=abjad.UP) >>> abjad.show(note) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(note) >>> print(string) c'4 ^ \staccato Use ``direction=abjad.DOWN`` like this: >>> note = abjad.Note("c'4") >>> articulation = abjad.Articulation("staccato") >>> abjad.attach(articulation, note, direction=abjad.DOWN) >>> abjad.show(note) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(note) >>> print(string) c'4 _ \staccato Use ``dataclasses.replace()`` like this: >>> import dataclasses >>> articulation = abjad.Articulation(".") >>> dataclasses.replace(articulation) Articulation(name='.') """ name: str post_event: typing.ClassVar[bool] = True shortcut_to_word: typing.ClassVar[dict[str, str]] = { "^": "marcato", "+": "stopped", "-": "tenuto", "|": "staccatissimo", ">": "accent", ".": "staccato", "_": "portato", } site: typing.ClassVar[str] = "after"
[docs] def __post_init__(self): assert isinstance(self.name, str), repr(self.name)
def _get_lilypond_format(self, wrapper=None): if self.name: string = self.shortcut_to_word.get(self.name) if not string: string = self.name if wrapper is not None and wrapper.direction: direction_ = _string.to_tridirectional_lilypond_symbol( wrapper.direction ) direction = direction_ else: direction = "-" return rf"{direction} \{string}" else: return "" def _get_contributions(self, *, component=None, wrapper=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._get_lilypond_format(wrapper=wrapper) site.articulations.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class BarLine: r""" Bar line. .. container:: example Final bar line: >>> staff = abjad.Staff("c'4 d'4 e'4 f'4", name="Staff") >>> score = abjad.Score([staff], name="Score") >>> bar_line = abjad.BarLine("|.") >>> abjad.attach(bar_line, staff[-1]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \context Score = "Score" << \context Staff = "Staff" { c'4 d'4 e'4 f'4 \bar "|." } >> .. container:: example Specify repeat bars like this: >>> staff = abjad.Staff("c'4 d' e' f' g' a' b' c''", name="Staff") >>> score = abjad.Score([staff], name="Score") >>> abjad.attach(abjad.BarLine(".|:"), staff[3]) >>> abjad.attach(abjad.BarLine(":|."), staff[-1]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \context Score = "Score" << \context Staff = "Staff" { c'4 d'4 e'4 f'4 \bar ".|:" g'4 a'4 b'4 c''4 \bar ":|." } >> This allows you to bypass LilyPond's ``\volta`` command. """ abbreviation: str = "|" site: str = "after" context: typing.ClassVar[str] = "Score" # find_context_on_attach: typing.ClassVar[bool] = True def _get_lilypond_format(self): return rf'\bar "{self.abbreviation}"' def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._get_lilypond_format() site.commands.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class BeamCount: r""" LilyPond ``\setLeftBeamCount``, ``\setRightBeamCount`` command. .. container:: example >>> abjad.BeamCount() BeamCount(left=0, right=0) """ left: int = 0 right: int = 0 site: typing.ClassVar[str] = "before" def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = rf"\set stemLeftBeamCount = {self.left}" site.commands.append(string) string = rf"\set stemRightBeamCount = {self.right}" site.commands.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class BendAfter: r""" Fall or doit. .. container:: example A fall: >>> note = abjad.Note("c'4") >>> bend = abjad.BendAfter(-4) >>> abjad.attach(bend, note) >>> abjad.show(note) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(note) >>> print(string) c'4 - \bendAfter #'-4 .. container:: example A doit: >>> note = abjad.Note("c'4") >>> bend = abjad.BendAfter(2) >>> abjad.attach(bend, note) >>> abjad.show(note) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(note) >>> print(string) c'4 - \bendAfter #'2 .. container:: example Tweaks: >>> staff = abjad.Staff("c'4 d' e' f'") >>> bend_after = abjad.BendAfter(-4) >>> bundle = abjad.bundle(bend_after, r"- \tweak color #blue") >>> abjad.attach(bundle, staff[0]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4 - \tweak color #blue - \bendAfter #'-4 d'4 e'4 f'4 } """ bend_amount: int | float = -4 post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" time_orientation: typing.ClassVar[_enums.Horizontal] = _enums.RIGHT def _get_lilypond_format(self): return rf"- \bendAfter #'{self.bend_amount}" def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._get_lilypond_format() site.articulations.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class BreathMark: r""" Breath mark. .. container:: example >>> voice = abjad.Voice("c'8 d' e' f' g' a' b' c''") >>> staff = abjad.Staff([voice]) >>> abjad.beam(voice[:4]) >>> abjad.beam(voice[4:]) >>> abjad.attach(abjad.BreathMark(), voice[3]) >>> abjad.attach(abjad.BreathMark(), voice[7]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \new Voice { c'8 [ d'8 e'8 f'8 ] \breathe g'8 [ a'8 b'8 c''8 ] \breathe } } .. container:: example REGRESSION. Abjad parses LilyPond's ``\breathe`` command correctly: >>> staff = abjad.Staff(r"c'4 d' e' f' \breathe") >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4 d'4 e'4 f'4 \breathe } .. container:: example Tweaks: >>> note = abjad.Note("c'4") >>> breath = abjad.BreathMark() >>> bundle = abjad.bundle(breath, r"\tweak color #blue") >>> abjad.attach(bundle, note) >>> staff = abjad.Staff([note]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4 \tweak color #blue \breathe } """ site: typing.ClassVar[str] = "after" time_orientation: typing.ClassVar[_enums.Horizontal] = _enums.RIGHT def _before_attach(self, deactivate, component): for indicator in component._get_indicators(): if isinstance(indicator, type(self)) and indicator.site == self.site: classname = type(component).__name__ message = f"can not attach {self!r} to {classname}:" message += f"\n {indicator!r} is already attached to {classname}." raise Exception(message) def _get_lilypond_format(self): return r"\breathe" def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._get_lilypond_format() site.commands.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class Clef: r""" Clef. .. container:: example Some available clefs: >>> staff = abjad.Staff("c'8 d'8 e'8 f'8 g'8 a'8 b'8 c''8") >>> clef = abjad.Clef("treble") >>> abjad.attach(clef, staff[0]) >>> clef = abjad.Clef("alto") >>> abjad.attach(clef, staff[1]) >>> clef = abjad.Clef("bass") >>> abjad.attach(clef, staff[2]) >>> clef = abjad.Clef("treble^8") >>> abjad.attach(clef, staff[3]) >>> clef = abjad.Clef("bass_8") >>> abjad.attach(clef, staff[4]) >>> clef = abjad.Clef("tenor") >>> abjad.attach(clef, staff[5]) >>> clef = abjad.Clef("bass^15") >>> abjad.attach(clef, staff[6]) >>> clef = abjad.Clef("percussion") >>> abjad.attach(clef, staff[7]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \clef "treble" c'8 \clef "alto" d'8 \clef "bass" e'8 \clef "treble^8" f'8 \clef "bass_8" g'8 \clef "tenor" a'8 \clef "bass^15" b'8 \clef "percussion" c''8 } .. container:: example Clefs can be tagged: >>> staff = abjad.Staff("c'4 d' e' f'") >>> abjad.attach( ... abjad.Clef("treble"), staff[0], tag=abjad.Tag("+PARTS") ... ) >>> abjad.show(staff) # doctest: +SKIP >>> string = abjad.lilypond(staff, tags=True) >>> print(string) \new Staff { %! +PARTS \clef "treble" c'4 d'4 e'4 f'4 } .. container:: example LilyPond can not handle simultaneous clefs: >>> voice_1 = abjad.Voice("e'8 g' f' a' g' b'") >>> voice_2 = abjad.Voice("c'4. c,8 b,, a,,") >>> staff = abjad.Staff([voice_1, voice_2], simultaneous=True) >>> abjad.attach(abjad.Clef("treble"), voice_1[0], context="Voice") >>> command = abjad.VoiceNumber(1) >>> abjad.attach(command, voice_1[0]) >>> voice_1.consists_commands.append("Clef_engraver") >>> abjad.attach(abjad.Clef("treble"), voice_2[0], context="Voice") >>> abjad.attach(abjad.Clef("bass"), voice_2[1], context="Voice") >>> command = abjad.VoiceNumber(2) >>> abjad.attach(command, voice_2[0]) >>> voice_2.consists_commands.append("Clef_engraver") >>> staff.remove_commands.append("Clef_engraver") >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \remove Clef_engraver } << \new Voice \with { \consists Clef_engraver } { \clef "treble" \voiceOne e'8 g'8 f'8 a'8 g'8 b'8 } \new Voice \with { \consists Clef_engraver } { \clef "treble" \voiceTwo c'4. \clef "bass" c,8 b,,8 a,,8 } >> But Abjad components work fine: >>> for leaf in abjad.select.leaves(voice_1): ... leaf, abjad.get.effective(leaf, abjad.Clef) ... (Note("e'8"), Clef(name='treble', hide=False)) (Note("g'8"), Clef(name='treble', hide=False)) (Note("f'8"), Clef(name='treble', hide=False)) (Note("a'8"), Clef(name='treble', hide=False)) (Note("g'8"), Clef(name='treble', hide=False)) (Note("b'8"), Clef(name='treble', hide=False)) >>> for leaf in abjad.select.leaves(voice_2): ... leaf, abjad.get.effective(leaf, abjad.Clef) ... (Note("c'4."), Clef(name='treble', hide=False)) (Note('c,8'), Clef(name='bass', hide=False)) (Note('b,,8'), Clef(name='bass', hide=False)) (Note('a,,8'), Clef(name='bass', hide=False)) .. container:: example Middle-C position: >>> abjad.Clef("treble").middle_c_position() StaffPosition(number=-6) >>> abjad.Clef("alto").middle_c_position() StaffPosition(number=0) .. container:: example Set ``hide=True`` when clef should not appear in output (but should still determine effective clef). .. container:: example >>> staff = abjad.Staff("c'4 d' e' f'") >>> abjad.attach(abjad.Clef("treble"), staff[0]) >>> abjad.attach(abjad.Clef("alto", hide=True), staff[2]) >>> abjad.show(staff) # doctest: +SKIP >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \clef "treble" c'4 d'4 e'4 f'4 } >>> for leaf in abjad.iterate.leaves(staff): ... leaf, abjad.get.effective(leaf, abjad.Clef) ... (Note("c'4"), Clef(name='treble', hide=False)) (Note("d'4"), Clef(name='treble', hide=False)) (Note("e'4"), Clef(name='alto', hide=True)) (Note("f'4"), Clef(name='alto', hide=True)) """ name: str = "treble" hide: bool = dataclasses.field(compare=False, default=False) context: typing.ClassVar[str] = "Staff" # find_context_on_attach: typing.ClassVar[bool] = True persistent: typing.ClassVar[bool] = True redraw: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "before" clef_name_to_middle_c_position = { "treble": -6, "alto": 0, "varC": 0, "tenor": 2, "tenorvarC": 2, "bass": 6, "french": -8, "soprano": -4, "mezzosoprano": -2, "baritone": 4, "varbaritone": 4, "percussion": 0, "tab": 0, } _to_width = { "alto": 2.75, "varC": 2.75, "bass": 2.75, "percussion": 2.5, "tenor": 2.75, "tenorvarC": 2.75, "treble": 2.5, }
[docs] def __post_init__(self): assert isinstance(self.name, str), repr(self.name) assert isinstance(self.hide, bool), repr(self.hide)
def _calculate_middle_c_position(self, clef_name): alteration = 0 if "_" in self.name: base_name, part, suffix = clef_name.partition("_") if suffix == "8": alteration = 7 elif suffix == "15": alteration = 13 else: raise Exception(f"bad clef alteration suffix: {suffix!r}.") elif "^" in self.name: base_name, part, suffix = clef_name.partition("^") if suffix == "8": alteration = -7 elif suffix == "15": alteration = -13 else: raise Exception(f"bad clef alteration suffix: {suffix!r}.") else: base_name = clef_name return self.clef_name_to_middle_c_position[base_name] + alteration def _clef_name_to_staff_position_zero(self, clef_name): return { "treble": _pitch.NamedPitch("B4"), "alto": _pitch.NamedPitch("C4"), "varC": _pitch.NamedPitch("C4"), "tenor": _pitch.NamedPitch("A3"), "tenorvarC": _pitch.NamedPitch("A3"), "bass": _pitch.NamedPitch("D3"), "french": _pitch.NamedPitch("D5"), "soprano": _pitch.NamedPitch("G4"), "mezzosoprano": _pitch.NamedPitch("E4"), "baritone": _pitch.NamedPitch("F3"), "varbaritone": _pitch.NamedPitch("F3"), "percussion": None, "tab": None, }[clef_name] def _get_lilypond_format(self): return rf'\clef "{self.name}"' def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() if not self.hide: site = getattr(contributions, self.site) string = self._get_lilypond_format() site.commands.append(string) return contributions
[docs] @classmethod def from_pitches(class_, pitches) -> "Clef": """ Makes clef from ``pitches``. .. container:: example >>> notes = abjad.makers.make_notes(list(range(-12, -6)), [(1, 4)]) >>> staff = abjad.Staff(notes) >>> pitches = abjad.iterate.pitches(staff) >>> abjad.Clef.from_pitches(pitches) Clef(name='bass', hide=False) Chooses between treble and bass based on minimal number of ledger lines. """ diatonic_pitch_numbers = [ pitch._get_diatonic_pitch_number() for pitch in pitches ] max_diatonic_pitch_number = max(diatonic_pitch_numbers) min_diatonic_pitch_number = min(diatonic_pitch_numbers) lowest_treble_line_pitch = _pitch.NamedPitch("E4") lowest_treble_line_diatonic_pitch_number = ( lowest_treble_line_pitch._get_diatonic_pitch_number() ) candidate_steps_below_treble = ( lowest_treble_line_diatonic_pitch_number - min_diatonic_pitch_number ) highest_bass_line_pitch = _pitch.NamedPitch("A3") highest_bass_line_diatonic_pitch_number = ( highest_bass_line_pitch._get_diatonic_pitch_number() ) candidate_steps_above_bass = ( max_diatonic_pitch_number - highest_bass_line_diatonic_pitch_number ) if candidate_steps_above_bass < candidate_steps_below_treble: return class_("bass") else: return class_("treble")
[docs] def middle_c_position(self): """ Gets middle C position. """ middle_c_position = self._calculate_middle_c_position(self.name) middle_c_position = _pitch.StaffPosition(middle_c_position) return middle_c_position
[docs] def to_pitch(self, staff_position) -> _pitch.NamedPitch: """ Changes ``staff_position`` to pitch. Treble clef: .. container:: example >>> clef = abjad.Clef("treble") >>> for n in range(-6, 6): ... staff_position = abjad.StaffPosition(n) ... pitch = clef.to_pitch(staff_position) ... print(f"{staff_position!r:25}{pitch!r}") ... StaffPosition(number=-6) NamedPitch("c'") StaffPosition(number=-5) NamedPitch("d'") StaffPosition(number=-4) NamedPitch("e'") StaffPosition(number=-3) NamedPitch("f'") StaffPosition(number=-2) NamedPitch("g'") StaffPosition(number=-1) NamedPitch("a'") StaffPosition(number=0) NamedPitch("b'") StaffPosition(number=1) NamedPitch("c''") StaffPosition(number=2) NamedPitch("d''") StaffPosition(number=3) NamedPitch("e''") StaffPosition(number=4) NamedPitch("f''") StaffPosition(number=5) NamedPitch("g''") .. container:: example Bass clef: >>> clef = abjad.Clef("bass") >>> for n in range(-6, 6): ... staff_position = abjad.StaffPosition(n) ... pitch = clef.to_pitch(staff_position) ... print(f"{staff_position!r:25}{pitch!r}") ... StaffPosition(number=-6) NamedPitch('e,') StaffPosition(number=-5) NamedPitch('f,') StaffPosition(number=-4) NamedPitch('g,') StaffPosition(number=-3) NamedPitch('a,') StaffPosition(number=-2) NamedPitch('b,') StaffPosition(number=-1) NamedPitch('c') StaffPosition(number=0) NamedPitch('d') StaffPosition(number=1) NamedPitch('e') StaffPosition(number=2) NamedPitch('f') StaffPosition(number=3) NamedPitch('g') StaffPosition(number=4) NamedPitch('a') StaffPosition(number=5) NamedPitch('b') .. container:: example Alto clef: >>> clef = abjad.Clef("alto") >>> for n in range(-6, 6): ... staff_position = abjad.StaffPosition(n) ... pitch = clef.to_pitch(staff_position) ... print(f"{staff_position!r:25}{pitch!r}") ... StaffPosition(number=-6) NamedPitch('d') StaffPosition(number=-5) NamedPitch('e') StaffPosition(number=-4) NamedPitch('f') StaffPosition(number=-3) NamedPitch('g') StaffPosition(number=-2) NamedPitch('a') StaffPosition(number=-1) NamedPitch('b') StaffPosition(number=0) NamedPitch("c'") StaffPosition(number=1) NamedPitch("d'") StaffPosition(number=2) NamedPitch("e'") StaffPosition(number=3) NamedPitch("f'") StaffPosition(number=4) NamedPitch("g'") StaffPosition(number=5) NamedPitch("a'") .. container:: example Percussion clef: >>> clef = abjad.Clef("percussion") >>> for n in range(-6, 6): ... staff_position = abjad.StaffPosition(n) ... pitch = clef.to_pitch(staff_position) ... print(f"{staff_position!r:25}{pitch!r}") ... StaffPosition(number=-6) NamedPitch('d') StaffPosition(number=-5) NamedPitch('e') StaffPosition(number=-4) NamedPitch('f') StaffPosition(number=-3) NamedPitch('g') StaffPosition(number=-2) NamedPitch('a') StaffPosition(number=-1) NamedPitch('b') StaffPosition(number=0) NamedPitch("c'") StaffPosition(number=1) NamedPitch("d'") StaffPosition(number=2) NamedPitch("e'") StaffPosition(number=3) NamedPitch("f'") StaffPosition(number=4) NamedPitch("g'") StaffPosition(number=5) NamedPitch("a'") """ assert isinstance(staff_position, _pitch.StaffPosition), repr(staff_position) offset_staff_position_number = staff_position.number offset_staff_position_number -= self.middle_c_position().number offset_staff_position = _pitch.StaffPosition(offset_staff_position_number) octave_number = offset_staff_position.number // 7 + 4 diatonic_pc_number = offset_staff_position.number % 7 pitch_class_number = _pitch._diatonic_pc_number_to_pitch_class_number[ diatonic_pc_number ] pitch_number = 12 * (octave_number - 4) pitch_number += pitch_class_number named_pitch = _pitch.NamedPitch(pitch_number) return named_pitch
[docs] def to_staff_position(self, pitch) -> _pitch.StaffPosition: r""" Changes ``pitch`` to staff position. .. container:: example Changes C#5 to absolute staff position: >>> pitch = abjad.NamedPitch("C#5") >>> abjad.Clef("alto").to_staff_position(pitch) StaffPosition(number=7) >>> abjad.Clef("treble").to_staff_position(pitch) StaffPosition(number=1) >>> abjad.Clef("bass").to_staff_position(pitch) StaffPosition(number=13) .. container:: example Labels absolute staff position: >>> string = "g16 a b c' d' e' f' g' a' b' c'' d'' e'' f'' g'' a''" >>> staff = abjad.Staff(string) >>> clef = abjad.Clef("alto") >>> for note in staff: ... staff_position = clef.to_staff_position(note.written_pitch) ... markup = abjad.Markup(rf"\markup {staff_position.number}") ... abjad.attach(markup, note) ... >>> abjad.override(staff).TextScript.staff_padding = 5 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 5 } { g16 - \markup -3 a16 - \markup -2 b16 - \markup -1 c'16 - \markup 0 d'16 - \markup 1 e'16 - \markup 2 f'16 - \markup 3 g'16 - \markup 4 a'16 - \markup 5 b'16 - \markup 6 c''16 - \markup 7 d''16 - \markup 8 e''16 - \markup 9 f''16 - \markup 10 g''16 - \markup 11 a''16 - \markup 12 } .. container:: example Labels staff position in bass clef: >>> string = "g,16 a, b, c d e f g a b c' d' e' f' g' a'" >>> staff = abjad.Staff(string) >>> clef = abjad.Clef("bass") >>> for note in staff: ... staff_position = clef.to_staff_position(note.written_pitch) ... markup = abjad.Markup(rf"\markup {staff_position.number}") ... abjad.attach(markup, note) ... >>> abjad.attach(abjad.Clef("bass"), staff[0]) >>> abjad.override(staff).TextScript.staff_padding = 5 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 5 } { \clef "bass" g,16 - \markup -4 a,16 - \markup -3 b,16 - \markup -2 c16 - \markup -1 d16 - \markup 0 e16 - \markup 1 f16 - \markup 2 g16 - \markup 3 a16 - \markup 4 b16 - \markup 5 c'16 - \markup 6 d'16 - \markup 7 e'16 - \markup 8 f'16 - \markup 9 g'16 - \markup 10 a'16 - \markup 11 } """ assert isinstance(pitch, _pitch.NamedPitch) staff_position_number = pitch._get_diatonic_pitch_number() staff_position_number += self.middle_c_position().number staff_position = _pitch.StaffPosition(staff_position_number) return staff_position
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class ColorFingering: r""" Color fingering. .. container:: example >>> fingering = abjad.ColorFingering(1) >>> note = abjad.Note("c'4") >>> abjad.attach(fingering, note, direction=abjad.UP) >>> abjad.show(note) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(note) >>> print(string) c'4 ^ \markup { \override #'(circle-padding . 0.25) \circle \finger 1 } .. container:: example Tweaks: >>> staff = abjad.Staff("c'4 d' e' f'") >>> fingering = abjad.ColorFingering(1) >>> bundle = abjad.bundle(fingering, r"- \tweak color #blue") >>> abjad.attach(bundle, staff[0], direction=abjad.UP) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4 - \tweak color #blue ^ \markup { \override #'(circle-padding . 0.25) \circle \finger 1 } d'4 e'4 f'4 } Color fingerings indicate alternate woodwind fingerings by amount of pitch of timbre deviation. """ number: int directed: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after"
[docs] def __post_init__(self): assert isinstance(self.number, int), repr(self.number)
def _get_lilypond_format(self): return self.markup._get_lilypond_format() def _get_contributions(self, *, component=None, wrapper=None): return self.markup._get_contributions(component=component, wrapper=wrapper) @property def markup(self) -> typing.Optional["Markup"]: r""" Gets markup of color fingering. .. container:: example First color fingering: >>> fingering = abjad.ColorFingering(1) >>> string = abjad.lilypond(fingering.markup) >>> print(string) \markup { \override #'(circle-padding . 0.25) \circle \finger 1 } """ if self.number is None: return None string = rf"\override #'(circle-padding . 0.25) \circle \finger {self.number}" string = rf"\markup {{ {string} }}" markup = Markup(string) return markup
[docs] @dataclasses.dataclass(eq=False, frozen=True, slots=True, unsafe_hash=True) class Dynamic: r""" Dynamic. .. container:: example Initializes from dynamic name: >>> voice = abjad.Voice("c'8 d'8 e'8 f'8") >>> dynamic = abjad.Dynamic("f") >>> abjad.attach(dynamic, voice[0]) .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'8 \f d'8 e'8 f'8 } >>> abjad.show(voice) # doctest: +SKIP .. container:: example Simultaneous dynamics in a single staff: >>> voice_1 = abjad.Voice("e'8 g'8 f'8 a'8") >>> abjad.attach(abjad.Dynamic('f'), voice_1[0], context='Voice') >>> command = abjad.VoiceNumber(1) >>> abjad.attach(command, voice_1[0]) >>> abjad.override(voice_1).DynamicLineSpanner.direction = abjad.UP >>> voice_2 = abjad.Voice("c'2") >>> command = abjad.VoiceNumber(2) >>> abjad.attach(command, voice_2[0]) >>> abjad.attach(abjad.Dynamic("mf"), voice_2[0], context="Voice") >>> staff = abjad.Staff([voice_1, voice_2], simultaneous=True) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff << \new Voice \with { \override DynamicLineSpanner.direction = #up } { \voiceOne e'8 \f g'8 f'8 a'8 } \new Voice { \voiceTwo c'2 \mf } >> >>> for leaf in abjad.select.leaves(staff): ... dynamic = abjad.get.effective(leaf, abjad.Dynamic) ... print(f"{leaf!r}:") ... print(f" {dynamic!r}") Note("e'8"): Dynamic(name='f', command=None, hide=False, leak=False, name_is_textual=False, ordinal=None) Note("g'8"): Dynamic(name='f', command=None, hide=False, leak=False, name_is_textual=False, ordinal=None) Note("f'8"): Dynamic(name='f', command=None, hide=False, leak=False, name_is_textual=False, ordinal=None) Note("a'8"): Dynamic(name='f', command=None, hide=False, leak=False, name_is_textual=False, ordinal=None) Note("c'2"): Dynamic(name='mf', command=None, hide=False, leak=False, name_is_textual=False, ordinal=None) .. container:: example exception Errors on nondynamic input: >>> abjad.Dynamic("text") Traceback (most recent call last): ... Exception: letter 't' (in 'text') is not a dynamic. .. container:: example Tweaks: >>> voice = abjad.Voice("c'4") >>> dynamic = abjad.Dynamic("f") >>> bundle = abjad.bundle(dynamic, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak color #blue \f } .. container:: example Use ``command`` like this: >>> abjad.Dynamic("f", command=r"\sub_f").command '\\sub_f' Use to override LilyPond output when a custom dynamic has been defined in an external stylesheet. (In the example above, ``\sub_f`` is a nonstandard LilyPond dynamic. LilyPond will interpret the output above only when the command ``\sub_f`` is defined somewhere in an external stylesheet.) .. container:: example Direction: With ``direction`` unset: >>> voice = abjad.Voice("c'2 c''2") >>> abjad.attach(abjad.Dynamic("p"), voice[0]) >>> abjad.attach(abjad.Dynamic("f"), voice[1]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'2 \p c''2 \f } With ``direction=abjad.UP``: >>> voice = abjad.Voice("c'2 c''2") >>> abjad.attach(abjad.Dynamic("p"), voice[0], direction=abjad.UP) >>> abjad.attach(abjad.Dynamic("f"), voice[1], direction=abjad.UP) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'2 ^ \p c''2 ^ \f } With ``direction=abjad.DOWN``: >>> voice = abjad.Voice("c'2 c''2") >>> abjad.attach(abjad.Dynamic("p"), voice[0], direction=abjad.DOWN) >>> abjad.attach(abjad.Dynamic("f"), voice[1], direction=abjad.DOWN) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'2 _ \p c''2 _ \f } .. container:: example REGRESSION. Effort dynamics default to down: >>> voice = abjad.Voice("c'2 c''2") >>> abjad.attach(abjad.Dynamic('"p"'), voice[0]) >>> abjad.attach(abjad.Dynamic('"f"'), voice[1]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'2 _ #(make-dynamic-script (markup #:whiteout #:line ( #:general-align Y -2 #:normal-text #:larger "“" #:hspace -0.1 #:dynamic "p" #:hspace -0.25 #:general-align Y -2 #:normal-text #:larger "”" ) ) ) c''2 _ #(make-dynamic-script (markup #:whiteout #:line ( #:general-align Y -2 #:normal-text #:larger "“" #:hspace -0.4 #:dynamic "f" #:hspace -0.2 #:general-align Y -2 #:normal-text #:larger "”" ) ) ) } And may be overriden: >>> voice = abjad.Voice("c'2 c''2") >>> abjad.attach(abjad.Dynamic('"p"'), voice[0], direction=abjad.UP) >>> abjad.attach(abjad.Dynamic('"f"'), voice[1], direction=abjad.UP) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'2 ^ #(make-dynamic-script (markup #:whiteout #:line ( #:general-align Y -2 #:normal-text #:larger "“" #:hspace -0.1 #:dynamic "p" #:hspace -0.25 #:general-align Y -2 #:normal-text #:larger "”" ) ) ) c''2 ^ #(make-dynamic-script (markup #:whiteout #:line ( #:general-align Y -2 #:normal-text #:larger "“" #:hspace -0.4 #:dynamic "f" #:hspace -0.2 #:general-align Y -2 #:normal-text #:larger "”" ) ) ) } .. container:: example Set ``hide=True`` when dynamic should not appear in output (but should still determine effective dynamic): >>> voice = abjad.Voice("c'4 d' e' f'") >>> abjad.attach(abjad.Dynamic("f"), voice[0]) >>> abjad.attach(abjad.Dynamic("mf", hide=True), voice[2]) >>> abjad.show(voice) # doctest: +SKIP >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 \f d'4 e'4 f'4 } >>> for leaf in abjad.iterate.leaves(voice): ... dynamic = abjad.get.effective(leaf, abjad.Dynamic) ... print(f"{leaf!r}:") ... print(f" {dynamic!r}") Note("c'4"): Dynamic(name='f', command=None, hide=False, leak=False, name_is_textual=False, ordinal=None) Note("d'4"): Dynamic(name='f', command=None, hide=False, leak=False, name_is_textual=False, ordinal=None) Note("e'4"): Dynamic(name='mf', command=None, hide=True, leak=False, name_is_textual=False, ordinal=None) Note("f'4"): Dynamic(name='mf', command=None, hide=True, leak=False, name_is_textual=False, ordinal=None) .. container:: example Set ``leak=True`` Is true to format LilyPond empty chord ``<>`` symbol: Without leaked stop dynamic: >>> voice = abjad.Voice("c'4 d' e' r") >>> start_dynamic = abjad.Dynamic("mf") >>> start_hairpin = abjad.StartHairpin(">") >>> stop_dynamic = abjad.Dynamic("pp") >>> abjad.attach(start_dynamic, voice[0]) >>> abjad.attach(start_hairpin, voice[0]) >>> abjad.attach(stop_dynamic, voice[-2]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4 } { c'4 \mf \> d'4 e'4 \pp r4 } With leaked stop dynamic: >>> voice = abjad.Voice("c'4 d' e' r") >>> start_dynamic = abjad.Dynamic("mf") >>> start_hairpin = abjad.StartHairpin(">") >>> stop_dynamic = abjad.Dynamic("pp", leak=True) >>> abjad.attach(start_dynamic, voice[0]) >>> abjad.attach(start_hairpin, voice[0]) >>> abjad.attach(stop_dynamic, voice[-2]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4 } { c'4 \mf \> d'4 e'4 <> \pp r4 } .. container:: example Leaks format after spanners: >>> voice = abjad.Voice("c'8 [ d' e' ] f'") >>> start_dynamic = abjad.Dynamic("mf") >>> start_hairpin = abjad.StartHairpin(">") >>> stop_dynamic = abjad.Dynamic("pp", leak=True) >>> abjad.attach(start_dynamic, voice[0]) >>> abjad.attach(start_hairpin, voice[0]) >>> abjad.attach(stop_dynamic, voice[-2]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4 } { c'8 \mf [ \> d'8 e'8 ] <> \pp f'8 } .. container:: example Leaked and nonleaked dynamic may be attached to the same leaf: >>> voice = abjad.Voice("c'4 d' e' f'") >>> abjad.attach(abjad.Dynamic("f"), voice[0]) >>> abjad.attach(abjad.Dynamic("p", leak=True), voice[0]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 \f <> \p d'4 e'4 f'4 } .. container:: example Leaks and tweaks on the same dynamic format correctly; LilyPond empty chord ``<>`` symbol appears before postevents: >>> voice = abjad.Voice("r4 d' e' f'") >>> dynamic = abjad.Dynamic("f", leak=True) >>> bundle = abjad.bundle(dynamic, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> abjad.show(voice) # doctest: +SKIP >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { r4 <> - \tweak color #blue \f d'4 e'4 f'4 } .. container:: example Leak survives copy: >>> import copy >>> dynamic = abjad.Dynamic("pp", leak=True) >>> copy.copy(dynamic) Dynamic(name='pp', command=None, hide=False, leak=True, name_is_textual=False, ordinal=None) .. container:: example Name-is-textual: >>> abjad.Dynamic("f").name_is_textual False >>> dynamic = abjad.Dynamic("appena udibile", name_is_textual=True) >>> dynamic.name_is_textual True Textual dynamics format like this when initialized without an explicit command: >>> voice = abjad.Voice("c'4 d' e' f'") >>> dynamic = abjad.Dynamic("appena udibile", name_is_textual=True) >>> abjad.attach(dynamic, voice[0]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4 >>> abjad.override(voice).DynamicText.X_extent = "#'(0 . 0)" >>> abjad.override(voice).DynamicText.self_alignment_X = abjad.LEFT >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4 \override DynamicText.X-extent = #'(0 . 0) \override DynamicText.self-alignment-X = #left } { c'4 _ #(make-dynamic-script (markup #:whiteout #:normal-text #:italic "appena udibile")) d'4 e'4 f'4 } Textual dynamics format like this when initialized with an explicit command: >>> voice = abjad.Voice("c'4 d' e' f'") >>> dynamic = abjad.Dynamic( ... "appena udibile", ... command=r"\appena_udibile", ... name_is_textual=True, ... ) >>> abjad.attach(dynamic, voice[0]) Only LilyPond output is shown here because dynamic commands (like ``\appena_udibile`` shown here) are meant to be user-defined (and not included in Abjad): >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 \appena_udibile d'4 e'4 f'4 } REGRESSION. Textual names work with replace: >>> import dataclasses >>> dynamic = abjad.Dynamic("appena udibile", name_is_textual=True) >>> dataclasses.replace(dynamic) Dynamic(name='appena udibile', command=None, hide=False, leak=False, name_is_textual=True, ordinal=None) """ name: "str" = "f" command: str | None = None hide: bool = dataclasses.field(compare=False, default=False) leak: bool = dataclasses.field(compare=False, default=False) name_is_textual: bool = False ordinal: int | _math.Infinity | _math.NegativeInfinity | None = None context: typing.ClassVar[str] = "Voice" directed: typing.ClassVar[bool] = True # find_context_on_attach: typing.ClassVar[bool] = True parameter: typing.ClassVar[str] = "DYNAMIC" persistent: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True
[docs] def __post_init__(self): if self.name is not None: assert isinstance(self.name, str), repr(self.name) if not self.name_is_textual: for letter in self.name.strip('"'): if letter not in self._lilypond_dynamic_alphabet: message = f"letter {letter!r} (in {self.name!r}) is not a dynamic." raise Exception(message) if self.command is not None: assert isinstance(self.command, str), repr(self.command) assert self.command.startswith("\\"), repr(self.command) assert isinstance(self.hide, bool), repr(self.hide) assert isinstance(self.leak, bool), repr(self.leak) assert isinstance(self.name_is_textual, bool), repr(self.name_is_textual) if self.ordinal is not None: assert isinstance( self.ordinal, int | _math.Infinity | _math.NegativeInfinity )
_composite_dynamic_name_to_steady_state_dynamic_name: typing.ClassVar = { "fp": "p", "sf": "f", "sff": "ff", "sfp": "p", "sfpp": "pp", "sffp": "p", "sffpp": "pp", "sfz": "f", "sp": "p", "spp": "pp", "rfz": "f", } _dynamic_name_to_dynamic_ordinal: typing.ClassVar = { "ppppp": -6, "pppp": -5, "ppp": -4, "pp": -3, "p": -2, "mp": -1, "mf": 1, "f": 2, "ff": 3, "fff": 4, "ffff": 5, "fffff": 6, } _dynamic_names: typing.ClassVar = ( "ppppp", "pppp", "ppp", "pp", "p", "mp", "mf", "f", "ff", "fff", "ffff", "fffff", "fp", "sf", "sff", "sp", "spp", "sfz", "sffz", "sfffz", "sffp", "sffpp", "sfp", "sfpp", "rfz", ) _dynamic_ordinal_to_dynamic_name: typing.ClassVar = { -6: "ppppp", -5: "pppp", -4: "ppp", -3: "pp", -2: "p", -1: "mp", 1: "mf", 2: "f", 3: "ff", 4: "fff", 5: "ffff", 6: "fffff", } _lilypond_dynamic_commands: typing.ClassVar = [_ for _ in _dynamic_names] _lilypond_dynamic_alphabet: typing.ClassVar = "fmprsz" _to_width: typing.ClassVar = { '"f"': 2, '"mf"': 3.5, '"mp"': 3.5, '"p"': 2, "sfz": 2.5, } site: typing.ClassVar[str] = "after"
[docs] def __eq__(self, argument) -> bool: """ Is true when ``argument`` equals dynamic. .. container:: example >>> dynamic_1 = abjad.Dynamic("p") >>> dynamic_2 = abjad.Dynamic("p") >>> dynamic_3 = abjad.Dynamic("f") >>> dynamic_1 == dynamic_1 True >>> dynamic_1 == dynamic_2 True >>> dynamic_1 == dynamic_3 False >>> dynamic_2 == dynamic_1 True >>> dynamic_2 == dynamic_2 True >>> dynamic_2 == dynamic_3 False >>> dynamic_3 == dynamic_1 False >>> dynamic_3 == dynamic_2 False >>> dynamic_3 == dynamic_3 True """ if not isinstance(argument, type(self)): return False if self.name == argument.name and self.ordinal == argument.ordinal: return True return False
def _attachment_test_all(self, component_expression): if not hasattr(component_expression, "written_duration"): strings = [f"Must be leaf (not {component_expression})."] return strings return True def _format_effort_dynamic(self, *, wrapper=None): name = self.name.strip('"') before = {"f": -0.4, "m": -0.1, "p": -0.1, "r": -0.1, "s": -0.3, "z": -0.2}[ name[0] ] after = {"f": -0.2, "m": -0.1, "p": -0.25, "r": 0, "s": 0, "z": -0.2}[name[-1]] # direction = self.direction direction = wrapper.direction or _enums.DOWN direction = _string.to_tridirectional_lilypond_symbol(direction) strings = [] strings.append(f"{direction} #(make-dynamic-script") strings.append(" (markup") strings.append(" #:whiteout") strings.append(" #:line (") strings.append(' #:general-align Y -2 #:normal-text #:larger "“"') strings.append(f" #:hspace {before}") strings.append(f' #:dynamic "{name}"') strings.append(f" #:hspace {after}") strings.append(' #:general-align Y -2 #:normal-text #:larger "”"') strings.append(" )") strings.append(" )") strings.append(" )") string = "\n".join(strings) return string @staticmethod def _format_textual(string, *, wrapper=None): if wrapper.direction is None: direction = _enums.DOWN direction = _string.to_tridirectional_lilypond_symbol(direction) assert isinstance(string, str), repr(string) string = f'(markup #:whiteout #:normal-text #:italic "{string}")' string = f"{direction} #(make-dynamic-script {string})" return string def _get_lilypond_format(self, *, wrapper=None): if self.command: string = self.command elif self.effort: string = self._format_effort_dynamic(wrapper=wrapper) elif self.name_is_textual: string = self._format_textual(self.name, wrapper=wrapper) else: string = rf"\{self.name}" if wrapper.direction is not None: direction_ = wrapper.direction direction = _string.to_tridirectional_lilypond_symbol(direction_) string = f"{direction} {string}" return string def _get_contributions(self, *, component=None, wrapper=None): contributions = _contributions.ContributionsBySite() command = self._get_lilypond_format(wrapper=wrapper) if self.leak: contributions.after.leak.append(_EMPTY_CHORD) if not self.hide: contributions.after.leaks.append(command) else: if not self.hide: contributions.after.articulations.append(command) return contributions # TODO: make markup function and shorten docstring @property def effort(self) -> bool: r""" Is true when double quotes enclose dynamic. .. container:: example >>> voice = abjad.Voice("c'4 r d' r e' r f' r") >>> abjad.attach(abjad.Dynamic('"pp"'), voice[0]) >>> abjad.attach(abjad.Dynamic('"mp"'), voice[2]) >>> abjad.attach(abjad.Dynamic('"mf"'), voice[4]) >>> abjad.attach(abjad.Dynamic('"ff"'), voice[6]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4 } { c'4 _ #(make-dynamic-script (markup #:whiteout #:line ( #:general-align Y -2 #:normal-text #:larger "“" #:hspace -0.1 #:dynamic "pp" #:hspace -0.25 #:general-align Y -2 #:normal-text #:larger "”" ) ) ) r4 d'4 _ #(make-dynamic-script (markup #:whiteout #:line ( #:general-align Y -2 #:normal-text #:larger "“" #:hspace -0.1 #:dynamic "mp" #:hspace -0.25 #:general-align Y -2 #:normal-text #:larger "”" ) ) ) r4 e'4 _ #(make-dynamic-script (markup #:whiteout #:line ( #:general-align Y -2 #:normal-text #:larger "“" #:hspace -0.1 #:dynamic "mf" #:hspace -0.2 #:general-align Y -2 #:normal-text #:larger "”" ) ) ) r4 f'4 _ #(make-dynamic-script (markup #:whiteout #:line ( #:general-align Y -2 #:normal-text #:larger "“" #:hspace -0.4 #:dynamic "ff" #:hspace -0.2 #:general-align Y -2 #:normal-text #:larger "”" ) ) ) r4 } .. container:: example >>> voice = abjad.Voice("c'4 r d' r e' r f' r") >>> abjad.attach(abjad.Dynamic('"sf"'), voice[0]) >>> abjad.attach(abjad.Dynamic('"sfz"'), voice[2]) >>> abjad.attach(abjad.Dynamic('"rf"'), voice[4]) >>> abjad.attach(abjad.Dynamic('"rfz"'), voice[6]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4 } { c'4 _ #(make-dynamic-script (markup #:whiteout #:line ( #:general-align Y -2 #:normal-text #:larger "“" #:hspace -0.3 #:dynamic "sf" #:hspace -0.2 #:general-align Y -2 #:normal-text #:larger "”" ) ) ) r4 d'4 _ #(make-dynamic-script (markup #:whiteout #:line ( #:general-align Y -2 #:normal-text #:larger "“" #:hspace -0.3 #:dynamic "sfz" #:hspace -0.2 #:general-align Y -2 #:normal-text #:larger "”" ) ) ) r4 e'4 _ #(make-dynamic-script (markup #:whiteout #:line ( #:general-align Y -2 #:normal-text #:larger "“" #:hspace -0.1 #:dynamic "rf" #:hspace -0.2 #:general-align Y -2 #:normal-text #:larger "”" ) ) ) r4 f'4 _ #(make-dynamic-script (markup #:whiteout #:line ( #:general-align Y -2 #:normal-text #:larger "“" #:hspace -0.1 #:dynamic "rfz" #:hspace -0.2 #:general-align Y -2 #:normal-text #:larger "”" ) ) ) r4 } """ assert isinstance(self.name, str), repr(self.name) return bool(self.name) and self.name[0] == '"' @property def sforzando(self) -> bool: """ Is true when dynamic name begins in s- and ends in -z. .. container:: example >>> abjad.Dynamic("f").sforzando False >>> abjad.Dynamic("sfz").sforzando True >>> abjad.Dynamic("sffz").sforzando True >>> abjad.Dynamic("sfp").sforzando False >>> abjad.Dynamic("sf").sforzando False >>> abjad.Dynamic("rfz").sforzando False """ assert isinstance(self.name, str), repr(self.name) if self.name and self.name.startswith("s") and self.name.endswith("z"): return True return False
[docs] @staticmethod def composite_dynamic_name_to_steady_state_dynamic_name(name) -> str: """ Changes composite ``name`` to steady state dynamic name. .. container:: example >>> abjad.Dynamic.composite_dynamic_name_to_steady_state_dynamic_name("sfp") 'p' >>> abjad.Dynamic.composite_dynamic_name_to_steady_state_dynamic_name("rfz") 'f' """ return Dynamic._composite_dynamic_name_to_steady_state_dynamic_name[name]
[docs] @staticmethod def dynamic_name_to_dynamic_ordinal(name): """ Changes ``name`` to dynamic ordinal. .. container:: example >>> abjad.Dynamic.dynamic_name_to_dynamic_ordinal("fff") 4 """ try: return Dynamic._dynamic_name_to_dynamic_ordinal[name] except KeyError: name = Dynamic.composite_dynamic_name_to_steady_state_dynamic_name(name) return Dynamic._dynamic_name_to_dynamic_ordinal[name]
[docs] @staticmethod def dynamic_ordinal_to_dynamic_name(dynamic_ordinal) -> str: """ Changes ``dynamic_ordinal`` to dynamic name. .. container:: example >>> abjad.Dynamic.dynamic_ordinal_to_dynamic_name(-5) 'pppp' """ return Dynamic._dynamic_ordinal_to_dynamic_name[dynamic_ordinal]
[docs] @staticmethod def is_dynamic_name(argument) -> bool: """ Is true when ``argument`` is dynamic name. .. container:: example >>> abjad.Dynamic.is_dynamic_name("f") True >>> abjad.Dynamic.is_dynamic_name("sfz") True """ return argument in Dynamic._dynamic_names
[docs] def get_ordinal(self) -> int | _math.Infinity | _math.NegativeInfinity | None: """ Gets ordinal. .. container:: example >>> abjad.Dynamic("f").get_ordinal() 2 >>> abjad.Dynamic("p").get_ordinal() -2 >>> abjad.Dynamic('"f"').get_ordinal() 2 >>> abjad.Dynamic('"p"').get_ordinal() -2 User-defined ordinals: >>> barely_audible = abjad.Dynamic( ... "barely audible", ... name_is_textual=True, ... ordinal=-99, ... ) >>> barely_audible.get_ordinal() -99 >>> extremely_loud = abjad.Dynamic( ... "extremely loud", ... name_is_textual=True, ... ordinal=99, ... ) >>> extremely_loud.get_ordinal() 99 REGRESSION. Textual names without explicit ordinal return none: >>> dynamic = abjad.Dynamic("appena udibile", name_is_textual=True) >>> dynamic.get_ordinal() is None True """ if self.ordinal is not None: return self.ordinal stripped_name = None if self.name: stripped_name = self.name.strip('"') if stripped_name in self._composite_dynamic_name_to_steady_state_dynamic_name: stripped_name = self._composite_dynamic_name_to_steady_state_dynamic_name[ stripped_name ] if stripped_name is None: ordinal = None else: assert isinstance(stripped_name, str), repr(stripped_name) ordinal_ = self._dynamic_name_to_dynamic_ordinal.get(stripped_name) prototype = (int, _math.Infinity, _math.NegativeInfinity, type(None)) assert isinstance(ordinal_, prototype), repr(ordinal_) ordinal = ordinal_ return ordinal
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class Fermata: r""" Fermata. .. container:: example A short fermata: >>> score = abjad.Score([abjad.Staff([abjad.Note("c'4")])]) >>> fermata = abjad.Fermata(command='shortfermata') >>> abjad.attach(fermata, score[0][0]) >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { c'4 \shortfermata } >> .. container:: example A fermata: >>> score = abjad.Score([abjad.Staff([abjad.Note("c'4")])]) >>> fermata = abjad.Fermata() >>> abjad.attach(fermata, score[0][0]) >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { c'4 \fermata } >> .. container:: example A long fermata: >>> score = abjad.Score([abjad.Staff([abjad.Note("c'4")])]) >>> fermata = abjad.Fermata('longfermata') >>> abjad.attach(fermata, score[0][0]) >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { c'4 \longfermata } >> .. container:: example A very long fermata: >>> score = abjad.Score([abjad.Staff([abjad.Note("c'4")])]) >>> fermata = abjad.Fermata('verylongfermata') >>> abjad.attach(fermata, score[0][0]) >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { c'4 \verylongfermata } >> .. container:: example Tweaks: >>> note = abjad.Note("c'4") >>> staff = abjad.Staff([note], name="Staff") >>> score = abjad.Score([staff], name="Score") >>> fermata = abjad.Fermata() >>> bundle = abjad.bundle(fermata, r"- \tweak color #blue") >>> abjad.attach(bundle, note) >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(note) >>> print(string) c'4 - \tweak color #blue \fermata """ command: str = "fermata" allowable_commands: typing.ClassVar = ( "fermata", "longfermata", "shortfermata", "verylongfermata", ) context: typing.ClassVar[str] = "Score" # find_context_on_attach: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after"
[docs] def __post_init__(self): assert self.command in self.allowable_commands, repr(self.command)
def _get_lilypond_format(self): return rf"\{self.command}" def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._get_lilypond_format() site.articulations.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class Glissando: r""" LilyPond ``\glissando`` command. .. container:: example >>> voice = abjad.Voice("c'4 d' e' f'", name="Voice") >>> glissando = abjad.Glissando() >>> bundle = abjad.bundle(glissando, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \context Voice = "Voice" { c'4 - \tweak color #blue \glissando d'4 e'4 f'4 } """ zero_padding: bool = False context: typing.ClassVar[str] = "Voice" # find_context_on_attach: typing.ClassVar[bool] = True persistent: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after"
[docs] def __post_init__(self): assert isinstance(self.zero_padding, bool), repr(self.zero_padding)
def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) strings = [] if self.zero_padding: strings.append(r"- \abjad-zero-padding-glissando") strings.append(r"\glissando") site.spanner_starts.extend(strings) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class InstrumentName: r""" LilyPond ``\instrumentName`` setting. .. container:: example Set instrument name to markup like this: >>> staff = abjad.Staff("c'4 d'4 e'4 f'4") >>> markup = abjad.Markup(r"\markup Cellos") >>> instrument_name = abjad.InstrumentName(markup) >>> abjad.attach(instrument_name, staff[0]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \set Staff.instrumentName = \markup Cellos c'4 d'4 e'4 f'4 } Set instrument name to custom-defined function like this: >>> staff = abjad.Staff("c'4 d'4 e'4 f'4") >>> instrument_name = abjad.InstrumentName(r"\my_instrument_name") >>> abjad.attach(instrument_name, staff[0]) >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \set Staff.instrumentName = \my_instrument_name c'4 d'4 e'4 f'4 } .. container:: example Equality testing: >>> name_1 = abjad.InstrumentName(r"\markup Harp", context="PianoStaff") >>> name_2 = abjad.InstrumentName(r"\markup Harp", context="PianoStaff") >>> name_3 = abjad.InstrumentName(r"\markup Harp", context="Staff") >>> name_1 == name_1 True >>> name_1 == name_2 True >>> name_1 == name_3 False >>> name_2 == name_1 True >>> name_2 == name_2 True >>> name_2 == name_3 False >>> name_3 == name_1 False >>> name_3 == name_2 False >>> name_3 == name_3 True """ markup: typing.Union[str, "Markup"] _: dataclasses.KW_ONLY context: str = "Staff" # find_context_on_attach: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "before"
[docs] def __post_init__(self): assert isinstance(self.markup, str | Markup), repr(self.markup) assert isinstance(self.context, str), repr(self.context)
@property def _lilypond_type(self): if isinstance(self.context, type): return self.context.__name__ elif isinstance(self.context, str): return self.context else: return type(self.context).__name__ def _get_lilypond_format(self, context=None): result = [] if isinstance(context, str): pass elif context is not None: context = context.lilypond_type else: context = self._lilypond_type if isinstance(self.markup, Markup): string = self.markup.string else: assert isinstance(self.markup, str) string = self.markup string = rf"\set {context!s}.instrumentName = {string}" result.append(string) return result def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) strings = self._get_lilypond_format() site.commands.extend(strings) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class KeyCluster: r""" Key cluster. .. container:: example Includes flat markup: >>> chord = abjad.Chord("<c' e' g' b' d'' f''>8") >>> key_cluster = abjad.KeyCluster(include_flat_markup=True) >>> abjad.attach(key_cluster, chord, direction=abjad.UP) >>> staff = abjad.Staff([chord]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \once \override Accidental.stencil = ##f \once \override AccidentalCautionary.stencil = ##f \once \override Arpeggio.X-offset = #-2 \once \override NoteHead.stencil = #ly:text-interface::print \once \override NoteHead.text = \markup \filled-box #'(-0.6 . 0.6) #'(-0.7 . 0.7) #0.25 <c' e' g' b' d'' f''>8 ^ \markup \center-column { \natural \flat } } .. container:: example Includes natural markup: >>> chord = abjad.Chord("<c' e' g' b' d'' f''>8") >>> key_cluster = abjad.KeyCluster(include_natural_markup=True) >>> abjad.attach(key_cluster, chord, direction=abjad.UP) >>> staff = abjad.Staff([chord]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \once \override Accidental.stencil = ##f \once \override AccidentalCautionary.stencil = ##f \once \override Arpeggio.X-offset = #-2 \once \override NoteHead.stencil = #ly:text-interface::print \once \override NoteHead.text = \markup \filled-box #'(-0.6 . 0.6) #'(-0.7 . 0.7) #0.25 <c' e' g' b' d'' f''>8 ^ \markup \center-column { \natural \flat } } .. container:: example Key cluster output includes overrides instead of tweaks: >>> chord = abjad.Chord("<c' e' g' b' d'' f''>8") >>> key_cluster = abjad.KeyCluster() >>> abjad.attach(key_cluster, chord, direction=abjad.UP) >>> string = abjad.lilypond(chord) >>> print(string) \once \override Accidental.stencil = ##f \once \override AccidentalCautionary.stencil = ##f \once \override Arpeggio.X-offset = #-2 \once \override NoteHead.stencil = #ly:text-interface::print \once \override NoteHead.text = \markup \filled-box #'(-0.6 . 0.6) #'(-0.7 . 0.7) #0.25 <c' e' g' b' d'' f''>8 ^ \markup \center-column { \natural \flat } The reason for this is that chords contain multiple note-heads: if key cluster formatted tweaks instead of overrides, the five format commands shown above would need to be duplicated immediately before each note-head. """ include_flat_markup: bool = True include_natural_markup: bool = True directed: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after"
[docs] def __post_init__(self): assert isinstance(self.include_flat_markup, bool), repr( self.include_flat_markup ) assert isinstance(self.include_natural_markup, bool), repr( self.include_natural_markup )
def _get_contributions(self, wrapper=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) contributions.grob_overrides.append( "\\once \\override Accidental.stencil = ##f\n" "\\once \\override AccidentalCautionary.stencil = ##f\n" "\\once \\override Arpeggio.X-offset = #-2\n" "\\once \\override NoteHead.stencil = #ly:text-interface::print\n" "\\once \\override NoteHead.text =\n" "\\markup \\filled-box #'(-0.6 . 0.6) #'(-0.7 . 0.7) #0.25" ) if self.include_flat_markup and self.include_natural_markup: string = r"\center-column { \natural \flat }" elif self.include_flat_markup: string = r"\center-column \flat" else: string = r"\center-column \natural" string = rf"\markup {string}" if wrapper.direction is _enums.UP: string = rf"^ {string}" elif wrapper.direction is _enums.DOWN: string = rf"_ {string}" site.markup.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class LilyPondLiteral: r""" LilyPond literal. .. container:: example Dotted slur: >>> voice = abjad.Voice("c'8 d'8 e'8 f'8") >>> abjad.slur(voice[:]) >>> literal = abjad.LilyPondLiteral(r"\slurDotted", site="before") >>> abjad.attach(literal, voice[0]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { \slurDotted c'8 ( d'8 e'8 f'8 ) } .. container:: example Use the absolute before and absolute after format sites like this: >>> voice = abjad.Voice("c'8 d'8 e'8 f'8") >>> abjad.slur(voice[:]) >>> literal = abjad.LilyPondLiteral(r"\slurDotted", site="before") >>> abjad.attach(literal, voice[0]) >>> literal = abjad.LilyPondLiteral("", site="absolute_before") >>> abjad.attach(literal, voice[0]) >>> literal = abjad.LilyPondLiteral( ... "% before all formatting", ... site="absolute_before", ... ) >>> abjad.attach(literal, voice[0]) >>> literal = abjad.LilyPondLiteral("", site="absolute_after") >>> abjad.attach(literal, voice[-1]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { <BLANKLINE> % before all formatting \slurDotted c'8 ( d'8 e'8 f'8 ) <BLANKLINE> } .. container:: example LilyPond literals can be tagged: >>> voice = abjad.Voice("c'8 d'8 e'8 f'8") >>> abjad.slur(voice[:]) >>> literal = abjad.LilyPondLiteral(r"\slurDotted", site="before") >>> abjad.attach(literal, voice[0], tag=abjad.Tag("+PARTS")) >>> abjad.show(voice) # doctest: +SKIP >>> string = abjad.lilypond(voice, tags=True) >>> print(string) \new Voice { %! +PARTS \slurDotted c'8 ( d'8 e'8 f'8 ) } .. container:: example Multiline input is allowed: >>> voice = abjad.Voice("c'8 d'8 e'8 f'8") >>> abjad.slur(voice[:]) >>> lines = [ ... r"\stopStaff", ... r"\startStaff", ... r"\once \override Staff.StaffSymbol.color = #red", ... ] >>> literal = abjad.LilyPondLiteral(lines, site="before") >>> abjad.attach(literal, voice[2], tag=abjad.Tag("+PARTS")) >>> abjad.show(voice) # doctest: +SKIP >>> string = abjad.lilypond(voice, tags=True) >>> print(string) \new Voice { c'8 ( d'8 %! +PARTS \stopStaff %! +PARTS \startStaff %! +PARTS \once \override Staff.StaffSymbol.color = #red e'8 f'8 ) } .. container:: example REGRESSION. Duplicate literals are allowed: >>> staff = abjad.Staff("c'4 d' e' f'") >>> literal = abjad.LilyPondLiteral("% text", site="before") >>> abjad.attach(literal, staff[0]) >>> literal = abjad.LilyPondLiteral("% text", site="before") >>> abjad.attach(literal, staff[0]) >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { % text % text c'4 d'4 e'4 f'4 } .. container:: example Directed literal: >>> staff = abjad.Staff("c'4 d' e' f'") >>> literal = abjad.LilyPondLiteral(r"\f", directed=True, site="after") >>> bundle = abjad.bundle( ... literal, ... r"- \tweak color #blue", ... r"- \tweak DynamicLineSpanner.staff-padding 5", ... ) >>> abjad.attach(bundle, staff[0]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4 - \tweak DynamicLineSpanner.staff-padding 5 - \tweak color #blue \f d'4 e'4 f'4 } .. container:: example Nondirected literal: >>> staff = abjad.Staff("c'4 d' e' f'") >>> literal = abjad.LilyPondLiteral( ... r"\breathe", ... directed=False, ... site="after", ... ) >>> bundle = abjad.bundle(literal, r"\tweak color #blue") >>> abjad.attach(bundle, staff[0]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4 \tweak color #blue \breathe d'4 e'4 f'4 } Proper use of the ``directed`` property entails searching the LilyPond docs to understand whether LilyPond treats any particular command as directed or not. Most LilyPond commands are directed. LilyPond insists that a few commands (include ``\breathe``, ``\key``, ``\mark``) must not be directed. .. container:: example Tweaks: >>> staff = abjad.Staff("c'4 d' e' f'") >>> literal = abjad.LilyPondLiteral(r"\f", directed=True, site="after") >>> bundle = abjad.bundle(literal, r"- \tweak color #blue") >>> abjad.attach(bundle, staff[0]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4 - \tweak color #blue \f d'4 e'4 f'4 } """ argument: str | list[str] = "" _: dataclasses.KW_ONLY site: str = "before" directed: bool = False allowable_sites: typing.ClassVar[tuple[str, ...]] = ( "absolute_after", "absolute_before", "after", "before", "closing", "opening", ) can_attach_to_containers: typing.ClassVar[bool] = True format_leaf_children: typing.ClassVar[bool] = False
[docs] def __post_init__(self): if not isinstance(self.argument, str): assert all(isinstance(_, str) for _ in self.argument), repr(self.argument) assert isinstance(self.directed, bool), repr(self.directed) assert isinstance(self.site, str), repr(self.site) assert self.site in self.allowable_sites, repr(self.site)
def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) if isinstance(self.argument, str): pieces = [self.argument] else: assert isinstance(self.argument, list) pieces = self.argument[:] site.commands.extend(pieces) return contributions @property def post_event(self): """ Is true when literal is directed. """ return self.directed
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class Mode: """ Mode. .. container:: example >>> abjad.Mode("major") Mode(name='major') """ name: str = "major"
[docs] def __post_init__(self): assert isinstance(self.name, str), repr(self.name)
[docs] def intervals(self): """ Gets intervals in mode. .. container:: example >>> for _ in abjad.Mode("major").intervals(): _ NamedInterval('+M2') NamedInterval('+M2') NamedInterval('+m2') NamedInterval('+M2') NamedInterval('+M2') NamedInterval('+M2') NamedInterval('+m2') >>> for _ in abjad.Mode("dorian").intervals(): _ NamedInterval('+M2') NamedInterval('+m2') NamedInterval('+M2') NamedInterval('+M2') NamedInterval('+M2') NamedInterval('+m2') NamedInterval('+M2') """ intervals = [] m2 = _pitch.NamedInterval("m2") M2 = _pitch.NamedInterval("M2") A2 = _pitch.NamedInterval("aug2") dorian = [M2, m2, M2, M2, M2, m2, M2] if self.name == "dorian": intervals.extend(_sequence.rotate(dorian, n=0)) elif self.name == "phrygian": intervals.extend(_sequence.rotate(dorian, n=-1)) elif self.name == "lydian": intervals.extend(_sequence.rotate(dorian, n=-2)) elif self.name == "mixolydian": intervals.extend(_sequence.rotate(dorian, n=-3)) elif self.name in ("aeolian", "minor", "natural minor"): intervals.extend(_sequence.rotate(dorian, n=-4)) elif self.name == "locrian": intervals.extend(_sequence.rotate(dorian, n=-5)) elif self.name in ("ionian", "major"): intervals.extend(_sequence.rotate(dorian, n=-6)) elif self.name == "melodic minor": intervals.extend([M2, m2, M2, M2, M2, M2, m2]) elif self.name == "harmonic minor": intervals.extend([M2, m2, M2, M2, m2, A2, m2]) else: raise ValueError(f"unknown mode name: {self.name!r}.") return intervals
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class KeySignature: r""" Key signature. .. container:: example >>> staff = abjad.Staff("e'8 fs'8 gs'8 a'8") >>> key_signature = abjad.KeySignature( ... abjad.NamedPitchClass("e"), abjad.Mode("major") ... ) >>> abjad.attach(key_signature, staff[0]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \key e \major e'8 fs'8 gs'8 a'8 } .. container:: example >>> staff = abjad.Staff("e'8 fs'8 g'8 a'8") >>> key_signature = abjad.KeySignature( ... abjad.NamedPitchClass("e"), abjad.Mode("minor") ... ) >>> abjad.attach(key_signature, staff[0]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \key e \minor e'8 fs'8 g'8 a'8 } .. container:: example Tweaks: >>> staff = abjad.Staff("c'4 d' e' f'") >>> key_signature = abjad.KeySignature( ... abjad.NamedPitchClass("e"), abjad.Mode("minor") ... ) >>> bundle = abjad.bundle(key_signature, r"\tweak color #blue") >>> abjad.attach(bundle, staff[0]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \tweak color #blue \key e \minor c'4 d'4 e'4 f'4 } """ tonic: _pitch.NamedPitchClass = _pitch.NamedPitchClass("c") mode: Mode = Mode("major") context: typing.ClassVar[str] = "Staff" # find_context_on_attach: typing.ClassVar[bool] = True persistent: typing.ClassVar[bool] = True redraw: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "before"
[docs] def __post_init__(self): assert isinstance(self.tonic, _pitch.NamedPitchClass), repr(self.tonic) assert isinstance(self.mode, Mode), repr(self.mod)
def _get_lilypond_format(self): assert isinstance(self.mode, Mode), repr(self.mode) return rf"\key {self.tonic.name} \{self.mode.name}" def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._get_lilypond_format() site.commands.append(string) return contributions @property def name(self) -> str: """ Gets name of key signature. .. container:: example >>> abjad.KeySignature(abjad.NamedPitchClass("e"), abjad.Mode("major")).name 'E major' >>> abjad.KeySignature(abjad.NamedPitchClass("e"), abjad.Mode("minor")).name 'e minor' """ assert isinstance(self.mode, Mode) if self.mode.name == "major": tonic = self.tonic.name.upper() else: tonic = self.tonic.name return f"{tonic!s} {self.mode.name}"
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class LaissezVibrer: r""" Laissez vibrer. .. container:: example >>> chord = abjad.Chord("<c' e' g' c''>4") >>> laissez_vibrer = abjad.LaissezVibrer() >>> abjad.attach(laissez_vibrer, chord) >>> abjad.show(chord) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(chord) >>> print(string) <c' e' g' c''>4 \laissezVibrer .. container:: example Tweaks: >>> note = abjad.Note("c'4") >>> lv = abjad.LaissezVibrer() >>> bundle = abjad.bundle(lv, r"- \tweak color #blue") >>> abjad.attach(bundle, note) >>> abjad.show(note) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(note) >>> print(string) c'4 - \tweak color #blue \laissezVibrer """ post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" time_orientation: typing.ClassVar[_enums.Horizontal] = _enums.RIGHT def _get_lilypond_format(self): return r"\laissezVibrer" def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._get_lilypond_format() site.articulations.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class Markup: r""" LilyPond markup. .. container:: example >>> staff = abjad.Staff("c'8 d'8 e'8 f'8") >>> markup = abjad.Markup(r'\markup \italic "Allegro assai"') >>> abjad.attach(markup, staff[0], direction=abjad.UP) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'8 ^ \markup \italic "Allegro assai" d'8 e'8 f'8 } .. container:: example Tweaks: >>> staff = abjad.Staff("c'4 d' e' f'") >>> markup = abjad.Markup(r'\markup \bold "Allegro assai"') >>> bundle = abjad.bundle(markup, r"- \tweak color #blue") >>> abjad.attach(bundle, staff[0], direction=abjad.UP) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4 - \tweak color #blue ^ \markup \bold "Allegro assai" d'4 e'4 f'4 } .. container:: example Markup can be tagged: >>> staff = abjad.Staff("c'4 d' e' f'") >>> markup = abjad.Markup(r"\markup \italic Allegro") >>> abjad.attach(markup, staff[0], direction=abjad.UP, tag=abjad.Tag("RED:M1")) >>> abjad.show(staff) # doctest: +SKIP >>> string = abjad.lilypond(staff, tags=True) >>> print(string) \new Staff { c'4 %! M1 %! RED ^ \markup \italic Allegro d'4 e'4 f'4 } Tagged markup can be deactivated: >>> staff = abjad.Staff("c'4 d' e' f'") >>> markup = abjad.Markup(r"\markup \italic Allegro") >>> abjad.attach( ... markup, ... staff[0], ... deactivate=True, ... direction=abjad.UP, ... tag=abjad.Tag("RED:M1"), ... ) >>> abjad.show(staff) # doctest: +SKIP >>> string = abjad.lilypond(staff, tags=True) >>> print(string) \new Staff { c'4 %! M1 %! RED %@% ^ \markup \italic Allegro d'4 e'4 f'4 } .. container:: example REGRESSION: markup 1 doesn't disappear after markup 2 is attached: >>> staff = abjad.Staff("c'4 d' e' f'") >>> markup_1 = abjad.Markup(r"\markup \italic Allegro") >>> markup_2 = abjad.Markup(r'\markup \italic "non troppo"') >>> abjad.attach(markup_1, staff[0], direction=abjad.UP) >>> abjad.attach(markup_2, staff[0], direction=abjad.UP) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4 ^ \markup \italic "non troppo" ^ \markup \italic Allegro d'4 e'4 f'4 } """ string: str directed: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after"
[docs] def __post_init__(self): assert isinstance(self.string, str), repr(self.string)
def _get_contributions(self, *, component=None, wrapper=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._get_lilypond_format(wrapper=wrapper) site.markup.append(string) return contributions def _get_lilypond_format(self, *, wrapper=None): string = self.string if wrapper: direction = wrapper.direction or "-" direction = _string.to_tridirectional_lilypond_symbol(direction) string = rf"{direction} {self.string}" return string
[docs] @functools.total_ordering @dataclasses.dataclass(frozen=True, slots=True, unsafe_hash=True) class MetronomeMark: r""" MetronomeMark. .. container:: example Initializes integer-valued metronome mark: >>> score = abjad.Score() >>> staff = abjad.Staff("c'8 d'8 e'8 f'8") >>> score.append(staff) >>> mark = abjad.MetronomeMark(abjad.Duration(1, 4), 90) >>> abjad.attach(mark, staff[0]) >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { \tempo 4=90 c'8 d'8 e'8 f'8 } >> .. container:: example Initializes rational-valued metronome mark: >>> import fractions >>> score = abjad.Score() >>> staff = abjad.Staff("c'8 d'8 e'8 f'8") >>> score.append(staff) >>> mark = abjad.MetronomeMark(abjad.Duration(1, 4), fractions.Fraction(272, 3)) >>> abjad.attach(mark, staff[0]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', score]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { \tempo \markup \abjad-metronome-mark-mixed-number-markup #2 #0 #1 #"90" #"2" #"3" c'8 d'8 e'8 f'8 } >> Overrides rational-valued metronome mark with decimal string: >>> score = abjad.Score() >>> staff = abjad.Staff("c'8 d'8 e'8 f'8") >>> score.append(staff) >>> mark = abjad.MetronomeMark( ... abjad.Duration(1, 4), ... fractions.Fraction(272, 3), ... decimal="90.66", ... ) >>> abjad.attach(mark, staff[0]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', score]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { \tempo \markup \abjad-metronome-mark-markup #2 #0 #1 #"90.66" c'8 d'8 e'8 f'8 } >> Overrides rational-valued metronome mark with exact decimal: >>> score = abjad.Score() >>> staff = abjad.Staff("c'8 d'8 e'8 f'8") >>> score.append(staff) >>> mark = abjad.MetronomeMark( ... abjad.Duration(1, 4), ... fractions.Fraction(901, 10), ... decimal=True, ... ) >>> abjad.attach(mark, staff[0]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', score]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { \tempo \markup \abjad-metronome-mark-markup #2 #0 #1 #"90.1" c'8 d'8 e'8 f'8 } >> .. container:: example Initializes from text, duration and range: >>> score = abjad.Score() >>> staff = abjad.Staff("c'8 d'8 e'8 f'8") >>> score.append(staff) >>> mark = abjad.MetronomeMark(abjad.Duration(1, 4), (120, 133), "Quick") >>> abjad.attach(mark, staff[0]) >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { \tempo Quick 4=120-133 c'8 d'8 e'8 f'8 } >> .. container:: example Custom markup: >>> import fractions >>> markup = abjad.MetronomeMark.make_tempo_equation_markup( ... abjad.Duration(1, 4), ... 67.5, ... ) >>> mark = abjad.MetronomeMark( ... reference_duration=abjad.Duration(1, 4), ... units_per_minute=fractions.Fraction(135, 2), ... custom_markup=markup, ... ) >>> staff = abjad.Staff("c'4 d'4 e'4 f'4") >>> score = abjad.Score([staff]) >>> abjad.attach(mark, staff[0]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', score]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { \tempo \markup \abjad-metronome-mark-markup #2 #0 #1 #"67.5" c'4 d'4 e'4 f'4 } >> .. container:: example Decimal overrides: >>> import fractions >>> mark = abjad.MetronomeMark(abjad.Duration(1, 4), fractions.Fraction(272, 3)) >>> mark.decimal False >>> mark = abjad.MetronomeMark( ... abjad.Duration(1, 4), fractions.Fraction(272, 3), decimal="90.66", ... ) >>> mark.decimal '90.66' >>> mark = abjad.MetronomeMark( ... abjad.Duration(1, 4), fractions.Fraction(901, 10), decimal=True, ... ) >>> mark.decimal True .. container:: example Set ``hide=True`` when metronome mark should not appear in output (but should still determine effective metronome mark): >>> staff = abjad.Staff("c'4 d' e' f'") >>> score = abjad.Score([staff]) >>> metronome_mark_1 = abjad.MetronomeMark(abjad.Duration(1, 4), 72) >>> abjad.attach(metronome_mark_1, staff[0]) >>> metronome_mark_2 = abjad.MetronomeMark( ... textual_indication="Allegro", ... hide=True, ... ) >>> abjad.attach(metronome_mark_2, staff[2]) >>> abjad.show(score) # doctest: +SKIP >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { \tempo 4=72 c'4 d'4 e'4 f'4 } >> >>> for leaf in abjad.iterate.leaves(staff): ... prototype = abjad.MetronomeMark ... leaf, abjad.get.effective(leaf, prototype) ... (Note("c'4"), MetronomeMark(reference_duration=Duration(1, 4), units_per_minute=72, textual_indication=None, custom_markup=None, decimal=False, hide=False)) (Note("d'4"), MetronomeMark(reference_duration=Duration(1, 4), units_per_minute=72, textual_indication=None, custom_markup=None, decimal=False, hide=False)) (Note("e'4"), MetronomeMark(reference_duration=None, units_per_minute=None, textual_indication='Allegro', custom_markup=None, decimal=False, hide=True)) (Note("f'4"), MetronomeMark(reference_duration=None, units_per_minute=None, textual_indication='Allegro', custom_markup=None, decimal=False, hide=True)) .. container:: example Equality testing: >>> mark_1 = abjad.MetronomeMark(abjad.Duration(3, 32), 52) >>> mark_2 = abjad.MetronomeMark(abjad.Duration(3, 32), 52) >>> mark_1 == mark_2 True >>> mark_2 == mark_1 True .. container:: example >>> mark_1 = abjad.MetronomeMark(abjad.Duration(3, 32), 52) >>> mark_2 = abjad.MetronomeMark(abjad.Duration(6, 32), 104) >>> mark_1 == mark_2 False >>> mark_2 == mark_1 False .. container:: example >>> mark_1 = abjad.MetronomeMark(abjad.Duration(3, 32), 52, "Langsam") >>> mark_2 = abjad.MetronomeMark(abjad.Duration(3, 32), 52, "Langsam") >>> mark_3 = abjad.MetronomeMark(abjad.Duration(3, 32), 52, "Slow") >>> mark_1 == mark_2 True >>> mark_1 == mark_3 False """ reference_duration: _duration.Duration | None = None units_per_minute: int | fractions.Fraction | None = None textual_indication: str | None = None custom_markup: Markup | None = None decimal: bool | str = False hide: bool = dataclasses.field(compare=False, default=False) context: typing.ClassVar[str] = "Score" # TODO: make this work: # find_context_on_attach: typing.ClassVar[bool] = True mutates_offsets_in_seconds: typing.ClassVar[bool] = True parameter: typing.ClassVar[str] = "METRONOME_MARK" persistent: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "before"
[docs] def __post_init__(self): if self.reference_duration: assert isinstance(self.reference_duration, _duration.Duration), repr( self.reference_duration ) if isinstance(self.units_per_minute, float): raise Exception( f"do not set units-per-minute to float ({self.units_per_minute});" " use fraction with decimal override instead." ) prototype = (int, fractions.Fraction, collections.abc.Sequence, type(None)) assert isinstance(self.units_per_minute, prototype) if isinstance(self.units_per_minute, collections.abc.Sequence): assert len(self.units_per_minute) == 2 item_prototype = (int, _duration.Duration) assert all(isinstance(_, item_prototype) for _ in self.units_per_minute) assert isinstance(self.textual_indication, str | type(None)) if self.custom_markup is not None: assert isinstance(self.custom_markup, Markup) if self.decimal is not None: assert isinstance(self.decimal, bool | str), repr(self.decimal) assert isinstance(self.hide, bool), repr(self.hide)
[docs] def __lt__(self, argument) -> bool: """ Is true when ``argument`` is a metronome mark with quarters per minute greater than that of this metronome mark. """ assert isinstance(argument, type(self)), repr(argument) self_quarters_per_minute = self.quarters_per_minute or 0 argument_quarters_per_minute = argument.quarters_per_minute or 0 assert isinstance(self_quarters_per_minute, int | float | fractions.Fraction) assert isinstance( argument_quarters_per_minute, (int, float, fractions.Fraction) ) return self_quarters_per_minute < argument_quarters_per_minute
@property def _dotted(self): return self.reference_duration.lilypond_duration_string @property def _equation(self): if self.reference_duration is None: return if isinstance(self.units_per_minute, tuple): first, second = self.units_per_minute string = f"{self._dotted}={first}-{second}" return string elif isinstance(self.units_per_minute, fractions.Fraction): markup = MetronomeMark.make_tempo_equation_markup( self.reference_duration, self.units_per_minute, decimal=self.decimal, ) string = markup.string return string string = f"{self._dotted}={self.units_per_minute}" return string def _get_lilypond_format(self): equation = None if self.reference_duration is not None and self.units_per_minute is not None: equation = self._equation if self.custom_markup is not None: return rf"\tempo {self.custom_markup.string}" elif self.textual_indication and equation: return rf"\tempo {self.textual_indication} {equation}" elif equation: return rf"\tempo {equation}" elif self.textual_indication: return rf"\tempo {self.textual_indication}" else: return r"\tempo \default" def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) if not self.hide: string = self._get_lilypond_format() site.commands.append(string) return contributions def _get_markup(self): if self.custom_markup is not None: return self.custom_markup duration_log = int(math.log(self.reference_duration.denominator, 2)) stem_height = 1 string = r"\abjad-metronome-mark-markup" string += f" #{duration_log}" string += f" #{self.reference_duration.dot_count}" string += f" #{stem_height}" string += f' #"{self.units_per_minute}"' markup = Markup(rf"\markup {string}") return markup def _get_markup_arguments(self): assert self.custom_markup is None duration_log = int(math.log(self.reference_duration.denominator, 2)) dot_count = self.reference_duration.dot_count stem_height = 1 if not self.decimal: return { "duration_log": duration_log, "dot_count": dot_count, "stem_height": stem_height, "base": self.units_per_minute, } if isinstance(self.decimal, str): return (duration_log, dot_count, stem_height, self.decimal) assert self.decimal is True, repr(self.decimal) fraction = fractions.Fraction(self.units_per_minute) n, d = fraction.numerator, fraction.denominator base = n // d n = n % d return { "duration_log": duration_log, "dot_count": dot_count, "stem_height": stem_height, "base": base, "n": n, "d": d, } @property def is_imprecise(self) -> bool: """ Is true if metronome mark is entirely textual or if metronome mark's units_per_minute is a range. .. container:: example Imprecise metronome marks: >>> duration = abjad.Duration(1, 4) >>> abjad.MetronomeMark(textual_indication="Langsam").is_imprecise True >>> abjad.MetronomeMark(duration, (35, 50), "Langsam").is_imprecise True >>> abjad.MetronomeMark(duration, (35, 50)).is_imprecise True Precise metronome marks: >>> abjad.MetronomeMark(duration, 60).is_imprecise False >>> abjad.MetronomeMark(duration, 60, "Langsam").is_imprecise False """ if self.reference_duration is not None: if self.units_per_minute is not None: if not isinstance(self.units_per_minute, tuple): return False return True @property def quarters_per_minute(self) -> tuple | None | fractions.Fraction: """ Gets metronome mark quarters per minute. .. container:: example >>> mark = abjad.MetronomeMark(abjad.Duration(1, 8), 52) >>> mark.quarters_per_minute Fraction(104, 1) Gives tuple when metronome mark ``units_per_minute`` is a range. Gives none when metronome mark is imprecise. Gives fraction otherwise. """ if self.is_imprecise: return None if isinstance(self.units_per_minute, tuple): low = ( _duration.Duration(1, 4) / self.reference_duration * self.units_per_minute[0] ) high = ( _duration.Duration(1, 4) / self.reference_duration * self.units_per_minute[1] ) return (low, high) result = ( _duration.Duration(1, 4) / self.reference_duration * self.units_per_minute ) return fractions.Fraction(result)
[docs] def duration_to_milliseconds(self, duration) -> _duration.Duration: """ Gets millisecond value of ``duration`` under a given metronome mark. Dotted sixteenth lasts 1500 msec at quarter equals 60: .. container:: example >>> mark = abjad.MetronomeMark(abjad.Duration(1, 4), 60) >>> mark.duration_to_milliseconds((3, 8)) Duration(1500, 1) """ assert isinstance(self.reference_duration, _duration.Duration) denominator = self.reference_duration.denominator numerator = self.reference_duration.numerator whole_note_duration = fractions.Fraction(1000) whole_note_duration *= fractions.Fraction(denominator, numerator) whole_note_duration *= fractions.Fraction(60, self.units_per_minute) duration = _duration.Duration(duration) return _duration.Duration(duration * whole_note_duration)
[docs] @staticmethod def make_tempo_equation_markup( reference_duration, units_per_minute, *, decimal=False ) -> Markup: r""" Makes tempo equation markup. Integer-valued metronome mark: .. container:: example >>> markup = abjad.MetronomeMark.make_tempo_equation_markup((1, 4), 90) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', markup]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(markup) >>> print(string) \markup \abjad-metronome-mark-markup #2 #0 #1 #"90" .. container:: example Float-valued metronome mark: >>> markup = abjad.MetronomeMark.make_tempo_equation_markup((1, 4), 90.1) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', markup]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(markup) >>> print(string) \markup \abjad-metronome-mark-markup #2 #0 #1 #"90.1" .. container:: example Rational-valued metronome mark: >>> import fractions >>> markup = abjad.MetronomeMark.make_tempo_equation_markup( ... abjad.Duration(1, 4), ... fractions.Fraction(272, 3), ... ) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', markup]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(markup) >>> print(string) \markup \abjad-metronome-mark-mixed-number-markup #2 #0 #1 #"90" #"2" #"3" """ reference_duration_ = _duration.Duration(reference_duration) log = reference_duration_.exponent dots = reference_duration_.dot_count stem = 1 if isinstance( units_per_minute, fractions.Fraction ) and not _math.is_integer_equivalent_number(units_per_minute): if decimal: decimal_: float | str if decimal is True: decimal_ = float(units_per_minute) else: assert isinstance(decimal, str), repr(decimal) decimal_ = decimal markup = Markup( r"\markup \abjad-metronome-mark-markup" f' #{log} #{dots} #{stem} #"{decimal_}"' ) else: nonreduced = fractions.Fraction(units_per_minute) base = int(nonreduced) remainder = nonreduced - base n, d = remainder.numerator, remainder.denominator markup = Markup( r"\markup \abjad-metronome-mark-mixed-number-markup" f" #{log} #{dots} #{stem}" f' #"{base}" #"{n}" #"{d}"' ) else: markup = Markup( r"\markup \abjad-metronome-mark-markup" f' #{log} #{dots} #{stem} #"{units_per_minute}"' ) return markup
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class Ottava: r""" LilyPond ``\ottava`` command. .. container:: example >>> staff = abjad.Staff("c'4 d' e' f'") >>> ottava = abjad.Ottava(n=1) >>> abjad.attach(ottava, staff[0]) >>> ottava = abjad.Ottava(n=0, site="after") >>> abjad.attach(ottava, staff[-1]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \ottava 1 c'4 d'4 e'4 f'4 \ottava 0 } """ n: int = 0 _: dataclasses.KW_ONLY site: str = "before" context: typing.ClassVar[str] = "Staff" # find_context_on_attach: typing.ClassVar[bool] = True persistent: typing.ClassVar[bool] = True
[docs] def __post_init__(self): assert isinstance(self.n, int), repr(self.n) assert isinstance(self.site, str), repr(self.site)
def _before_attach(self, deactivate, component): for indicator in component._get_indicators(): if isinstance(indicator, type(self)) and indicator.site == self.site: classname = type(component).__name__ message = f"can not attach {self!r} to {classname}:" message += f"\n {indicator!r} is already attached to {classname}." raise Exception(message) def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) n = self.n or 0 string = rf"\ottava {n}" site.commands.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class RehearsalMark: r""" LilyPond ``\mark`` command. .. container:: example >>> staff = abjad.Staff("c'4 d' e' f'") >>> score = abjad.Score([staff]) >>> mark = abjad.RehearsalMark(number=1) >>> abjad.attach(mark, staff[0]) >>> scheme = "#format-mark-box-alphabet" >>> abjad.setting(score).markFormatter = scheme >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score \with { markFormatter = #format-mark-box-alphabet } << \new Staff { \mark #1 c'4 d'4 e'4 f'4 } >> .. container:: example Tweaks: >>> note = abjad.Note("c'4") >>> staff = abjad.Staff([note]) >>> score = abjad.Score([staff], name="Score") >>> mark = abjad.RehearsalMark(markup="A") >>> bundle = abjad.bundle(mark, r"\tweak color #blue") >>> abjad.attach(bundle, note) >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \context Score = "Score" << \new Staff { \tweak color #blue \mark A c'4 } >> """ markup: Markup | str | None = None number: int | None = None _: dataclasses.KW_ONLY site: str = "before" # find_context_on_attach: typing.ClassVar[bool] = True context: typing.ClassVar[str] = "Score"
[docs] def __post_init__(self): if self.markup is not None: assert isinstance(self.markup, str | Markup), repr(self.markup) if self.number is not None: assert isinstance(self.number, int), repr(self.number)
def _get_lilypond_format(self): if self.markup is not None: result = rf"\mark {self.markup}" elif self.number is not None: result = rf"\mark #{self.number}" else: result = r"\mark \default" return result def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._get_lilypond_format() site.commands.append(string) return contributions
[docs] @staticmethod def from_string(string) -> "RehearsalMark": """ Makes rehearsal mark from ``string``. .. container:: example >>> abjad.RehearsalMark.from_string("A") RehearsalMark(markup=None, number=1, site='before') .. container:: example >>> abjad.RehearsalMark.from_string("AA") RehearsalMark(markup=None, number=27, site='before') .. container:: example >>> abjad.RehearsalMark.from_string("AB") RehearsalMark(markup=None, number=28, site='before') .. container:: example >>> abjad.RehearsalMark.from_string("BA") RehearsalMark(markup=None, number=53, site='before') .. container:: example >>> abjad.RehearsalMark.from_string("BB") RehearsalMark(markup=None, number=54, site='before') """ number = 0 for place, letter in enumerate(reversed(string)): integer = ord(letter) - ord("A") + 1 multiplier = 26**place integer *= multiplier number += integer return RehearsalMark(number=number)
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class Repeat: r""" Repeat. .. container:: example Volta repeat: >>> container = abjad.Container("c'4 d'4 e'4 f'4") >>> staff = abjad.Staff([container]) >>> score = abjad.Score([staff]) >>> repeat = abjad.Repeat() >>> abjad.attach(repeat, container) >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { \repeat volta 2 { c'4 d'4 e'4 f'4 } } >> .. container:: example Unfold repeat: >>> container = abjad.Container("c'4 d'4 e'4 f'4") >>> staff = abjad.Staff([container]) >>> score = abjad.Score([staff]) >>> repeat = abjad.Repeat(repeat_type='unfold') >>> abjad.attach(repeat, container) >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { \repeat unfold 2 { c'4 d'4 e'4 f'4 } } >> """ repeat_count: int = 2 repeat_type: str = "volta" can_attach_to_containers: typing.ClassVar[bool] = True format_leaf_children: typing.ClassVar[bool] = False # find_context_on_attach: typing.ClassVar[bool] = True context: typing.ClassVar[str] = "Score" site: typing.ClassVar[str] = "before"
[docs] def __post_init__(self): assert isinstance(self.repeat_count, int), repr(self.repeat_count) assert isinstance(self.repeat_type, str), repr(self.repeat_type)
def _get_lilypond_format(self): return rf"\repeat {self.repeat_type} {self.repeat_count}" def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._get_lilypond_format() site.commands.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class RepeatTie: r""" LilyPond ``\repeatTie`` command. .. container:: example >>> voice = abjad.Voice("c'4 c' d' d'", name="Voice") >>> repeat_tie = abjad.RepeatTie() >>> bundle = abjad.bundle(repeat_tie, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[1]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \context Voice = "Voice" { c'4 c'4 - \tweak color #blue \repeatTie d'4 d'4 } >>> abjad.get.indicators(voice[1], abjad.RepeatTie) [RepeatTie()] >>> wrapper = abjad.get.indicator(voice[1], abjad.RepeatTie, unwrap=False) >>> wrapper.get_item() Bundle(indicator=RepeatTie(), tweaks=(Tweak(string='- \\tweak color #blue', i=None, tag=None),), comment=None) >>> for leaf in voice: ... leaf, abjad.get.logical_tie(leaf) ... (Note("c'4"), LogicalTie(items=[Note("c'4"), Note("c'4")])) (Note("c'4"), LogicalTie(items=[Note("c'4"), Note("c'4")])) (Note("d'4"), LogicalTie(items=[Note("d'4")])) (Note("d'4"), LogicalTie(items=[Note("d'4")])) .. container:: example With ``direction`` unset: >>> voice = abjad.Voice("c'4 c'4 c''4 c''4", name="Voice") >>> tie = abjad.RepeatTie() >>> bundle = abjad.bundle(tie, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[1]) >>> tie = abjad.RepeatTie() >>> bundle = abjad.bundle(tie, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[3]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \context Voice = "Voice" { c'4 c'4 - \tweak color #blue \repeatTie c''4 c''4 - \tweak color #blue \repeatTie } With ``direction=abjad.UP``: >>> voice = abjad.Voice("c'4 c'4 c''4 c''4", name="Voice") >>> tie = abjad.RepeatTie() >>> bundle = abjad.bundle(tie, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[1], direction=abjad.UP) >>> tie = abjad.RepeatTie() >>> bundle = abjad.bundle(tie, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[3], direction=abjad.UP) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \context Voice = "Voice" { c'4 c'4 - \tweak color #blue ^ \repeatTie c''4 c''4 - \tweak color #blue ^ \repeatTie } With ``direction=abjad.DOWN``: >>> voice = abjad.Voice("c'4 c'4 c''4 c''4", name="Voice") >>> tie = abjad.RepeatTie() >>> bundle = abjad.bundle(tie, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[1], direction=abjad.DOWN) >>> tie = abjad.RepeatTie() >>> bundle = abjad.bundle(tie, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[3], direction=abjad.DOWN) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \context Voice = "Voice" { c'4 c'4 - \tweak color #blue _ \repeatTie c''4 c''4 - \tweak color #blue _ \repeatTie } """ directed: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" def _attachment_test_all(self, argument): if not ( hasattr(argument, "written_pitch") or hasattr(argument, "written_pitches") ): string = f"Must be note or chord (not {argument})." return [string] return True def _get_contributions(self, *, component=None, wrapper=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) strings = [] if wrapper.direction is not None: string = _string.to_tridirectional_lilypond_symbol(wrapper.direction) strings.append(string) strings.append(r"\repeatTie") site.spanner_starts.extend(strings) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class ShortInstrumentName: r""" LilyPond ``\shortInstrumentName`` command. .. container:: example Set ``\shortInstrumentName`` to markup like this: >>> staff = abjad.Staff("c'4 d'4 e'4 f'4") >>> short_instrument_name = abjad.ShortInstrumentName(r"\markup Vc.") >>> abjad.attach(short_instrument_name, staff[0]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \set Staff.shortInstrumentName = \markup Vc. c'4 d'4 e'4 f'4 } Set ``\shortInstrumentName`` to custom function like this: >>> staff = abjad.Staff("c'4 d'4 e'4 f'4") >>> short_instrument_name = abjad.ShortInstrumentName(r"\my_custom_function") >>> abjad.attach(short_instrument_name, staff[0]) .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \set Staff.shortInstrumentName = \my_custom_function c'4 d'4 e'4 f'4 } """ markup: typing.Union[str, Markup] _: dataclasses.KW_ONLY context: str = "Staff" site: str = "before" latent: typing.ClassVar[bool] = True persistent: typing.ClassVar[bool] = True redraw: typing.ClassVar[bool] = True
[docs] def __post_init__(self): assert isinstance(self.markup, str | Markup), repr(self.markup) assert isinstance(self.context, str), repr(self.context) assert isinstance(self.site, str), repr(self.site)
def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) strings = self._get_lilypond_format() site.commands.extend(strings) return contributions def _get_lilypond_format(self, context=None): result = [] if isinstance(context, str): pass elif context is not None: context = context.lilypond_type else: context = self._get_lilypond_type() if isinstance(self.markup, Markup): string = self.markup.string else: assert isinstance(self.markup, str) string = self.markup string = rf"\set {context!s}.shortInstrumentName = {string}" result.append(string) return result def _get_lilypond_type(self): if isinstance(self.context, type): return self.context.__name__ elif isinstance(self.context, str): return self.context else: return type(self.context).__name__
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StaffChange: r""" Staff change. .. container:: example >>> staff_group = abjad.StaffGroup() >>> staff_group.lilypond_type = "PianoStaff" >>> rh_staff = abjad.Staff("c'8 d'8 e'8 f'8", name="RH_Staff") >>> lh_staff = abjad.Staff("s2", name="LH_Staff") >>> staff_group.extend([rh_staff, lh_staff]) >>> staff_change = abjad.StaffChange("LH_Staff") >>> abjad.attach(staff_change, rh_staff[2]) >>> abjad.show(staff_group) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff_group) >>> print(string) \new PianoStaff << \context Staff = "RH_Staff" { c'8 d'8 \change Staff = LH_Staff e'8 f'8 } \context Staff = "LH_Staff" { s2 } >> """ staff_name: str | bool context: typing.ClassVar[str] = "Staff" format_leaf_children: typing.ClassVar[bool] = False site: typing.ClassVar[str] = "before" time_orientation: typing.ClassVar[_enums.Horizontal] = _enums.RIGHT
[docs] def __post_init__(self): assert isinstance(self.staff_name, str | bool), repr(self.staff_name) if isinstance(self.staff_name, bool): assert self.staff_name is False, repr(self.staff_name)
def _get_lilypond_format(self): if self.staff_name is False: return r"\change Staff = ##f" return rf"\change Staff = {self.staff_name}" def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._get_lilypond_format() site.commands.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StartBeam: r""" LilyPond ``[`` command. .. container:: example >>> voice = abjad.Voice("c'8 d' e' f'") >>> start_beam = abjad.StartBeam() >>> bundle = abjad.bundle(start_beam, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_beam = abjad.StopBeam() >>> abjad.attach(stop_beam, voice[-1]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'8 - \tweak color #blue [ d'8 e'8 f'8 ] } .. container:: example With ``direction=abjad.DOWN``: >>> voice = abjad.Voice("c'8 d' e' f'") >>> start_beam = abjad.StartBeam() >>> bundle = abjad.bundle(start_beam, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0], direction=abjad.DOWN) >>> stop_beam = abjad.StopBeam() >>> abjad.attach(stop_beam, voice[-1]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'8 - \tweak color #blue _ [ d'8 e'8 f'8 ] } """ context: typing.ClassVar[str] = "Voice" directed: typing.ClassVar[bool] = True parameter: typing.ClassVar[str] = "BEAM" persistent: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_start: typing.ClassVar[bool] = True def _add_direction(self, string, wrapper): if wrapper.direction is not None: symbol = _string.to_tridirectional_lilypond_symbol(wrapper.direction) string = f"{symbol} {string}" return string def _get_contributions(self, *, component=None, wrapper=None): contributions = _contributions.ContributionsBySite() string = self._add_direction("[", wrapper) contributions.after.start_beam.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StartGroup: r""" LilyPond ``\startGroup`` command. .. container:: example >>> voice = abjad.Voice("c'4 d' e' f'") >>> start_group = abjad.StartGroup() >>> bundle = abjad.bundle(start_group, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_group = abjad.StopGroup() >>> abjad.attach(stop_group, voice[-1]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak color #blue \startGroup d'4 e'4 f'4 \stopGroup } """ context: typing.ClassVar[str] = "Voice" nestable_spanner: typing.ClassVar[bool] = True persistent: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_start: typing.ClassVar[bool] = True def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = r"\startGroup" site.spanner_starts.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StartHairpin: r""" Hairpin indicator. .. container:: example >>> voice = abjad.Voice("c'4 d' e' f'") >>> abjad.attach(abjad.Dynamic('p'), voice[0]) >>> abjad.attach(abjad.StartHairpin('<'), voice[0]) >>> abjad.attach(abjad.Dynamic('f'), voice[-1]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4.5 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4.5 } { c'4 \p \< d'4 e'4 f'4 \f } .. container:: example Crescendo: >>> voice = abjad.Voice("c'4 d' e' f'") >>> abjad.attach(abjad.Dynamic('p'), voice[0]) >>> abjad.attach(abjad.StartHairpin('<'), voice[0]) >>> abjad.attach(abjad.Dynamic('f'), voice[-1]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4.5 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4.5 } { c'4 \p \< d'4 e'4 f'4 \f } Crescendo dal niente: >>> voice = abjad.Voice("c'4 d' e' f'") >>> abjad.attach(abjad.StartHairpin('o<'), voice[0]) >>> abjad.attach(abjad.Dynamic('f'), voice[-1]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4.5 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4.5 } { c'4 - \tweak circled-tip ##t \< d'4 e'4 f'4 \f } Subito crescendo: >>> voice = abjad.Voice("c'4 d' e' f'") >>> abjad.attach(abjad.Dynamic('p'), voice[0]) >>> abjad.attach(abjad.StartHairpin('<|'), voice[0]) >>> abjad.attach(abjad.Dynamic('f'), voice[-1]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4.5 >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', voice]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4.5 } { c'4 \p - \tweak stencil #abjad-flared-hairpin \< d'4 e'4 f'4 \f } Subito crescendo dal niente: >>> voice = abjad.Voice("c'4 d' e' f'") >>> abjad.attach(abjad.StartHairpin("o<|"), voice[0]) >>> abjad.attach(abjad.Dynamic("f"), voice[-1]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4.5 >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', voice]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4.5 } { c'4 - \tweak circled-tip ##t - \tweak stencil #abjad-flared-hairpin \< d'4 e'4 f'4 \f } .. container:: example Diminuendo: >>> voice = abjad.Voice("c'4 d' e' f'") >>> abjad.attach(abjad.Dynamic('f'), voice[0]) >>> abjad.attach(abjad.StartHairpin('>'), voice[0]) >>> abjad.attach(abjad.Dynamic('p'), voice[-1]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4.5 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4.5 } { c'4 \f \> d'4 e'4 f'4 \p } Diminuendo al niente: >>> voice = abjad.Voice("c'4 d' e' f'") >>> abjad.attach(abjad.Dynamic('f'), voice[0]) >>> abjad.attach(abjad.StartHairpin('>o'), voice[0]) >>> abjad.attach(abjad.StopHairpin(), voice[-1]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4.5 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4.5 } { c'4 \f - \tweak circled-tip ##t \> d'4 e'4 f'4 \! } Subito diminuendo: >>> voice = abjad.Voice("c'4 d' e' f'") >>> abjad.attach(abjad.Dynamic('f'), voice[0]) >>> abjad.attach(abjad.StartHairpin('|>'), voice[0]) >>> abjad.attach(abjad.Dynamic('p'), voice[-1]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4.5 >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', voice]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4.5 } { c'4 \f - \tweak stencil #abjad-flared-hairpin \> d'4 e'4 f'4 \p } Subito diminuendo al niente: >>> voice = abjad.Voice("c'4 d' e' f'") >>> abjad.attach(abjad.Dynamic('f'), voice[0]) >>> abjad.attach(abjad.StartHairpin('|>o'), voice[0]) >>> abjad.attach(abjad.StopHairpin(), voice[-1]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4.5 >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', voice]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4.5 } { c'4 \f - \tweak circled-tip ##t - \tweak stencil #abjad-flared-hairpin \> d'4 e'4 f'4 \! } .. container:: example Constante: >>> voice = abjad.Voice("c'4 d' e' f'") >>> abjad.attach(abjad.Dynamic('p'), voice[0]) >>> abjad.attach(abjad.StartHairpin('--'), voice[0]) >>> abjad.attach(abjad.Dynamic('f'), voice[-1]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4.5 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4.5 } { c'4 \p - \tweak stencil #constante-hairpin \< d'4 e'4 f'4 \f } .. container:: example Tweaks: >>> voice = abjad.Voice("c'4 d' e' f'") >>> abjad.attach(abjad.Dynamic('p'), voice[0]) >>> start_hairpin = abjad.StartHairpin('<') >>> bundle = abjad.bundle(start_hairpin, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> abjad.attach(abjad.Dynamic('f'), voice[-1]) >>> abjad.override(voice).DynamicLineSpanner.staff_padding = 4.5 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \override DynamicLineSpanner.staff-padding = 4.5 } { c'4 \p - \tweak color #blue \< d'4 e'4 f'4 \f } """ shape: str = "<" context: typing.ClassVar[str] = "Voice" crescendo_start: typing.ClassVar[str] = r"\<" decrescendo_start: typing.ClassVar[str] = r"\>" directed: typing.ClassVar[bool] = True known_shapes = ("<", "o<", "<|", "o<|", ">", ">o", "|>", "|>o", "--") parameter: typing.ClassVar[str] = "DYNAMIC" persistent: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_start: typing.ClassVar[bool] = True trend: typing.ClassVar[bool] = True
[docs] def __post_init__(self): assert self.shape in self.known_shapes, repr(self.shape)
def _add_direction(self, string, *, wrapper=None): if wrapper.direction is not None: symbol = _string.to_tridirectional_lilypond_symbol(wrapper.direction) string = f"{symbol} {string}" return string @staticmethod def _circled_tip(): return _overrides.LilyPondOverride( grob_name="Hairpin", once=True, property_path="circled-tip", value=True, ) @staticmethod def _constante_hairpin(): return _overrides.LilyPondOverride( grob_name="Hairpin", once=True, property_path="stencil", value="#constante-hairpin", ) @staticmethod def _flared_hairpin(): return _overrides.LilyPondOverride( grob_name="Hairpin", once=True, property_path="stencil", value="#abjad-flared-hairpin", ) def _get_lilypond_format(self, *, wrapper=None): strings = [] if "--" in self.shape: override = self._constante_hairpin() string = override.tweak_string() strings.append(string) if "o" in self.shape: override = self._circled_tip() string = override.tweak_string() strings.append(string) if "|" in self.shape: override = self._flared_hairpin() string = override.tweak_string() strings.append(string) if "<" in self.shape or "--" in self.shape: string = self.crescendo_start string = self._add_direction(string, wrapper=wrapper) strings.append(string) elif ">" in self.shape: string = self.decrescendo_start string = self._add_direction(string, wrapper=wrapper) strings.append(string) else: raise ValueError(self.shape) return strings def _get_contributions(self, *, component=None, wrapper=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) strings = self._get_lilypond_format(wrapper=wrapper) site.spanner_starts.extend(strings) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StartPhrasingSlur: r""" LilyPond ``(`` command. .. container:: example >>> voice = abjad.Voice("c'4 d' e' f'") >>> start_phrasing_slur = abjad.StartPhrasingSlur() >>> bundle = abjad.bundle(start_phrasing_slur, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_phrasing_slur = abjad.StopPhrasingSlur() >>> abjad.attach(stop_phrasing_slur, voice[-1]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak color #blue \( d'4 e'4 f'4 \) } """ context: typing.ClassVar[str] = "Voice" directed: typing.ClassVar[bool] = True parameter: typing.ClassVar[str] = "PHRASING_SLUR" persistent: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_start: typing.ClassVar[bool] = True def _add_direction(self, string, *, wrapper=None): if wrapper.direction is not None: symbol = _string.to_tridirectional_lilypond_symbol(wrapper.direction) string = f"{symbol} {string}" return string def _get_contributions(self, *, component=None, wrapper=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._add_direction(r"\(", wrapper=wrapper) site.spanner_starts.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StartPianoPedal: r""" LilyPond ``\sustainOn``, ``\sostenutoOn``, ``\unaCorda`` commands. .. container:: example >>> staff = abjad.Staff("c'4 d' e' r") >>> start_piano_pedal = abjad.StartPianoPedal() >>> bundle = abjad.bundle( ... start_piano_pedal, ... r"- \tweak color #blue", ... r"- \tweak parent-alignment-X #center", ... ) >>> abjad.attach(bundle, staff[0], context="Staff") >>> stop_piano_pedal = abjad.StopPianoPedal() >>> abjad.attach(stop_piano_pedal, staff[1], context="Staff") >>> start_piano_pedal = abjad.StartPianoPedal() >>> bundle = abjad.bundle(start_piano_pedal, r"- \tweak color #red") >>> abjad.attach(bundle, staff[1], context="Staff") >>> stop_piano_pedal = abjad.StopPianoPedal() >>> abjad.attach(stop_piano_pedal, staff[2], context="Staff") >>> start_piano_pedal = abjad.StartPianoPedal() >>> bundle = abjad.bundle(start_piano_pedal, r"- \tweak color #green") >>> abjad.attach(bundle, staff[2], context="Staff") >>> stop_piano_pedal = abjad.StopPianoPedal() >>> abjad.attach(stop_piano_pedal, staff[3], context="Staff") >>> abjad.override(staff).SustainPedalLineSpanner.staff_padding = 5 >>> abjad.setting(staff).pedalSustainStyle = "#'mixed" >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override SustainPedalLineSpanner.staff-padding = 5 pedalSustainStyle = #'mixed } { c'4 - \tweak color #blue - \tweak parent-alignment-X #center \sustainOn d'4 \sustainOff - \tweak color #red \sustainOn e'4 \sustainOff - \tweak color #green \sustainOn r4 \sustainOff } """ # TODO: change "kind" to "command" with "\unaCorda", "\sostenutoOn", "\sustainOn" kind: str = "sustain" context: typing.ClassVar[str] = "StaffGroup" parameter: typing.ClassVar[str] = "PEDAL" persistent: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_start: typing.ClassVar[bool] = True
[docs] def __post_init__(self): assert self.kind in ("sustain", "sostenuto", "corda"), repr(self.kind)
def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) if self.kind == "corda": string = r"\unaCorda" elif self.kind == "sostenuto": string = r"\sostenutoOn" else: assert self.kind == "sustain" string = r"\sustainOn" site.spanner_starts.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StartSlur: r""" LilyPond ``(`` command. .. container:: example >>> voice = abjad.Voice("c'4 d' e' f'") >>> start_slur = abjad.StartSlur() >>> bundle = abjad.bundle(start_slur, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_slur = abjad.StopSlur() >>> abjad.attach(stop_slur, voice[-1]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak color #blue ( d'4 e'4 f'4 ) } .. container:: example With ``direction`` unset: >>> voice = abjad.Voice("c'8 d' e' f' c'' d'' e'' f''") >>> abjad.attach(abjad.StartSlur(), voice[0]) >>> abjad.attach(abjad.StopSlur(), voice[3]) >>> abjad.attach(abjad.StartSlur(), voice[4]) >>> abjad.attach(abjad.StopSlur(), voice[7]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'8 ( d'8 e'8 f'8 ) c''8 ( d''8 e''8 f''8 ) } With ``direction=abjad.UP``: >>> voice = abjad.Voice("c'8 d' e' f' c'' d'' e'' f''") >>> abjad.attach(abjad.StartSlur(), voice[0], direction=abjad.UP) >>> abjad.attach(abjad.StopSlur(), voice[3]) >>> abjad.attach(abjad.StartSlur(), voice[4], direction=abjad.UP) >>> abjad.attach(abjad.StopSlur(), voice[7]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'8 ^ ( d'8 e'8 f'8 ) c''8 ^ ( d''8 e''8 f''8 ) } With ``direction=abjad.DOWN``: >>> voice = abjad.Voice("c'8 d' e' f' c'' d'' e'' f''") >>> abjad.attach(abjad.StartSlur(), voice[0], direction=abjad.DOWN) >>> abjad.attach(abjad.StopSlur(), voice[3]) >>> abjad.attach(abjad.StartSlur(), voice[4], direction=abjad.DOWN) >>> abjad.attach(abjad.StopSlur(), voice[7]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'8 _ ( d'8 e'8 f'8 ) c''8 _ ( d''8 e''8 f''8 ) } """ context: typing.ClassVar[str] = "Voice" directed: typing.ClassVar[bool] = True parameter: typing.ClassVar[str] = "SLUR" persistent: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_start: typing.ClassVar[bool] = True def _add_direction(self, string, *, wrapper=None): if wrapper.direction is not None: symbol = _string.to_tridirectional_lilypond_symbol(wrapper.direction) string = f"{symbol} {string}" return string def _get_contributions(self, *, component=None, wrapper=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._add_direction("(", wrapper=wrapper) site.spanner_starts.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StartTextSpan: r""" LilyPond ``\startTextSpan`` command. .. container:: example >>> voice = abjad.Voice("c'4 d' e' f'") >>> start_text_span = abjad.StartTextSpan( ... left_text=abjad.Markup(r"\upright pont."), ... right_text=abjad.Markup(r"\markup \upright tasto"), ... style=r"\abjad-solid-line-with-arrow", ... ) >>> bundle = abjad.bundle(start_text_span, r"- \tweak staff-padding 2.5") >>> abjad.attach(bundle, voice[0]) >>> stop_text_span = abjad.StopTextSpan() >>> abjad.attach(stop_text_span, voice[-1]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', voice]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \abjad-solid-line-with-arrow - \tweak bound-details.left.text \markup \concat { \upright pont. \hspace #0.5 } - \tweak bound-details.right.text \markup \upright tasto - \tweak staff-padding 2.5 \startTextSpan d'4 e'4 f'4 \stopTextSpan } .. container:: example >>> abjad.StartTextSpan() StartTextSpan(command='\\startTextSpan', concat_hspace_left=0.5, concat_hspace_right=None, left_broken_text=None, left_text=None, right_padding=None, right_text=None, style=None) .. container:: example >>> voice = abjad.Voice("c'4 d' e' f'") >>> start_text_span = abjad.StartTextSpan( ... left_text=abjad.Markup(r"\upright pont."), ... right_text=abjad.Markup(r"\markup \upright tasto"), ... style=r"\abjad-dashed-line-with-arrow", ... ) >>> bundle = abjad.bundle( ... start_text_span, ... r"- \tweak color #blue", ... r"- \tweak staff-padding 2.5", ... ) >>> abjad.attach(bundle, voice[0]) >>> stop_text_span = abjad.StopTextSpan() >>> abjad.attach(stop_text_span, voice[-1]) >>> start_text_span = abjad.StartTextSpan( ... command=r"\startTextSpanOne", ... left_text=abjad.Markup(r"\upright A"), ... right_text=abjad.Markup(r"\markup \upright B"), ... style=r"\abjad-dashed-line-with-arrow", ... ) >>> bundle = abjad.bundle( ... start_text_span, ... r"- \tweak color #red", ... r"- \tweak staff-padding 6", ... ) >>> abjad.attach(bundle, voice[0]) >>> stop_text_span = abjad.StopTextSpan(command=r"\stopTextSpanOne") >>> abjad.attach(stop_text_span, voice[-1]) >>> markup = abjad.Markup(r"\markup SPACER") >>> bundle = abjad.bundle(markup, r"- \tweak transparent ##t") >>> abjad.attach(bundle, voice[0], direction=abjad.UP) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', voice]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak transparent ##t ^ \markup SPACER - \abjad-dashed-line-with-arrow - \tweak bound-details.left.text \markup \concat { \upright A \hspace #0.5 } - \tweak bound-details.right.text \markup \upright B - \tweak color #red - \tweak staff-padding 6 \startTextSpanOne - \abjad-dashed-line-with-arrow - \tweak bound-details.left.text \markup \concat { \upright pont. \hspace #0.5 } - \tweak bound-details.right.text \markup \upright tasto - \tweak color #blue - \tweak staff-padding 2.5 \startTextSpan d'4 e'4 f'4 \stopTextSpan \stopTextSpanOne } (Spacer text included to prevent docs from clipping output.) .. container:: example String literals are allowed in place of markup: >>> voice = abjad.Voice("c'4 d' e' f'") >>> left_text = r'- \tweak bound-details.left.text \markup' >>> left_text += r' \concat { \upright pont. \hspace #0.5 }' >>> right_text = r'- \tweak bound-details.right.text \markup' >>> right_text += r' \upright tasto' >>> start_text_span = abjad.StartTextSpan( ... left_text=left_text, ... right_text=right_text, ... style=r"\abjad-solid-line-with-arrow", ... ) >>> bundle = abjad.bundle(start_text_span, r"- \tweak staff-padding 2.5") >>> abjad.attach(bundle, voice[0]) >>> stop_text_span = abjad.StopTextSpan() >>> abjad.attach(stop_text_span, voice[-1]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', voice]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \abjad-solid-line-with-arrow - \tweak bound-details.left.text \markup \concat { \upright pont. \hspace #0.5 } - \tweak bound-details.right.text \markup \upright tasto - \tweak staff-padding 2.5 \startTextSpan d'4 e'4 f'4 \stopTextSpan } .. container:: example Styles: >>> voice = abjad.Voice("c'4 d' e' fs'") >>> start_text_span = abjad.StartTextSpan( ... left_text=abjad.Markup(r"\upright pont."), ... right_text=abjad.Markup(r"\markup \upright tasto"), ... style=r"\abjad-dashed-line-with-arrow", ... ) >>> bundle = abjad.bundle(start_text_span, r"- \tweak staff-padding 2.5") >>> abjad.attach(bundle, voice[0]) >>> stop_text_span = abjad.StopTextSpan() >>> abjad.attach(stop_text_span, voice[-1]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', voice]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \abjad-dashed-line-with-arrow - \tweak bound-details.left.text \markup \concat { \upright pont. \hspace #0.5 } - \tweak bound-details.right.text \markup \upright tasto - \tweak staff-padding 2.5 \startTextSpan d'4 e'4 fs'4 \stopTextSpan } .. container:: example >>> voice = abjad.Voice("c'4 d' e' f'") >>> start_text_span = abjad.StartTextSpan( ... left_text=abjad.Markup(r"\upright pont."), ... style=r"\abjad-dashed-line-with-hook", ... ) >>> bundle = abjad.bundle(start_text_span, r"- \tweak staff-padding 2.5") >>> abjad.attach(bundle, voice[0]) >>> stop_text_span = abjad.StopTextSpan() >>> abjad.attach(stop_text_span, voice[-1]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', voice]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \abjad-dashed-line-with-hook - \tweak bound-details.left.text \markup \concat { \upright pont. \hspace #0.5 } - \tweak staff-padding 2.5 \startTextSpan d'4 e'4 f'4 \stopTextSpan } .. container:: example >>> voice = abjad.Voice("c'4 d' e' f'") >>> start_text_span = abjad.StartTextSpan( ... left_text=abjad.Markup(r"\upright pont."), ... right_text=abjad.Markup(r"\markup \upright tasto"), ... style=r"\abjad-invisible-line", ... ) >>> bundle = abjad.bundle(start_text_span, r"- \tweak staff-padding 2.5") >>> abjad.attach(bundle, voice[0]) >>> stop_text_span = abjad.StopTextSpan() >>> abjad.attach(stop_text_span, voice[-1]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', voice]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \abjad-invisible-line - \tweak bound-details.left.text \markup \concat { \upright pont. \hspace #0.5 } - \tweak bound-details.right.text \markup \upright tasto - \tweak staff-padding 2.5 \startTextSpan d'4 e'4 f'4 \stopTextSpan } .. container:: example >>> voice = abjad.Voice("c'4 d' e' f'") >>> start_text_span = abjad.StartTextSpan( ... left_text=abjad.Markup(r"\upright pont."), ... right_text=abjad.Markup(r"\markup \upright tasto"), ... style=r"\abjad-solid-line-with-arrow", ... ) >>> bundle = abjad.bundle(start_text_span, r"- \tweak staff-padding 2.5") >>> abjad.attach(bundle, voice[0]) >>> stop_text_span = abjad.StopTextSpan() >>> abjad.attach(stop_text_span, voice[-1]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', voice]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \abjad-solid-line-with-arrow - \tweak bound-details.left.text \markup \concat { \upright pont. \hspace #0.5 } - \tweak bound-details.right.text \markup \upright tasto - \tweak staff-padding 2.5 \startTextSpan d'4 e'4 f'4 \stopTextSpan } .. container:: example >>> voice = abjad.Voice("c'4 d' e' f'") >>> start_text_span = abjad.StartTextSpan( ... left_text=abjad.Markup(r"\upright pont."), ... style=r"\abjad-solid-line-with-hook", ... ) >>> bundle = abjad.bundle(start_text_span, r"- \tweak staff-padding 2.5") >>> abjad.attach(bundle, voice[0]) >>> stop_text_span = abjad.StopTextSpan() >>> abjad.attach(stop_text_span, voice[-1]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', voice]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \abjad-solid-line-with-hook - \tweak bound-details.left.text \markup \concat { \upright pont. \hspace #0.5 } - \tweak staff-padding 2.5 \startTextSpan d'4 e'4 f'4 \stopTextSpan } Styles constrained to ``r'\abjad-dashed-line-with-arrow'``, ``r'\abjad-dashed-line-with-hook'``, ``r'\abjad-invisible-line'``, ``r'\abjad-solid-line-with-arrow'``, ``r'\abjad-solid-line-with-hook'``, none. .. container:: example REGRESSION. Tweaking the dash-fraction of the start text span works correctly: >>> voice = abjad.Voice("c'4 d' e' f'") >>> start_text_span = abjad.StartTextSpan( ... left_text=abjad.Markup(r"\upright pont."), ... right_text=abjad.Markup(r"\markup \upright tasto"), ... style=r"\abjad-dashed-line-with-arrow", ... ) >>> bundle = abjad.bundle( ... start_text_span, ... r"- \tweak dash-fraction 0.85", ... r"- \tweak staff-padding 2.5", ... ) >>> abjad.attach(bundle, voice[0]) >>> stop_text_span = abjad.StopTextSpan() >>> abjad.attach(stop_text_span, voice[-1]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', voice]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \abjad-dashed-line-with-arrow - \tweak bound-details.left.text \markup \concat { \upright pont. \hspace #0.5 } - \tweak bound-details.right.text \markup \upright tasto - \tweak dash-fraction 0.85 - \tweak staff-padding 2.5 \startTextSpan d'4 e'4 f'4 \stopTextSpan } """ command: str = r"\startTextSpan" concat_hspace_left: int | float = 0.5 concat_hspace_right: int | float | None = None left_broken_text: bool | str | Markup | None = None left_text: str | Markup | None = None right_padding: int | float | None = None right_text: str | Markup | None = None style: str | None = None allow_multiple_with_different_values: typing.ClassVar[bool] = True context: typing.ClassVar[str] = "Voice" directed: typing.ClassVar[bool] = True parameter: typing.ClassVar[str] = "TEXT_SPANNER" persistent: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_start: typing.ClassVar[bool] = True
[docs] def __post_init__(self): assert isinstance(self.command, str), repr(self.command) assert isinstance(self.style, str | None), repr(self.style)
def _get_contributions(self, *, component=None, wrapper=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) if self.style is not None: site.spanner_starts.append(rf"- {self.style}") if self.left_text: string = self._get_left_text_directive() site.spanner_starts.append(string) if self.left_broken_text is not None: string = self._get_left_broken_text_tweak() site.spanner_starts.append(string) if self.right_text: string = self._get_right_text_tweak() site.spanner_starts.append(string) if self.right_padding: string = self._get_right_padding_tweak() site.spanner_starts.append(string) string = self._add_direction(self.command, wrapper=wrapper) site.spanner_starts.append(string) return contributions def _add_direction(self, string, *, wrapper=None): if wrapper.direction is not None: symbol = _string.to_tridirectional_lilypond_symbol(wrapper.direction) string = f"{symbol} {string}" return string def _get_left_broken_text_tweak(self): if isinstance(self.left_broken_text, str): left_broken_text = self.left_broken_text else: left_broken_text = self.left_broken_text.string override = _overrides.LilyPondOverride( grob_name="TextSpanner", property_path=("bound-details", "left-broken", "text"), value=left_broken_text, ) string = override.tweak_string(self) return string def _get_left_text_directive(self): if isinstance(self.left_text, str): return self.left_text left_text_string = self.left_text.string left_text_string = left_text_string.removeprefix(r"\markup").strip() hspace_string = rf"\hspace #{self.concat_hspace_left}" markup = Markup(rf"\markup \concat {{ {left_text_string} {hspace_string} }}") override = _overrides.LilyPondOverride( grob_name="TextSpanner", property_path=("bound-details", "left", "text"), value=markup.string, ) string = override.tweak_string() return string def _get_right_padding_tweak(self): override = _overrides.LilyPondOverride( grob_name="TextSpanner", property_path=("bound-details", "right", "padding"), value=self.right_padding, ) string = override.tweak_string() return string def _get_right_text_tweak(self): if isinstance(self.right_text, str): return self.right_text if self.concat_hspace_right is not None: number = self.concat_hspace_right right_text = self.right_text.string markup = Markup(rf"\markup \concat {{ {right_text} \hspace #{number} }}") else: markup = self.right_text if isinstance(markup, str): value = markup else: assert isinstance(markup, Markup) value = markup.string override = _overrides.LilyPondOverride( grob_name="TextSpanner", property_path=("bound-details", "right", "text"), value=value, ) string = override.tweak_string() return string def _get_start_command(self): string = self._get_parameter_name() return rf"\start{string}"
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StartTrillSpan: r""" LilyPond ``\startTrillSpan`` command. .. container:: example >>> voice = abjad.Voice("c'4 d' e' f'") >>> start_trill_span = abjad.StartTrillSpan() >>> bundle = abjad.bundle(start_trill_span, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_trill_span = abjad.StopTrillSpan() >>> abjad.attach(stop_trill_span, voice[-1]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak color #blue \startTrillSpan d'4 e'4 f'4 \stopTrillSpan } .. container:: example With interval: >>> voice = abjad.Voice("c'4 d' e' f'") >>> start_trill_span = abjad.StartTrillSpan(interval=abjad.NamedInterval("M2")) >>> bundle = abjad.bundle(start_trill_span, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_trill_span = abjad.StopTrillSpan() >>> abjad.attach(stop_trill_span, voice[-1]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { \pitchedTrill c'4 - \tweak color #blue \startTrillSpan d' d'4 e'4 f'4 \stopTrillSpan } .. container:: example With pitch: >>> voice = abjad.Voice("c'4 d' e' f'") >>> start_trill_span = abjad.StartTrillSpan(pitch=abjad.NamedPitch("C#4")) >>> bundle = abjad.bundle(start_trill_span, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_trill_span = abjad.StopTrillSpan() >>> abjad.attach(stop_trill_span, voice[-1]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { \pitchedTrill c'4 - \tweak color #blue \startTrillSpan cs' d'4 e'4 f'4 \stopTrillSpan } """ interval: str | _pitch.NamedInterval | None = None pitch: str | _pitch.NamedPitch | None = None _: dataclasses.KW_ONLY force_trill_pitch_head_accidental: bool = False context: typing.ClassVar[str] = "Voice" parameter: typing.ClassVar[str] = "TRILL" persistent: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_start: typing.ClassVar[bool] = True
[docs] def __post_init__(self): if self.interval is not None: assert isinstance(self.interval, _pitch.NamedInterval), repr(self.interval) if self.pitch is not None: assert isinstance(self.pitch, _pitch.NamedPitch), repr(self.pitch)
def _get_contributions(self, *, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = r"\startTrillSpan" if self.interval or self.pitch: contributions.before.pitched_trill.append(r"\pitchedTrill") if self.pitch: pitch = self.pitch else: pitch = component.written_pitch + self.interval string = string + f" {pitch.name}" if self.force_trill_pitch_head_accidental is True: # LilyPond's TrillPitchHead does not obey LilyPond's \accidentalStyle; # forcing the accidental here ensures the accidental always appears force_accidental = "!" string = string + force_accidental # crucial to use "after" format slot here: site.trill_spanner_starts.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StemTremolo: r""" Stem tremolo. .. container:: example Sixteenth-note tremolo: >>> note = abjad.Note("c'4") >>> stem_tremolo = abjad.StemTremolo(16) >>> abjad.attach(stem_tremolo, note) >>> abjad.show(note) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(note) >>> print(string) c'4 :16 .. container:: example Thirty-second-note tremolo: >>> note = abjad.Note("c'4") >>> stem_tremolo = abjad.StemTremolo(32) >>> abjad.attach(stem_tremolo, note) >>> abjad.show(note) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(note) >>> print(string) c'4 :32 .. container:: example REGRESSION. Consider a note, rest, chord to which a stem tremolo is attached. When such a note, rest, chord splits into two notes, rests, chords then a stem tremolo attaches to each of the resultant notes, rests, chords: >>> staff = abjad.Staff("c'4 c'2.") >>> abjad.attach(abjad.StemTremolo(), staff[1]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4 c'2. :16 } >>> abjad.Meter.rewrite_meter(staff[:], abjad.Meter((3, 4))) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4 c'2 :16 ~ c'4 :16 } """ tremolo_flags: int = 16 site: typing.ClassVar[str] = "after" time_orientation: typing.ClassVar[_enums.Horizontal] = _enums.MIDDLE
[docs] def __post_init__(self): if not _math.is_nonnegative_integer_power_of_two(self.tremolo_flags): raise ValueError(f"nonnegative integer power of 2: {self.tremolo_flags!r}.")
def _get_lilypond_format(self): return f":{self.tremolo_flags!s}" def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._get_lilypond_format() site.stem_tremolos.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StopBeam: r""" LilyPond ``]`` command. .. container:: example Without leak: >>> voice = abjad.Voice("c'8 d' e' r") >>> start_beam = abjad.StartBeam() >>> bundle = abjad.bundle(start_beam, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_beam = abjad.StopBeam() >>> abjad.attach(stop_beam, voice[-2]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'8 - \tweak color #blue [ d'8 e'8 ] r8 } With leak: >>> voice = abjad.Voice("c'8 d' e' r") >>> start_beam = abjad.StartBeam() >>> bundle = abjad.bundle(start_beam, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_beam = abjad.StopBeam(leak=True) >>> abjad.attach(stop_beam, voice[-2]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'8 - \tweak color #blue [ d'8 e'8 <> ] r8 } """ leak: bool = dataclasses.field(default=False, compare=False) time_orientation: typing.ClassVar[_enums.Horizontal] = _enums.RIGHT context: typing.ClassVar[str] = "Voice" parameter: typing.ClassVar[str] = "BEAM" persistent: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_stop: typing.ClassVar[bool] = True def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = "]" if self.leak: site.leak.append(_EMPTY_CHORD) site.leaks.append(string) else: site.stop_beam.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StopGroup: r""" LilyPond ``\stopGroup`` command. .. container:: example Without leak: >>> voice = abjad.Voice("c'4 d' e' r") >>> start_group = abjad.StartGroup() >>> bundle = abjad.bundle(start_group, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_group = abjad.StopGroup() >>> abjad.attach(stop_group, voice[-2]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak color #blue \startGroup d'4 e'4 \stopGroup r4 } With leak: >>> voice = abjad.Voice("c'4 d' e' r") >>> start_group = abjad.StartGroup() >>> bundle = abjad.bundle(start_group, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_group = abjad.StopGroup(leak=True) >>> abjad.attach(stop_group, voice[-2]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak color #blue \startGroup d'4 e'4 <> \stopGroup r4 } .. container:: example REGRESSION. Leaked contributions appear last in postevent format site: >>> voice = abjad.Voice("c'8 d' e' f' r2") >>> abjad.beam(voice[:4]) >>> start_group = abjad.StartGroup() >>> bundle = abjad.bundle(start_group, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_group = abjad.StopGroup(leak=True) >>> abjad.attach(stop_group, voice[3]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'8 [ - \tweak color #blue \startGroup d'8 e'8 f'8 ] <> \stopGroup r2 } The leaked text spanner above does not inadvertantly leak the beam. """ leak: bool = dataclasses.field(default=False, compare=False) context: typing.ClassVar[str] = "Voice" nestable_spanner: typing.ClassVar[bool] = True persistent: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_stop: typing.ClassVar[bool] = True time_orientation: typing.ClassVar[_enums.Horizontal] = _enums.RIGHT def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = r"\stopGroup" if self.leak: site.leak.append(_EMPTY_CHORD) site.leaks.append(string) else: site.spanner_stops.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StopHairpin: r""" LilyPond ``\!`` command. .. container:: example Without leak: >>> voice = abjad.Voice("c'4 d' e' r") >>> start_hairpin = abjad.StartHairpin('<') >>> bundle = abjad.bundle(start_hairpin, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_hairpin = abjad.StopHairpin() >>> abjad.attach(stop_hairpin, voice[-2]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak color #blue \< d'4 e'4 \! r4 } With leak: >>> voice = abjad.Voice("c'4 d' e' r") >>> start_hairpin = abjad.StartHairpin('<') >>> bundle = abjad.bundle(start_hairpin, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_hairpin = abjad.StopHairpin(leak=True) >>> abjad.attach(stop_hairpin, voice[-2]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak color #blue \< d'4 e'4 <> \! r4 } """ leak: bool = dataclasses.field(default=False, compare=False) context: typing.ClassVar[str] = "Voice" parameter: typing.ClassVar[str] = "DYNAMIC" persistent: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_stop: typing.ClassVar[bool] = True def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = r"\!" if self.leak: site.leak.append(_EMPTY_CHORD) site.leaks.append(string) else: site.spanner_stops.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StopPhrasingSlur: r""" LilyPond ``\)`` command. .. container:: example Without leak: >>> voice = abjad.Voice("c'4 d' e' r") >>> start_phrasing_slur = abjad.StartPhrasingSlur() >>> bundle = abjad.bundle(start_phrasing_slur, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_phrasing_slur = abjad.StopPhrasingSlur() >>> abjad.attach(stop_phrasing_slur, voice[-2]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak color #blue \( d'4 e'4 \) r4 } With leak: >>> voice = abjad.Voice("c'4 d' e' r") >>> start_phrasing_slur = abjad.StartPhrasingSlur() >>> bundle = abjad.bundle(start_phrasing_slur, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_phrasing_slur = abjad.StopPhrasingSlur(leak=True) >>> abjad.attach(stop_phrasing_slur, voice[-2]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak color #blue \( d'4 e'4 <> \) r4 } .. container:: example REGRESSION. Leaked contributions appear last in postevent format site: >>> voice = abjad.Voice("c'8 d' e' f' r2") >>> abjad.beam(voice[:4]) >>> start_phrasing_slur = abjad.StartPhrasingSlur() >>> bundle = abjad.bundle(start_phrasing_slur, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_phrasing_slur = abjad.StopPhrasingSlur(leak=True) >>> abjad.attach(stop_phrasing_slur, voice[3]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'8 [ - \tweak color #blue \( d'8 e'8 f'8 ] <> \) r2 } The leaked text spanner above does not inadvertantly leak the beam. """ leak: bool = dataclasses.field(default=False, compare=False) context: typing.ClassVar[str] = "Voice" parameter: typing.ClassVar[str] = "PHRASING_SLUR" persistent: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_stop: typing.ClassVar[bool] = True def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = r"\)" if self.leak: site.leak.append(_EMPTY_CHORD) site.leaks.append(string) else: site.spanner_stops.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StopPianoPedal: r""" LilyPond ``\sostenutoOff``, ``\sustainOff``, ``\treCorde`` commands. .. container:: example Without leak: >>> staff = abjad.Staff("c'4 d' e' r") >>> start_piano_pedal = abjad.StartPianoPedal() >>> bundle = abjad.bundle( ... start_piano_pedal, ... r"- \tweak color #blue", ... r"- \tweak parent-alignment-X #center", ... ) >>> abjad.attach(bundle, staff[0], context="Staff") >>> stop_piano_pedal = abjad.StopPianoPedal() >>> bundle = abjad.bundle( ... stop_piano_pedal, ... r"- \tweak color #red", ... r"- \tweak parent-alignment-X #center", ... ) >>> abjad.attach(bundle, staff[-2], context="Staff") >>> abjad.override(staff).SustainPedalLineSpanner.staff_padding = 5 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override SustainPedalLineSpanner.staff-padding = 5 } { c'4 - \tweak color #blue - \tweak parent-alignment-X #center \sustainOn d'4 e'4 - \tweak color #red - \tweak parent-alignment-X #center \sustainOff r4 } With leak: >>> staff = abjad.Staff("c'4 d' e' r") >>> start_piano_pedal = abjad.StartPianoPedal() >>> bundle = abjad.bundle( ... start_piano_pedal, ... r"- \tweak color #blue", ... r"- \tweak parent-alignment-X #center", ... ) >>> abjad.attach(bundle, staff[0], context="Staff") >>> stop_piano_pedal = abjad.StopPianoPedal(leak=True) >>> bundle = abjad.bundle( ... stop_piano_pedal, ... r"- \tweak color #red", ... r"- \tweak parent-alignment-X #center", ... ) >>> abjad.attach(bundle, staff[-2], context="Staff") >>> abjad.override(staff).SustainPedalLineSpanner.staff_padding = 5 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override SustainPedalLineSpanner.staff-padding = 5 } { c'4 - \tweak color #blue - \tweak parent-alignment-X #center \sustainOn d'4 e'4 <> - \tweak color #red - \tweak parent-alignment-X #center \sustainOff r4 } """ # TODO: change "kind" to "command" with "\treCorde", "\sostenutoOff", "\sustainOff" kind: str = "sustain" leak: bool = dataclasses.field(default=False, compare=False) time_orientation: typing.ClassVar[_enums.Horizontal] = _enums.RIGHT context: typing.ClassVar[str] = "StaffGroup" parameter: typing.ClassVar[str] = "PEDAL" persistent: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_stop: typing.ClassVar[bool] = True
[docs] def __post_init__(self): assert self.kind in ("sustain", "sostenuto", "corda") assert isinstance(self.leak, bool), repr(self.leak)
def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) if self.kind == "corda": command = r"\treCorde" elif self.kind == "sostenuto": command = r"\sostenutoOff" else: assert self.kind == "sustain" command = r"\sustainOff" if self.leak: site.leak.append(_EMPTY_CHORD) site.leaks.append(command) else: site.spanner_stops.append(command) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StopSlur: r""" LilyPond ``)`` command. .. container:: example Without leak: >>> voice = abjad.Voice("c'4 d' e' r") >>> start_slur = abjad.StartSlur() >>> bundle = abjad.bundle(start_slur, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_slur = abjad.StopSlur() >>> abjad.attach(stop_slur, voice[-2]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak color #blue ( d'4 e'4 ) r4 } With leak: >>> voice = abjad.Voice("c'4 d' e' r") >>> start_slur = abjad.StartSlur() >>> bundle = abjad.bundle(start_slur, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_slur = abjad.StopSlur(leak=True) >>> abjad.attach(stop_slur, voice[-2]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak color #blue ( d'4 e'4 <> ) r4 } .. container:: example REGRESSION. Leaked contributions appear last in postevent format site. The leaked text spanner above does not inadvertantly leak the beam: >>> voice = abjad.Voice("c'8 d' e' f' r2") >>> abjad.beam(voice[:4]) >>> start_slur = abjad.StartSlur() >>> bundle = abjad.bundle(start_slur, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_slur = abjad.StopSlur(leak=True) >>> abjad.attach(stop_slur, voice[3]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'8 [ - \tweak color #blue ( d'8 e'8 f'8 ] <> ) r2 } """ leak: bool = dataclasses.field(default=False, compare=False) time_orientation: typing.ClassVar[_enums.Horizontal] = _enums.RIGHT context: typing.ClassVar[str] = "Voice" parameter: typing.ClassVar[str] = "SLUR" persistent: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_stop: typing.ClassVar[bool] = True
[docs] def __post_init__(self): assert isinstance(self.leak, bool), repr(self.leak)
def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = ")" if self.leak: site.leak.append(_EMPTY_CHORD) site.leaks.append(string) else: site.spanner_stops.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StopTextSpan: r""" LilyPond ``\stopTextSpan`` command. .. container:: example Without leak: >>> voice = abjad.Voice("c'4 d' e' r") >>> start_text_span = abjad.StartTextSpan( ... left_text=abjad.Markup(r"\upright pont."), ... right_text=abjad.Markup(r"\markup \upright tasto"), ... style=r"\abjad-dashed-line-with-arrow", ... ) >>> bundle = abjad.bundle(start_text_span, r"- \tweak staff-padding 2.5") >>> abjad.attach(bundle, voice[0]) >>> stop_text_span = abjad.StopTextSpan() >>> abjad.attach(stop_text_span, voice[-2]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', voice]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \abjad-dashed-line-with-arrow - \tweak bound-details.left.text \markup \concat { \upright pont. \hspace #0.5 } - \tweak bound-details.right.text \markup \upright tasto - \tweak staff-padding 2.5 \startTextSpan d'4 e'4 \stopTextSpan r4 } With leak: >>> voice = abjad.Voice("c'4 d' e' r") >>> start_text_span = abjad.StartTextSpan( ... left_text=abjad.Markup(r"\upright pont."), ... right_text=abjad.Markup(r"\markup \upright tasto"), ... style=r"\abjad-dashed-line-with-arrow", ... ) >>> bundle = abjad.bundle(start_text_span, r"- \tweak staff-padding 2.5") >>> abjad.attach(bundle, voice[0]) >>> stop_text_span = abjad.StopTextSpan(leak=True) >>> abjad.attach(stop_text_span, voice[-2]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', voice]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \abjad-dashed-line-with-arrow - \tweak bound-details.left.text \markup \concat { \upright pont. \hspace #0.5 } - \tweak bound-details.right.text \markup \upright tasto - \tweak staff-padding 2.5 \startTextSpan d'4 e'4 <> \stopTextSpan r4 } """ command: str = r"\stopTextSpan" leak: bool = dataclasses.field(default=False, compare=False) context: typing.ClassVar[str] = "Voice" enchained: typing.ClassVar[bool] = True nestable_spanner: typing.ClassVar[bool] = True parameter: typing.ClassVar[str] = "TEXT_SPANNER" persistent: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_stop: typing.ClassVar[bool] = True
[docs] def __post_init__(self): assert isinstance(self.command, str), repr(self.command) assert self.command.startswith("\\"), repr(self.command) assert isinstance(self.leak, bool), repr(self.leak)
def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self.command if self.leak: site.leak.append(_EMPTY_CHORD) site.leaks.append(string) else: site.spanner_stops.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class StopTrillSpan: r""" LilyPond ``\stopTrillSpan`` command. .. container:: example Without leak: >>> voice = abjad.Voice("c'4 d' e' r") >>> start_trill_span = abjad.StartTrillSpan() >>> bundle = abjad.bundle(start_trill_span, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_trill_span = abjad.StopTrillSpan() >>> abjad.attach(stop_trill_span, voice[-2]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak color #blue \startTrillSpan d'4 e'4 \stopTrillSpan r4 } With leak: >>> voice = abjad.Voice("c'4 d' e' r") >>> start_trill_span = abjad.StartTrillSpan() >>> bundle = abjad.bundle(start_trill_span, r"- \tweak color #blue") >>> abjad.attach(bundle, voice[0]) >>> stop_trill_span = abjad.StopTrillSpan(leak=True) >>> abjad.attach(stop_trill_span, voice[-2]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice { c'4 - \tweak color #blue \startTrillSpan d'4 e'4 <> \stopTrillSpan r4 } """ leak: bool = dataclasses.field(default=False, compare=False) time_orientation: typing.ClassVar[_enums.Horizontal] = _enums.RIGHT context: typing.ClassVar[str] = "Voice" parameter: typing.ClassVar[str] = "TRILL" persistent: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" spanner_stop: typing.ClassVar[bool] = True
[docs] def __post_init__(self): assert isinstance(self.leak, bool), repr(self.leak)
def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = r"\stopTrillSpan" if self.leak: site.leak.append(_EMPTY_CHORD) site.leaks.append(string) else: site.spanner_stops.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class TextMark: r""" LilyPond ``\textMark``, ``\textEndMark`` commands. Text mark: .. container:: example >>> staff = abjad.Staff("c'4 d' e' f'") >>> score = abjad.Score([staff]) >>> mark = abjad.TextMark(r'\textMark \markup \italic "V.S."') >>> abjad.attach(mark, staff[-1]) >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { c'4 d'4 e'4 \textMark \markup \italic "V.S." f'4 } >> Tweaks: >>> staff = abjad.Staff("c'4 d' e' f'") >>> score = abjad.Score([staff]) >>> mark = abjad.TextMark(r'\textMark \markup \italic "V.S."') >>> bundle = abjad.bundle(mark, r"\tweak color #blue") >>> abjad.attach(bundle, staff[-1]) >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { c'4 d'4 e'4 \tweak color #blue \textMark \markup \italic "V.S." f'4 } >> .. container:: example Text end mark: >>> staff = abjad.Staff("c'4 d' e' f'") >>> score = abjad.Score([staff]) >>> mark = abjad.TextMark(r'\textEndMark \markup \italic "V.S."', site="after") >>> abjad.attach(mark, staff[-1]) >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { c'4 d'4 e'4 f'4 \textEndMark \markup \italic "V.S." } >> """ string: str _: dataclasses.KW_ONLY site: str = "before" # find_context_on_attach: typing.ClassVar[bool] = True context: typing.ClassVar[str] = "Score"
[docs] def __post_init__(self): assert isinstance(self.string, str), repr(self.string)
def _get_lilypond_format(self): return self.string def _get_contributions(self, component=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._get_lilypond_format() site.commands.append(string) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class Tie: r""" LilyPond ``~`` command. .. container:: example >>> staff = abjad.Staff("c'4 c' d' d'") >>> tie = abjad.Tie() >>> bundle = abjad.bundle(tie, r"- \tweak color #blue") >>> abjad.attach(bundle, staff[0]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4 - \tweak color #blue ~ c'4 d'4 d'4 } >>> for leaf in staff: ... leaf, abjad.get.logical_tie(leaf) ... (Note("c'4"), LogicalTie(items=[Note("c'4"), Note("c'4")])) (Note("c'4"), LogicalTie(items=[Note("c'4"), Note("c'4")])) (Note("d'4"), LogicalTie(items=[Note("d'4")])) (Note("d'4"), LogicalTie(items=[Note("d'4")])) .. container:: example With ``direction`` unset: >>> staff = abjad.Staff("c'4 c' c'' c''") >>> abjad.attach(abjad.Tie(), staff[0]) >>> abjad.attach(abjad.Tie(), staff[2]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4 ~ c'4 c''4 ~ c''4 } With ``direction=abjad.UP``: >>> staff = abjad.Staff("c'4 c' c'' c''") >>> abjad.attach(abjad.Tie(), staff[0], direction=abjad.UP) >>> abjad.attach(abjad.Tie(), staff[2], direction=abjad.UP) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4 ^ ~ c'4 c''4 ^ ~ c''4 } With ``direction=abjad.DOWN``: >>> staff = abjad.Staff("c'4 c' c'' c''") >>> abjad.attach(abjad.Tie(), staff[0], direction=abjad.DOWN) >>> abjad.attach(abjad.Tie(), staff[2], direction=abjad.DOWN) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4 _ ~ c'4 c''4 _ ~ c''4 } """ directed: typing.ClassVar[bool] = True post_event: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "after" def _add_direction(self, string, *, wrapper=None): if wrapper.direction is not None: symbol = _string.to_tridirectional_lilypond_symbol(wrapper.direction) string = f"{symbol} {string}" return string def _attachment_test_all(self, argument): if not ( hasattr(argument, "written_pitch") or hasattr(argument, "written_pitches") ): string = f"Must be note or chord (not {argument})." return [string] return True def _get_contributions(self, *, component=None, wrapper=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) string = self._add_direction("~", wrapper=wrapper) strings = [string] site.spanner_starts.extend(strings) return contributions
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class TimeSignature: r""" Time signature. .. container:: example >>> staff = abjad.Staff("c'8 d'8 e'8") >>> score = abjad.Score([staff], name="Score") >>> time_signature = abjad.TimeSignature((3, 8)) >>> abjad.attach(time_signature, staff[0]) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \time 3/8 c'8 d'8 e'8 } .. container:: example Create score-contexted time signatures like this: >>> staff = abjad.Staff("c'8 d'8 e'8 c'8 d'8 e'8") >>> score = abjad.Score([staff], name="Score") >>> time_signature = abjad.TimeSignature((3, 8)) >>> abjad.attach(time_signature, staff[0], context="Score") >>> score[:] = [] Score-contexted time signatures format behind comments when no Abjad score container is found: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { %%% \time 3/8 %%% c'8 d'8 e'8 c'8 d'8 e'8 } >>> abjad.show(staff) # doctest: +SKIP Score-contexted time signatures format normally when an Abjad score container is found: >>> score = abjad.Score([staff]) >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { \time 3/8 c'8 d'8 e'8 c'8 d'8 e'8 } >> >>> abjad.show(score) # doctest: +SKIP .. container:: example Time signatures can be tagged: >>> staff = abjad.Staff("c'8 d'8 e'8 c'8 d'8 e'8") >>> score = abjad.Score([staff], name="Score") >>> time_signature = abjad.TimeSignature((3, 8)) >>> abjad.attach( ... time_signature, ... staff[0], ... context="Score", ... tag=abjad.Tag("+PARTS"), ... ) >>> abjad.show(staff) # doctest: +SKIP >>> string = abjad.lilypond(score, tags=True) >>> print(string) \context Score = "Score" << \new Staff { %! +PARTS \time 3/8 c'8 d'8 e'8 c'8 d'8 e'8 } >> .. container:: example Set ``hide=True`` time signature should not appear in output (but should still determine effective time signature): >>> staff = abjad.Staff("c'4 d' e' f'") >>> score = abjad.Score([staff], name="Score") >>> time_signature = abjad.TimeSignature((4, 4)) >>> abjad.attach(time_signature, staff[0]) >>> time_signature = abjad.TimeSignature((2, 4), hide=True) >>> abjad.attach(time_signature, staff[2]) >>> abjad.show(staff) # doctest: +SKIP >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \time 4/4 c'4 d'4 e'4 f'4 } >>> for leaf in abjad.iterate.leaves(staff): ... prototype = abjad.TimeSignature ... leaf, abjad.get.effective(leaf, prototype) ... (Note("c'4"), TimeSignature(pair=(4, 4), hide=False, partial=None)) (Note("d'4"), TimeSignature(pair=(4, 4), hide=False, partial=None)) (Note("e'4"), TimeSignature(pair=(2, 4), hide=True, partial=None)) (Note("f'4"), TimeSignature(pair=(2, 4), hide=True, partial=None)) """ pair: tuple[int, int] hide: bool = dataclasses.field(compare=False, default=False) partial: _duration.Duration | None = None check_effective_context: typing.ClassVar[bool] = True context: typing.ClassVar[str] = "Score" persistent: typing.ClassVar[bool] = True site: typing.ClassVar[str] = "before"
[docs] def __post_init__(self): assert isinstance(self.pair, tuple), repr(self.pair) assert len(self.pair) == 2, repr(self.pair) assert isinstance(self.pair[0], int) assert isinstance(self.pair[1], int) if self.partial is not None: assert isinstance(self.partial, _duration.Duration), repr(self.partial)
def _get_contributions(self, *, component=None, wrapper=None): contributions = _contributions.ContributionsBySite() site = getattr(contributions, self.site) if self.is_non_dyadic_rational and not self.hide: string = '#(ly:expect-warning "strange time signature found")' site.commands.append(string) strings = self._get_lilypond_format() site.commands.extend(strings) return contributions def _get_lilypond_format(self): result = [] if self.hide: return result if self.partial is None: result.append(rf"\time {self.numerator}/{self.denominator}") else: duration_string = self.partial.lilypond_duration_string partial_directive = rf"\partial {duration_string}" result.append(partial_directive) string = rf"\time {self.numerator}/{self.denominator}" result.append(string) return result @property def denominator(self) -> int: """ Gets denominator of time signature: .. container:: example >>> abjad.TimeSignature((3, 8)).denominator 8 """ return self.pair[1] @property def duration(self) -> _duration.Duration: """ Gets duration of time signature. .. container:: example >>> abjad.TimeSignature((3, 8)).duration Duration(3, 8) """ return _duration.Duration(*self.pair) @property def is_non_dyadic_rational(self) -> bool: r""" Is true when time signature has non-power-of-two denominator. .. container:: example With non-power-of-two denominator: >>> time_signature = abjad.TimeSignature((7, 12)) >>> time_signature.is_non_dyadic_rational True With power-of-two denominator: >>> time_signature = abjad.TimeSignature((3, 8)) >>> time_signature.is_non_dyadic_rational False Suppresses LilyPond "strange time signature" warning: >>> tuplet = abjad.Tuplet((2, 3), "c'4 d' e' f'") >>> staff = abjad.Staff([tuplet]) >>> score = abjad.Score([staff], name="Score") >>> time_signature = abjad.TimeSignature((4, 3)) >>> abjad.attach(time_signature, tuplet[0]) >>> abjad.show(staff) # doctest: +SKIP >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \tweak edge-height #'(0.7 . 0) \tuplet 3/2 { #(ly:expect-warning "strange time signature found") \time 4/3 c'4 d'4 e'4 f'4 } } """ return not _math.is_nonnegative_integer_power_of_two(self.denominator) @property def implied_prolation(self) -> fractions.Fraction: """ Gets implied prolation of time signature. .. container:: example Implied prolation of dyadic time signature: >>> abjad.TimeSignature((3, 8)).implied_prolation Fraction(1, 1) Implied prolation of nondyadic time signature: >>> abjad.TimeSignature((7, 12)).implied_prolation Fraction(2, 3) """ return _duration.Duration(1, self.denominator).implied_prolation @property def numerator(self) -> int: """ Gets numerator of time signature. .. container:: example >>> abjad.TimeSignature((3, 8)).numerator 3 """ return self.pair[0]
[docs] @staticmethod def from_string(string) -> "TimeSignature": """ Makes new time signature from fraction ``string``. .. container:: example >>> abjad.TimeSignature.from_string("6/8") TimeSignature(pair=(6, 8), hide=False, partial=None) """ assert isinstance(string, str), repr(string) parts = string.split("/") assert len(parts) == 2, repr(parts) numbers = [int(_) for _ in parts] numerator, denominator = numbers return TimeSignature((numerator, denominator))
[docs] def to_dyadic_rational(self) -> "TimeSignature": """ Makes new time signature equivalent to current time signature with power-of-two denominator. .. container:: example >>> abjad.TimeSignature((1, 12)).to_dyadic_rational() TimeSignature(pair=(1, 12), hide=False, partial=None) >>> abjad.TimeSignature((2, 12)).to_dyadic_rational() TimeSignature(pair=(2, 12), hide=False, partial=None) >>> abjad.TimeSignature((3, 12)).to_dyadic_rational() TimeSignature(pair=(2, 8), hide=False, partial=None) >>> abjad.TimeSignature((4, 12)).to_dyadic_rational() TimeSignature(pair=(4, 12), hide=False, partial=None) >>> abjad.TimeSignature((5, 12)).to_dyadic_rational() TimeSignature(pair=(5, 12), hide=False, partial=None) >>> abjad.TimeSignature((6, 12)).to_dyadic_rational() TimeSignature(pair=(4, 8), hide=False, partial=None) >>> abjad.TimeSignature((1, 14)).to_dyadic_rational() TimeSignature(pair=(1, 14), hide=False, partial=None) >>> abjad.TimeSignature((2, 14)).to_dyadic_rational() TimeSignature(pair=(2, 14), hide=False, partial=None) >>> abjad.TimeSignature((3, 14)).to_dyadic_rational() TimeSignature(pair=(3, 14), hide=False, partial=None) >>> abjad.TimeSignature((4, 14)).to_dyadic_rational() TimeSignature(pair=(4, 14), hide=False, partial=None) >>> abjad.TimeSignature((5, 14)).to_dyadic_rational() TimeSignature(pair=(5, 14), hide=False, partial=None) >>> abjad.TimeSignature((6, 14)).to_dyadic_rational() TimeSignature(pair=(6, 14), hide=False, partial=None) """ non_power_of_two_denominator = self.denominator power_of_two_denominator = _math.greatest_power_of_two_less_equal( non_power_of_two_denominator ) pair = _duration.with_denominator(self.pair, power_of_two_denominator) return type(self)(pair)
[docs] @dataclasses.dataclass(frozen=True, order=True, slots=True, unsafe_hash=True) class VoiceNumber: r""" LilyPond ``\voiceOne``, ``\voiceTwo``, ``\voiceThree``, ``\voiceFour``, ``\oneVoice`` commands. .. container:: example >>> staff = abjad.Staff() >>> voice_1 = abjad.Voice("g'8 a' b' c''") >>> command = abjad.VoiceNumber(n=1) >>> abjad.attach(command, voice_1[0]) >>> voice_2 = abjad.Voice("e'8 f' g' a'") >>> command = abjad.VoiceNumber(n=2) >>> abjad.attach(command, voice_2[0]) >>> container = abjad.Container([voice_1, voice_2], simultaneous=True) >>> staff.append(container) >>> voice = abjad.Voice("c''4 a'") >>> command = abjad.VoiceNumber() >>> abjad.attach(command, voice[0]) >>> staff.append(voice) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { << \new Voice { \voiceOne g'8 a'8 b'8 c''8 } \new Voice { \voiceTwo e'8 f'8 g'8 a'8 } >> \new Voice { \oneVoice c''4 a'4 } } >>> for leaf in abjad.select.leaves(staff): ... command = abjad.get.effective(leaf, abjad.VoiceNumber) ... print(f"{leaf}, {command}") Note("g'8"), VoiceNumber(n=1, leak=False) Note("a'8"), VoiceNumber(n=1, leak=False) Note("b'8"), VoiceNumber(n=1, leak=False) Note("c''8"), VoiceNumber(n=1, leak=False) Note("e'8"), VoiceNumber(n=2, leak=False) Note("f'8"), VoiceNumber(n=2, leak=False) Note("g'8"), VoiceNumber(n=2, leak=False) Note("a'8"), VoiceNumber(n=2, leak=False) Note("c''4"), VoiceNumber(n=None, leak=False) Note("a'4"), VoiceNumber(n=None, leak=False) """ n: int | None = None leak: bool = dataclasses.field(default=False, compare=False) check_effective_context: typing.ClassVar[bool] = True context: typing.ClassVar[str] = "Voice" parameter: typing.ClassVar[str] = "VOICE_NUMBER" persistent: typing.ClassVar[bool] = True temporarily_do_not_check: typing.ClassVar[bool] = True
[docs] def __post_init__(self): assert self.n in (1, 2, 3, 4, None), repr(self.n) assert isinstance(self.leak, bool), repr(self.leak)
def _get_contributions(self, *, component=None, wrapper=None): contributions = _contributions.ContributionsBySite() string = self._get_lilypond_format() if self.leak: contributions.after.leak.append(_EMPTY_CHORD) contributions.after.leaks.append(string) else: contributions.before.commands.append(string) return contributions def _get_lilypond_format(self): if self.n == 1: string = r"\voiceOne" elif self.n == 2: string = r"\voiceTwo" elif self.n == 3: string = r"\voiceThree" elif self.n == 4: string = r"\voiceFour" else: assert self.n is None string = r"\oneVoice" return string