import dataclasses
import typing
from . import _getlib
from . import bind as _bind
from . import get as _get
from . import indicators as _indicators
from . import instruments as _instruments
from . import iterate as _iterate
from . import pcollections as _pcollections
from . import pitch as _pitch
from . import score as _score
[docs]
def iterate_out_of_range(argument) -> typing.Iterator[_score.Leaf]:
r"""
Iterates out-of-range notes and chords in ``argument``.
.. container:: example
>>> staff = abjad.Staff("c'8 r8 <d fs>8 r8")
>>> violin = abjad.Violin()
>>> abjad.attach(violin, staff[0])
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
c'8
r8
<d fs>8
r8
}
>>> for leaf in abjad.iterpitches.iterate_out_of_range(staff):
... leaf
...
Chord('<d fs>8')
"""
for leaf in _iterate.leaves(argument, pitched=True):
instrument = _getlib._get_effective(leaf, _instruments.Instrument)
if instrument is None:
raise ValueError("no instrument found.")
if not sounding_pitches_are_in_range(leaf, instrument.pitch_range):
yield leaf
[docs]
def respell_with_flats(argument) -> None:
r"""
Respells pitches in ``argument`` with flats.
.. container:: example
>>> staff = abjad.Staff("cs'8 ds' es' fs' gs' as' bs'4")
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
cs'8
ds'8
es'8
fs'8
gs'8
as'8
bs'4
}
>>> abjad.iterpitches.respell_with_flats(staff)
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
df'8
ef'8
f'8
gf'8
af'8
bf'8
c''4
}
"""
for leaf in _iterate.leaves(argument):
if isinstance(leaf, _score.Note):
assert leaf.written_pitch is not None
leaf.written_pitch = leaf.written_pitch.respell(accidental="flats")
elif isinstance(leaf, _score.Chord):
for note_head in leaf.note_heads:
pitch = note_head.written_pitch.respell(accidental="flats")
note_head.written_pitch = pitch
[docs]
def respell_with_sharps(argument) -> None:
r"""
Respells pitches in ``argument`` with sharps.
.. container:: example
>>> staff = abjad.Staff("cf'8 df' ef' ff' gf' af' bf'4")
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
cf'8
df'8
ef'8
ff'8
gf'8
af'8
bf'4
}
>>> abjad.iterpitches.respell_with_sharps(staff)
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
b8
cs'8
ds'8
e'8
fs'8
gs'8
as'4
}
"""
for leaf in _iterate.leaves(argument):
if isinstance(leaf, _score.Note):
assert leaf.written_pitch is not None
leaf.written_pitch = leaf.written_pitch.respell(accidental="sharps")
elif isinstance(leaf, _score.Chord):
for note_head in leaf.note_heads:
pitch = note_head.written_pitch.respell(accidental="sharps")
note_head.written_pitch = pitch
[docs]
def sounding_pitches_are_in_range(argument, pitch_range) -> bool:
"""
Returns true when all pitches in ``argument`` sound within ``pitch_range``.
"""
assert isinstance(pitch_range, _pcollections.PitchRange), repr(pitch_range)
if isinstance(argument, int | float):
pitch = _pitch.NamedPitch(argument)
return pitch in pitch_range
if isinstance(argument, _pitch.Pitch):
return argument in pitch_range
if hasattr(argument, "written_pitch"):
sounding_pitch = _getlib._get_sounding_pitch(argument)
return sounding_pitch in pitch_range
if hasattr(argument, "written_pitches"):
sounding_pitches = _getlib._get_sounding_pitches(argument)
return all(_ in pitch_range for _ in sounding_pitches)
pitches = list(_iterate.pitches(argument))
if pitches:
return all(_ in pitch_range for _ in pitches)
else:
try:
return all(_ in pitch_range for _ in argument)
except TypeError:
return False
return False
[docs]
def transpose_from_sounding_pitch(argument) -> None:
r"""
Transpose notes and chords in ``argument`` from sounding pitch to written pitch.
.. container:: example
>>> staff = abjad.Staff("<c' e' g'>4 d'4 r4 e'4")
>>> clarinet = abjad.ClarinetInBFlat()
>>> abjad.attach(clarinet, staff[0])
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
<c' e' g'>4
d'4
r4
e'4
}
>>> abjad.iterpitches.transpose_from_sounding_pitch(staff)
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
<d' fs' a'>4
e'4
r4
fs'4
}
"""
for leaf in _iterate.leaves(argument, pitched=True):
instrument = _getlib._get_effective(leaf, _instruments.Instrument)
if not instrument:
continue
sounding_pitch = instrument.middle_c_sounding_pitch
interval = _pitch.NamedPitch("C4") - sounding_pitch
interval *= -1
if hasattr(leaf, "note_head"):
pitch = leaf.written_pitch
pitch = interval.transpose(pitch)
leaf.written_pitch = pitch
elif hasattr(leaf, "note_heads"):
pitches = [interval.transpose(pitch) for pitch in leaf.written_pitches]
for note_head, pitch in zip(leaf.note_heads, pitches, strict=True):
note_head.written_pitch = pitch
wrapper = _get.indicator(leaf, _indicators.StartTrillSpan, unwrap=False)
if wrapper is not None:
start_trill_span = wrapper.unbundle_indicator()
new_pitch = interval.transpose(start_trill_span.pitch)
new_start_trill_span = dataclasses.replace(
start_trill_span, pitch=new_pitch
)
wrapper_tag = wrapper.tag
_bind.detach(wrapper, leaf)
if wrapper.bundled():
new_bundle = dataclasses.replace(
wrapper.get_item(), indicator=new_start_trill_span
)
_bind.attach(new_bundle, leaf, tag=wrapper_tag)
else:
_bind.attach(new_start_trill_span, leaf, tag=wrapper_tag)
[docs]
def transpose_from_written_pitch(argument) -> None:
r"""
Transposes notes and chords in ``argument`` from sounding pitch to written pitch.
.. container:: example
>>> staff = abjad.Staff("<c' e' g'>4 d'4 r4 e'4")
>>> clarinet = abjad.ClarinetInBFlat()
>>> abjad.attach(clarinet, staff[0])
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
<c' e' g'>4
d'4
r4
e'4
}
>>> abjad.iterpitches.transpose_from_written_pitch(staff)
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
<bf d' f'>4
c'4
r4
d'4
}
"""
for leaf in _iterate.leaves(argument, pitched=True):
instrument = _getlib._get_effective(leaf, _instruments.Instrument)
if not instrument:
continue
sounding_pitch = instrument.middle_c_sounding_pitch
interval = _pitch.NamedPitch("C4") - sounding_pitch
if hasattr(leaf, "note_head"):
written_pitch = leaf.written_pitch
written_pitch = interval.transpose(written_pitch)
leaf.written_pitch = written_pitch
elif hasattr(leaf, "note_heads"):
pitches = [interval.transpose(pitch) for pitch in leaf.written_pitches]
for note_head, pitch in zip(leaf.note_heads, pitches, strict=True):
note_head.written_pitch = pitch
wrapper = _get.indicator(leaf, _indicators.StartTrillSpan, unwrap=False)
if wrapper is not None:
start_trill_span = wrapper.unbundle_indicator()
new_pitch = interval.transpose(start_trill_span.pitch)
new_start_trill_span = dataclasses.replace(
start_trill_span, pitch=new_pitch
)
_bind.detach(wrapper, leaf)
if wrapper.bundled():
new_bundle = dataclasses.replace(
wrapper.get_item, indicator=new_start_trill_span
)
_bind.attach(new_bundle, leaf)
else:
_bind.attach(new_start_trill_span, leaf)