Source code for abjad.parentage

import collections
import fractions

from . import math as _math
from . import score as _score


[docs] class Parentage(collections.abc.Sequence): r''' Parentage of a component. .. container:: example >>> score = abjad.Score() >>> string = r"""\new Voice = "Treble_Voice" { e'4 }""" >>> treble_staff = abjad.Staff(string, name="Treble_Staff") >>> score.append(treble_staff) >>> string = r"""\new Voice = "Bass_Voice" { c4 }""" >>> bass_staff = abjad.Staff(string, name="Bass_Staff") >>> clef = abjad.Clef('bass') >>> abjad.attach(clef, bass_staff[0][0]) >>> score.append(bass_staff) >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \context Staff = "Treble_Staff" { \context Voice = "Treble_Voice" { e'4 } } \context Staff = "Bass_Staff" { \context Voice = "Bass_Voice" { \clef "bass" c4 } } >> >>> bass_voice = score["Bass_Voice"] >>> note = bass_voice[0] >>> for component in abjad.get.parentage(note): ... component ... Note('c4') Voice('c4', name='Bass_Voice') Staff('{ c4 }', name='Bass_Staff') Score("{ { e'4 } } { { c4 } }", simultaneous=True) ''' ### CLASS VARIABLES ### __slots__ = ("_component", "_components") ### INITIALIZER ### def __init__(self, component=None): components = [] if component is not None: assert isinstance(component, _score.Component), repr(component) components.extend(component._get_parentage()) self._component = component self._components = tuple(components) ### SPECIAL METHODS ###
[docs] def __eq__(self, argument): """ Is true when ``argument`` is a parent with the same components as this parentage. """ if isinstance(argument, type(self)): if len(self) == len(argument): for c, d in zip(self, argument): if c is not d: return False else: return True return False
[docs] def __getitem__(self, argument): """ Gets ``argument``. Returns component or tuple of components. """ return self.components.__getitem__(argument)
[docs] def __len__(self) -> int: """ Gets number of components in parentage. """ return len(self.components)
[docs] def __repr__(self): """ Gets repr. """ return f"{type(self).__name__}(component={self.component!r})"
### PRIVATE METHODS ### @staticmethod def _id_string(component): lhs = component.__class__.__name__ rhs = getattr(component, "name", None) or id(component) return f"{lhs}-{rhs!r}" def _prolations(self): prolations = [] default = fractions.Fraction(1) for parent in self: prolation = getattr(parent, "implied_prolation", default) prolations.append(prolation) return prolations ### PUBLIC PROPERTIES ### @property def component(self) -> _score.Component: r""" Gets component. .. container:: example REGRESSION. Works with grace notes (and containers): >>> music_voice = abjad.Voice("c'4 d' e' f'", name="MusicVoice") >>> container = abjad.BeforeGraceContainer("cs'16") >>> abjad.attach(container, music_voice[1]) >>> obgc = abjad.on_beat_grace_container("g'16 gs' a' as'", music_voice[2:3]) >>> abjad.attach(abjad.Articulation(">"), obgc[0]) >>> container = abjad.AfterGraceContainer("fs'16") >>> abjad.attach(container, music_voice[3]) >>> staff = abjad.Staff([music_voice]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', staff]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \context Voice = "MusicVoice" { c'4 \grace { cs'16 } d'4 << \context Voice = "On_Beat_Grace_Container" { \set fontSize = #-3 \slash \voiceOne < \tweak font-size 0 \tweak transparent ##t e' g' >16 - \accent [ ( gs'16 a'16 as'16 ) ] } \context Voice = "MusicVoice" { \voiceTwo e'4 } >> \oneVoice \afterGrace f'4 { fs'16 } } } >>> for component in abjad.select.components(staff): ... parentage = abjad.get.parentage(component) ... print(f"{repr(component):30} {repr(parentage.component)}") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Note("c'4") Note("c'4") BeforeGraceContainer("cs'16") BeforeGraceContainer("cs'16") Note("cs'16") Note("cs'16") Note("d'4") Note("d'4") Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") Chord("<e' g'>16") Chord("<e' g'>16") Note("gs'16") Note("gs'16") Note("a'16") Note("a'16") Note("as'16") Note("as'16") Voice("e'4", name='MusicVoice') Voice("e'4", name='MusicVoice') Note("e'4") Note("e'4") Note("f'4") Note("f'4") AfterGraceContainer("fs'16") AfterGraceContainer("fs'16") Note("fs'16") Note("fs'16") """ return self._component @property def components(self) -> tuple[_score.Component]: r""" Gets components. .. container:: example REGRESSION. Works with grace notes (and containers): >>> music_voice = abjad.Voice("c'4 d' e' f'", name="MusicVoice") >>> container = abjad.BeforeGraceContainer("cs'16") >>> abjad.attach(container, music_voice[1]) >>> obgc = abjad.on_beat_grace_container("g'16 gs' a' as'", music_voice[2:3]) >>> abjad.attach(abjad.Articulation(">"), obgc[0]) >>> container = abjad.AfterGraceContainer("fs'16") >>> abjad.attach(container, music_voice[3]) >>> staff = abjad.Staff([music_voice]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', staff]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \context Voice = "MusicVoice" { c'4 \grace { cs'16 } d'4 << \context Voice = "On_Beat_Grace_Container" { \set fontSize = #-3 \slash \voiceOne < \tweak font-size 0 \tweak transparent ##t e' g' >16 - \accent [ ( gs'16 a'16 as'16 ) ] } \context Voice = "MusicVoice" { \voiceTwo e'4 } >> \oneVoice \afterGrace f'4 { fs'16 } } } >>> for component in abjad.select.components(staff): ... parentage = abjad.get.parentage(component) ... components = parentage.components ... print(f"{repr(component)}:") ... for component_ in components: ... print(f" {repr(component_)}") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }"): Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice'): Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("c'4"): Note("c'4") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") BeforeGraceContainer("cs'16"): BeforeGraceContainer("cs'16") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("cs'16"): Note("cs'16") BeforeGraceContainer("cs'16") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("d'4"): Note("d'4") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }"): Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16"): OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Chord("<e' g'>16"): Chord("<e' g'>16") OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("gs'16"): Note("gs'16") OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("a'16"): Note("a'16") OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("as'16"): Note("as'16") OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Voice("e'4", name='MusicVoice'): Voice("e'4", name='MusicVoice') Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("e'4"): Note("e'4") Voice("e'4", name='MusicVoice') Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("f'4"): Note("f'4") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") AfterGraceContainer("fs'16"): AfterGraceContainer("fs'16") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("fs'16"): Note("fs'16") AfterGraceContainer("fs'16") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") """ return self._components @property def orphan(self) -> bool: r""" Is true when component has no parent. .. container:: example REGRESSION. Works with grace notes (and containers): >>> music_voice = abjad.Voice("c'4 d' e' f'", name="MusicVoice") >>> container = abjad.BeforeGraceContainer("cs'16") >>> abjad.attach(container, music_voice[1]) >>> obgc = abjad.on_beat_grace_container("g'16 gs' a' as'", music_voice[2:3]) >>> abjad.attach(abjad.Articulation(">"), obgc[0]) >>> container = abjad.AfterGraceContainer("fs'16") >>> abjad.attach(container, music_voice[3]) >>> staff = abjad.Staff([music_voice]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', staff]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \context Voice = "MusicVoice" { c'4 \grace { cs'16 } d'4 << \context Voice = "On_Beat_Grace_Container" { \set fontSize = #-3 \slash \voiceOne < \tweak font-size 0 \tweak transparent ##t e' g' >16 - \accent [ ( gs'16 a'16 as'16 ) ] } \context Voice = "MusicVoice" { \voiceTwo e'4 } >> \oneVoice \afterGrace f'4 { fs'16 } } } >>> for component in abjad.select.components(staff): ... parentage = abjad.get.parentage(component) ... print(f"{repr(component):30} {repr(parentage.orphan)}") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") True Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') False Note("c'4") False BeforeGraceContainer("cs'16") False Note("cs'16") False Note("d'4") False Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") False OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") False Chord("<e' g'>16") False Note("gs'16") False Note("a'16") False Note("as'16") False Voice("e'4", name='MusicVoice') False Note("e'4") False Note("f'4") False AfterGraceContainer("fs'16") False Note("fs'16") False """ return self.parent is None @property def parent(self) -> _score.Component | None: r""" Gets parent. .. container:: example REGRESSION. Works with grace notes (and containers): >>> music_voice = abjad.Voice("c'4 d' e' f'", name="MusicVoice") >>> container = abjad.BeforeGraceContainer("cs'16") >>> abjad.attach(container, music_voice[1]) >>> obgc = abjad.on_beat_grace_container("g'16 gs' a' as'", music_voice[2:3]) >>> abjad.attach(abjad.Articulation(">"), obgc[0]) >>> container = abjad.AfterGraceContainer("fs'16") >>> abjad.attach(container, music_voice[3]) >>> staff = abjad.Staff([music_voice]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', staff]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \context Voice = "MusicVoice" { c'4 \grace { cs'16 } d'4 << \context Voice = "On_Beat_Grace_Container" { \set fontSize = #-3 \slash \voiceOne < \tweak font-size 0 \tweak transparent ##t e' g' >16 - \accent [ ( gs'16 a'16 as'16 ) ] } \context Voice = "MusicVoice" { \voiceTwo e'4 } >> \oneVoice \afterGrace f'4 { fs'16 } } } >>> for component in abjad.select.components(staff): ... parentage = abjad.get.parentage(component) ... print(f"{repr(component):30} {repr(parentage.parent)}") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") None Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("c'4") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') BeforeGraceContainer("cs'16") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Note("cs'16") BeforeGraceContainer("cs'16") Note("d'4") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") Chord("<e' g'>16") OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") Note("gs'16") OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") Note("a'16") OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") Note("as'16") OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") Voice("e'4", name='MusicVoice') Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") Note("e'4") Voice("e'4", name='MusicVoice') Note("f'4") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') AfterGraceContainer("fs'16") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Note("fs'16") AfterGraceContainer("fs'16") """ return self.get(n=1) @property def prolation(self) -> fractions.Fraction: r""" Gets prolation. .. container:: example REGRESSION. Works with grace notes (and containers): >>> music_voice = abjad.Voice( ... r"\times 2/3 { c'4 d' e' } \times 2/3 { f' g' a' }", ... name="MusicVoice" ... ) >>> container = abjad.BeforeGraceContainer("cs'16") >>> abjad.attach(container, music_voice[0][1]) >>> obgc = abjad.on_beat_grace_container("a'8 b'", music_voice[1][:1]) >>> container = abjad.AfterGraceContainer("fs'16") >>> abjad.attach(container, music_voice[1][2]) >>> staff = abjad.Staff([music_voice]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', staff]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \context Voice = "MusicVoice" { \tuplet 3/2 { c'4 \grace { cs'16 } d'4 e'4 } \tuplet 3/2 { << \context Voice = "On_Beat_Grace_Container" { \set fontSize = #-3 \slash \voiceOne < \tweak font-size 0 \tweak transparent ##t f' a' >8 [ ( b'8 ) ] } \context Voice = "MusicVoice" { \voiceTwo f'4 } >> \oneVoice g'4 \afterGrace a'4 { fs'16 } } } } >>> for component in abjad.select.components(staff): ... parentage = abjad.get.parentage(component) ... print(f"{repr(component):30} {repr(parentage.prolation)}") Staff("{ { 2/3 c'4 d'4 e'4 } { 2/3 { { <f' a'>8 b'8 } { f'4 } } g'4 a'4 } }") Fraction(1, 1) Voice("{ 2/3 c'4 d'4 e'4 } { 2/3 { { <f' a'>8 b'8 } { f'4 } } g'4 a'4 }", name='MusicVoice') Fraction(1, 1) Tuplet('3:2', "c'4 d'4 e'4") Fraction(2, 3) Note("c'4") Fraction(2, 3) BeforeGraceContainer("cs'16") Fraction(2, 3) Note("cs'16") Fraction(2, 3) Note("d'4") Fraction(2, 3) Note("e'4") Fraction(2, 3) Tuplet('3:2', "{ { <f' a'>8 b'8 } { f'4 } } g'4 a'4") Fraction(2, 3) Container("{ <f' a'>8 b'8 } { f'4 }") Fraction(2, 3) OnBeatGraceContainer("<f' a'>8 b'8") Fraction(2, 3) Chord("<f' a'>8") Fraction(2, 3) Note("b'8") Fraction(2, 3) Voice("f'4", name='MusicVoice') Fraction(2, 3) Note("f'4") Fraction(2, 3) Note("g'4") Fraction(2, 3) Note("a'4") Fraction(2, 3) AfterGraceContainer("fs'16") Fraction(2, 3) Note("fs'16") Fraction(2, 3) """ prolations = [fractions.Fraction(1)] + self._prolations() products = _math.cumulative_products(prolations) return products[-1] @property def root(self) -> _score.Component: r""" Gets root. .. container:: example REGRESSION. Works with grace notes (and containers): >>> music_voice = abjad.Voice("c'4 d' e' f'", name="MusicVoice") >>> container = abjad.BeforeGraceContainer("cs'16") >>> abjad.attach(container, music_voice[1]) >>> obgc = abjad.on_beat_grace_container("g'16 gs' a' as'", music_voice[2:3]) >>> abjad.attach(abjad.Articulation(">"), obgc[0]) >>> container = abjad.AfterGraceContainer("fs'16") >>> abjad.attach(container, music_voice[3]) >>> staff = abjad.Staff([music_voice]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', staff]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \context Voice = "MusicVoice" { c'4 \grace { cs'16 } d'4 << \context Voice = "On_Beat_Grace_Container" { \set fontSize = #-3 \slash \voiceOne < \tweak font-size 0 \tweak transparent ##t e' g' >16 - \accent [ ( gs'16 a'16 as'16 ) ] } \context Voice = "MusicVoice" { \voiceTwo e'4 } >> \oneVoice \afterGrace f'4 { fs'16 } } } >>> for component in abjad.select.components(staff): ... parentage = abjad.get.parentage(component) ... print(f"{repr(component):30} {repr(parentage.root)}") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("c'4") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") BeforeGraceContainer("cs'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("cs'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("d'4") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Chord("<e' g'>16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("gs'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("a'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("as'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Voice("e'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("e'4") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("f'4") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") AfterGraceContainer("fs'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("fs'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") """ root = self.get(n=-1) assert isinstance(root, _score.Component), repr(root) return root ### PUBLIC METHODS ###
[docs] def count(self, prototype=None) -> int: r""" Gets number of ``prototype`` in parentage. .. container:: example Gets tuplet count: >>> staff = abjad.Staff( ... r"\times 2/3 { c'2 \times 2/3 { d'8 e' f' } } \times 2/3 { c'4 d' e' }" ... ) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \tuplet 3/2 { c'2 \tuplet 3/2 { d'8 e'8 f'8 } } \tuplet 3/2 { c'4 d'4 e'4 } } >>> for component in abjad.select.components(staff): ... parentage = abjad.get.parentage(component) ... count = parentage.count(abjad.Tuplet) ... print(f"{repr(component):55} {repr(count)}") Staff("{ 2/3 c'2 { 2/3 d'8 e'8 f'8 } } { 2/3 c'4 d'4 e'4 }") 0 Tuplet('3:2', "c'2 { 2/3 d'8 e'8 f'8 }") 1 Note("c'2") 1 Tuplet('3:2', "d'8 e'8 f'8") 2 Note("d'8") 2 Note("e'8") 2 Note("f'8") 2 Tuplet('3:2', "c'4 d'4 e'4") 1 Note("c'4") 1 Note("d'4") 1 Note("e'4") 1 .. container:: example Gets voice count: >>> outer_red_voice = abjad.Voice("e''8 d''", name="Red_Voice") >>> inner_red_voice = abjad.Voice("c''4 b' c''8", name="Red_Voice") >>> inner_blue_voice = abjad.Voice("e'4 f' e'8", name="Blue_Voice") >>> container = abjad.Container( ... [inner_red_voice, inner_blue_voice], ... simultaneous=True, ... ) >>> outer_red_voice.append(container) >>> outer_red_voice.extend("d''8") >>> abjad.override(outer_red_voice).NoteHead.color = "#red" >>> command = abjad.VoiceNumber(1) >>> abjad.attach(command, outer_red_voice[0]) >>> abjad.override(inner_blue_voice).NoteHead.color = "#blue" >>> command = abjad.VoiceNumber(2) >>> abjad.attach(command, inner_blue_voice[0]) >>> dynamic = abjad.Dynamic("f") >>> abjad.attach(dynamic, outer_red_voice[0]) >>> abjad.show(outer_red_voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(outer_red_voice) >>> print(string) \context Voice = "Red_Voice" \with { \override NoteHead.color = #red } { \voiceOne e''8 \f d''8 << \context Voice = "Red_Voice" { c''4 b'4 c''8 } \context Voice = "Blue_Voice" \with { \override NoteHead.color = #blue } { \voiceTwo e'4 f'4 e'8 } >> d''8 } >>> for leaf in abjad.iterate.leaves(outer_red_voice): ... depth = abjad.get.parentage(leaf).count(abjad.Voice) ... print(leaf, depth) ... Note("e''8") 1 Note("d''8") 1 Note("c''4") 2 Note("b'4") 2 Note("c''8") 2 Note("e'4") 2 Note("f'4") 2 Note("e'8") 2 Note("d''8") 1 .. container:: example REGRESSION. Works with grace notes (and containers): >>> music_voice = abjad.Voice("c'4 d' e' f'", name="MusicVoice") >>> container = abjad.BeforeGraceContainer("cs'16") >>> abjad.attach(container, music_voice[1]) >>> obgc = abjad.on_beat_grace_container("g'16 gs' a' as'", music_voice[2:3]) >>> abjad.attach(abjad.Articulation(">"), obgc[0]) >>> container = abjad.AfterGraceContainer("fs'16") >>> abjad.attach(container, music_voice[3]) >>> staff = abjad.Staff([music_voice]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', staff]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \context Voice = "MusicVoice" { c'4 \grace { cs'16 } d'4 << \context Voice = "On_Beat_Grace_Container" { \set fontSize = #-3 \slash \voiceOne < \tweak font-size 0 \tweak transparent ##t e' g' >16 - \accent [ ( gs'16 a'16 as'16 ) ] } \context Voice = "MusicVoice" { \voiceTwo e'4 } >> \oneVoice \afterGrace f'4 { fs'16 } } } >>> for component in abjad.select.components(staff): ... parentage = abjad.get.parentage(component) ... count = parentage.count(abjad.Staff) ... print(f"{repr(component):30} {repr(count)}") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") 1 Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') 1 Note("c'4") 1 BeforeGraceContainer("cs'16") 1 Note("cs'16") 1 Note("d'4") 1 Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") 1 OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") 1 Chord("<e' g'>16") 1 Note("gs'16") 1 Note("a'16") 1 Note("as'16") 1 Voice("e'4", name='MusicVoice') 1 Note("e'4") 1 Note("f'4") 1 AfterGraceContainer("fs'16") 1 Note("fs'16") 1 """ n = 0 if prototype is None: prototype = _score.Component for component in self: if isinstance(component, prototype): n += 1 return n
[docs] def get(self, prototype=None, n=0) -> _score.Component | None: r""" Gets instance ``n`` of ``prototype`` in parentage. .. container:: example >>> outer_red_voice = abjad.Voice("e''8 d''", name="Red_Voice") >>> inner_red_voice = abjad.Voice("c''4 b' c''8", name="Red_Voice") >>> inner_blue_voice = abjad.Voice("e'4 f' e'8", name="Blue_Voice") >>> container = abjad.Container( ... [inner_red_voice, inner_blue_voice], ... simultaneous=True, ... ) >>> outer_red_voice.append(container) >>> outer_red_voice.extend("d''8") >>> abjad.override(outer_red_voice).NoteHead.color = "#red" >>> command = abjad.VoiceNumber(1) >>> abjad.attach(command, outer_red_voice[0]) >>> abjad.override(inner_blue_voice).NoteHead.color = "#blue" >>> command = abjad.VoiceNumber(2) >>> abjad.attach(command, inner_blue_voice[0]) >>> dynamic = abjad.Dynamic("f") >>> abjad.attach(dynamic, outer_red_voice[0]) >>> abjad.show(outer_red_voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(outer_red_voice) >>> print(string) \context Voice = "Red_Voice" \with { \override NoteHead.color = #red } { \voiceOne e''8 \f d''8 << \context Voice = "Red_Voice" { c''4 b'4 c''8 } \context Voice = "Blue_Voice" \with { \override NoteHead.color = #blue } { \voiceTwo e'4 f'4 e'8 } >> d''8 } .. container:: example >>> leaf = abjad.get.leaf(inner_red_voice, 0) >>> leaf Note("c''4") >>> parentage = abjad.get.parentage(leaf) Returns self when ``n=0``: >>> parentage.get(abjad.Component, 0) Note("c''4") Returns parents with positive ``n``: >>> parentage.get(abjad.Component, 1) Voice("c''4 b'4 c''8", name='Red_Voice') >>> parentage.get(abjad.Component, 2) Container("{ c''4 b'4 c''8 } { e'4 f'4 e'8 }") >>> parentage.get(abjad.Component, 3) Voice("e''8 d''8 { { c''4 b'4 c''8 } { e'4 f'4 e'8 } } d''8", name='Red_Voice') Returns none with ``n`` greater than score depth: >>> parentage.get(abjad.Component, 4) is None True >>> parentage.get(abjad.Component, 5) is None True >>> parentage.get(abjad.Component, 99) is None True Returns score root with ``n=-1``: >>> parentage.get(abjad.Component, -1) Voice("e''8 d''8 { { c''4 b'4 c''8 } { e'4 f'4 e'8 } } d''8", name='Red_Voice') With other negative ``n``: >>> parentage.get(abjad.Component, -2) Container("{ c''4 b'4 c''8 } { e'4 f'4 e'8 }") >>> parentage.get(abjad.Component, -3) Voice("c''4 b'4 c''8", name='Red_Voice') >>> parentage.get(abjad.Component, -4) Note("c''4") Returns none for sufficiently negative ``n``: >>> parentage.get(abjad.Component, -5) is None True >>> parentage.get(abjad.Component, -7) is None True >>> parentage.get(abjad.Component, -99) is None True .. container:: example Works with nested voices and tuplets: >>> leaf = abjad.get.leaf(inner_red_voice, 0) >>> parentage = abjad.get.parentage(leaf) >>> leaf Note("c''4") Nonnegative ``n``: >>> parentage.get(abjad.Voice, 0) Voice("c''4 b'4 c''8", name='Red_Voice') >>> parentage.get(abjad.Voice, 1) Voice("e''8 d''8 { { c''4 b'4 c''8 } { e'4 f'4 e'8 } } d''8", name='Red_Voice') >>> parentage.get(abjad.Voice, 2) is None True >>> parentage.get(abjad.Voice, 9) is None True Negative ``n``: >>> parentage.get(abjad.Voice, -1) Voice("e''8 d''8 { { c''4 b'4 c''8 } { e'4 f'4 e'8 } } d''8", name='Red_Voice') >>> parentage.get(abjad.Voice, -2) Voice("c''4 b'4 c''8", name='Red_Voice') >>> parentage.get(abjad.Voice, -3) is None True >>> parentage.get(abjad.Voice, -99) is None True .. container:: example REGRESSION. Works with grace notes (and containers): >>> music_voice = abjad.Voice("c'4 d' e' f'", name="MusicVoice") >>> container = abjad.BeforeGraceContainer("cs'16") >>> abjad.attach(container, music_voice[1]) >>> obgc = abjad.on_beat_grace_container("g'16 gs' a' as'", music_voice[2:3]) >>> abjad.attach(abjad.Articulation(">"), obgc[0]) >>> container = abjad.AfterGraceContainer("fs'16") >>> abjad.attach(container, music_voice[3]) >>> staff = abjad.Staff([music_voice]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', staff]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \context Voice = "MusicVoice" { c'4 \grace { cs'16 } d'4 << \context Voice = "On_Beat_Grace_Container" { \set fontSize = #-3 \slash \voiceOne < \tweak font-size 0 \tweak transparent ##t e' g' >16 - \accent [ ( gs'16 a'16 as'16 ) ] } \context Voice = "MusicVoice" { \voiceTwo e'4 } >> \oneVoice \afterGrace f'4 { fs'16 } } } >>> for component in abjad.select.components(staff): ... parentage = abjad.get.parentage(component) ... result = parentage.get(abjad.Staff) ... print(f"{repr(component):30} {repr(result)}") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("c'4") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") BeforeGraceContainer("cs'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("cs'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("d'4") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Chord("<e' g'>16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("gs'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("a'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("as'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Voice("e'4", name='MusicVoice') Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("e'4") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("f'4") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") AfterGraceContainer("fs'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") Note("fs'16") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") """ if prototype is None: prototype = (_score.Component,) if not isinstance(prototype, tuple): prototype = (prototype,) if 0 <= n: i = 0 for component in self: if isinstance(component, prototype): if i == n: return component i += 1 else: i = -1 for component in reversed(self): if isinstance(component, prototype): if i == n: return component i -= 1 return None
[docs] def logical_voice(self) -> dict: r""" Gets logical voice. .. container:: example Gets logical voice of note: >>> voice = abjad.Voice("c'4 d'4 e'4 f'4", name="MusicVoice") >>> staff = abjad.Staff([voice], name="Music_Staff") >>> score = abjad.Score([staff], name="Score") >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \context Score = "Score" << \context Staff = "Music_Staff" { \context Voice = "MusicVoice" { c'4 d'4 e'4 f'4 } } >> >>> note = voice[0] >>> parentage = abjad.get.parentage(note) >>> parentage.logical_voice() {'score': "Score-'Score'", 'staff group': '', 'staff': "Staff-'Music_Staff'", 'voice': "Voice-'MusicVoice'"} .. container:: example REGRESSION. Works with grace notes (and containers): >>> voice = abjad.Voice("c'4 d' e' f'", name="MusicVoice") >>> container_1 = abjad.BeforeGraceContainer("cs'16") >>> abjad.attach(container_1, voice[1]) >>> container_2 = abjad.AfterGraceContainer("fs'16") >>> abjad.attach(container_2, voice[3]) >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \context Voice = "MusicVoice" { c'4 \grace { cs'16 } d'4 e'4 \afterGrace f'4 { fs'16 } } >>> abjad.get.parentage(voice).logical_voice() {'score': '', 'staff group': '', 'staff': '', 'voice': "Voice-'MusicVoice'"} >>> abjad.get.parentage(container_1).logical_voice() {'score': '', 'staff group': '', 'staff': '', 'voice': "Voice-'MusicVoice'"} >>> abjad.get.parentage(container_1[0]).logical_voice() {'score': '', 'staff group': '', 'staff': '', 'voice': "Voice-'MusicVoice'"} >>> abjad.get.parentage(container_2).logical_voice() {'score': '', 'staff group': '', 'staff': '', 'voice': "Voice-'MusicVoice'"} >>> abjad.get.parentage(container_2[0]).logical_voice() {'score': '', 'staff group': '', 'staff': '', 'voice': "Voice-'MusicVoice'"} """ keys = ("score", "staff group", "staff", "voice") logical_voice = dict.fromkeys(keys, "") for component in self: if isinstance(component, _score.Voice): if not logical_voice["voice"]: logical_voice["voice"] = self._id_string(component) elif isinstance(component, _score.Staff): if not logical_voice["staff"]: logical_voice["staff"] = self._id_string(component) # explicit staff demands a nested voice: # if no explicit voice has been found, # create implicit voice here with random integer if not logical_voice["voice"]: logical_voice["voice"] = str(id(component)) elif isinstance(component, _score.StaffGroup): if not logical_voice["staff group"]: logical_voice["staff group"] = self._id_string(component) elif isinstance(component, _score.Score): if not logical_voice["score"]: logical_voice["score"] = self._id_string(component) logical_voice_ = dict(logical_voice) return logical_voice_
[docs] def score_index(self) -> tuple[int | str, ...]: r""" Gets score index. .. container:: example >>> staff_1 = abjad.Staff(r"\times 2/3 { c''2 b'2 a'2 }") >>> staff_2 = abjad.Staff("c'2 d'2") >>> score = abjad.Score([staff_1, staff_2]) >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff { \tuplet 3/2 { c''2 b'2 a'2 } } \new Staff { c'2 d'2 } >> >>> for component in abjad.select.components(score): ... parentage = abjad.get.parentage(component) ... component, parentage.score_index() ... (Score("{ { 2/3 c''2 b'2 a'2 } } { c'2 d'2 }", simultaneous=True), ()) (Staff("{ 2/3 c''2 b'2 a'2 }"), (0,)) (Tuplet('3:2', "c''2 b'2 a'2"), (0, 0)) (Note("c''2"), (0, 0, 0)) (Note("b'2"), (0, 0, 1)) (Note("a'2"), (0, 0, 2)) (Staff("c'2 d'2"), (1,)) (Note("c'2"), (1, 0)) (Note("d'2"), (1, 1)) Score root sets score index to ``()``. .. container:: example REGRESSION. Works with grace notes (and containers): >>> music_voice = abjad.Voice("c'4 d' e' f'", name="MusicVoice") >>> container = abjad.BeforeGraceContainer("cs'16") >>> abjad.attach(container, music_voice[1]) >>> obgc = abjad.on_beat_grace_container("g'16 gs' a' as'", music_voice[2:3]) >>> abjad.attach(abjad.Articulation(">"), obgc[0]) >>> container = abjad.AfterGraceContainer("fs'16") >>> abjad.attach(container, music_voice[3]) >>> staff = abjad.Staff([music_voice]) >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', staff]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \context Voice = "MusicVoice" { c'4 \grace { cs'16 } d'4 << \context Voice = "On_Beat_Grace_Container" { \set fontSize = #-3 \slash \voiceOne < \tweak font-size 0 \tweak transparent ##t e' g' >16 - \accent [ ( gs'16 a'16 as'16 ) ] } \context Voice = "MusicVoice" { \voiceTwo e'4 } >> \oneVoice \afterGrace f'4 { fs'16 } } } >>> for component in abjad.select.components(staff): ... parentage = abjad.get.parentage(component) ... score_index = parentage.score_index() ... print(f"{repr(component):30} {repr(score_index)}") Staff("{ c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4 }") () Voice("c'4 d'4 { { <e' g'>16 gs'16 a'16 as'16 } { e'4 } } f'4", name='MusicVoice') (0,) Note("c'4") (0, 0) BeforeGraceContainer("cs'16") (0, 0, 1, '-G') Note("cs'16") (0, 0, 1, '-G', 0) Note("d'4") (0, 1) Container("{ <e' g'>16 gs'16 a'16 as'16 } { e'4 }") (0, 2) OnBeatGraceContainer("<e' g'>16 gs'16 a'16 as'16") (0, 2, 0) Chord("<e' g'>16") (0, 2, 0, 0) Note("gs'16") (0, 2, 0, 1) Note("a'16") (0, 2, 0, 2) Note("as'16") (0, 2, 0, 3) Voice("e'4", name='MusicVoice') (0, 2, 1) Note("e'4") (0, 2, 1, 0) Note("f'4") (0, 3) AfterGraceContainer("fs'16") (0, 0, 3, '+G') Note("fs'16") (0, 0, 3, '+G', 0) """ result: list[int | str] = [] current = self[0] for parent in self[1:]: if isinstance(current, _score.BeforeGraceContainer): tuple_ = type(self)(current._main_leaf).score_index() list_ = list(tuple_) + ["-G"] result[0:0] = list_ elif isinstance(current, _score.AfterGraceContainer): tuple_ = type(self)(current._main_leaf).score_index() list_ = list(tuple_) + ["+G"] result[0:0] = list_ else: index = parent.index(current) result.insert(0, index) current = parent return tuple(result)