import collections
import copy as python_copy
import itertools
from . import _iterlib
from . import bind as _bind
from . import duration as _duration
from . import enums as _enums
from . import exceptions as _exceptions
from . import get as _get
from . import indicators as _indicators
from . import iterate as _iterate
from . import makers as _makers
from . import parentage as _parentage
from . import pitch as _pitch
from . import score as _score
from . import select as _select
from . import sequence as _sequence
from . import spanners as _spanners
from . import tag as _tag
def _are_contiguous_logical_voice(
selection, prototype=None, *, ignore_before_after_grace=None
) -> bool:
r"""
Is true when items in selection are contiguous components in the same logical voice.
.. container:: example
>>> staff = abjad.Staff("c'4 d'4 e'4 f'4")
>>> abjad.mutate._are_contiguous_logical_voice(staff[:])
True
>>> staves = [staff[0], staff[-1]]
>>> abjad.mutate._are_contiguous_logical_voice(staves)
False
.. container:: example
REGRESSION. Before-grace music may be ignored:
>>> voice = abjad.Voice("c'4 d' e' f'")
>>> container = abjad.BeforeGraceContainer("cs'16")
>>> abjad.attach(container, voice[1])
>>> abjad.show(voice) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(voice)
>>> print(string)
\new Voice
{
c'4
\grace {
cs'16
}
d'4
e'4
f'4
}
>>> voice[:]
[Note("c'4"), Note("d'4"), Note("e'4"), Note("f'4")]
>>> abjad.mutate._are_contiguous_logical_voice(voice[:])
False
>>> abjad.mutate._are_contiguous_logical_voice(
... voice[:],
... ignore_before_after_grace=True
... )
True
After-grace music may be ignored, too:
>>> voice = abjad.Voice("c'4 d' e' f'")
>>> container = abjad.AfterGraceContainer("cs'16")
>>> abjad.attach(container, voice[0])
>>> abjad.show(voice) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(voice)
>>> print(string)
\new Voice
{
\afterGrace
c'4
{
cs'16
}
d'4
e'4
f'4
}
>>> voice[:]
[Note("c'4"), Note("d'4"), Note("e'4"), Note("f'4")]
>>> abjad.mutate._are_contiguous_logical_voice(voice[:])
False
>>> abjad.mutate._are_contiguous_logical_voice(
... voice[:],
... ignore_before_after_grace=True
... )
True
"""
if not isinstance(selection, collections.abc.Sequence):
return False
prototype = prototype or (_score.Component,)
if not isinstance(prototype, tuple):
prototype = (prototype,)
assert isinstance(prototype, tuple)
if len(selection) == 0:
return True
if all(isinstance(_, prototype) and _._parent is None for _ in selection):
return True
first = selection[0]
if not isinstance(first, prototype):
return False
first_parentage = _parentage.Parentage(first)
first_logical_voice = first_parentage.logical_voice()
first_root = first_parentage.root
previous = first
for current in selection[1:]:
current_parentage = _parentage.Parentage(current)
current_logical_voice = current_parentage.logical_voice()
# false if wrong type of component found
if not isinstance(current, prototype):
return False
# false if in different logical voices
if current_logical_voice != first_logical_voice:
return False
# false if components are in same score and are discontiguous
if current_parentage.root == first_root:
if not _immediately_precedes(
previous,
current,
ignore_before_after_grace=ignore_before_after_grace,
):
return False
previous = current
return True
def _are_contiguous_same_parent(
self, prototype=None, *, ignore_before_after_grace=None
) -> bool:
r"""
Is true when items in selection are all contiguous components in the same parent.
.. container:: example
>>> staff = abjad.Staff("c'4 d'4 e'4 f'4")
>>> abjad.mutate._are_contiguous_same_parent(staff[:])
True
>>> staves = [staff[0], staff[-1]]
>>> abjad.mutate._are_contiguous_same_parent(staves)
False
.. container:: example
REGRESSION. Before-grace music music may be ignored:
>>> voice = abjad.Voice("c'4 d' e' f'")
>>> container = abjad.BeforeGraceContainer("cs'16")
>>> abjad.attach(container, voice[1])
>>> abjad.show(voice) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(voice)
>>> print(string)
\new Voice
{
c'4
\grace {
cs'16
}
d'4
e'4
f'4
}
>>> voice[:]
[Note("c'4"), Note("d'4"), Note("e'4"), Note("f'4")]
>>> abjad.mutate._are_contiguous_same_parent(voice[:])
False
>>> abjad.mutate._are_contiguous_same_parent(
... voice[:],
... ignore_before_after_grace=True
... )
True
After-grace music may be ignored, too:
>>> voice = abjad.Voice("c'4 d' e' f'")
>>> container = abjad.AfterGraceContainer("cs'16")
>>> abjad.attach(container, voice[0])
>>> abjad.show(voice) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(voice)
>>> print(string)
\new Voice
{
\afterGrace
c'4
{
cs'16
}
d'4
e'4
f'4
}
>>> voice[:]
[Note("c'4"), Note("d'4"), Note("e'4"), Note("f'4")]
>>> abjad.mutate._are_contiguous_same_parent(voice[:])
False
>>> abjad.mutate._are_contiguous_same_parent(
... voice[:],
... ignore_before_after_grace=True
... )
True
"""
prototype = prototype or (_score.Component,)
if not isinstance(prototype, tuple):
prototype = (prototype,)
assert isinstance(prototype, tuple)
if len(self) == 0:
return True
if all(isinstance(_, prototype) and _._parent is None for _ in self):
return True
first = self[0]
if not isinstance(first, prototype):
return False
first_parent = first._parent
same_parent = True
strictly_contiguous = True
previous = first
for current in self[1:]:
if not isinstance(current, prototype):
return False
if current._parent is not first_parent:
same_parent = False
if not _immediately_precedes(
previous,
current,
ignore_before_after_grace=ignore_before_after_grace,
):
strictly_contiguous = False
if current._parent is not None and (not same_parent or not strictly_contiguous):
return False
previous = current
return True
def _attach_tie_to_leaves(selection):
for leaf in selection[:-1]:
_bind.detach(_indicators.Tie, leaf)
_bind.attach(_indicators.Tie(), leaf)
def _copy_selection(selection):
assert _are_contiguous_logical_voice(selection)
new_components = []
for component in selection:
if isinstance(component, _score.Container):
new_component = component._copy_with_children()
else:
new_component = component.__copy__()
new_components.append(new_component)
new_components = type(selection)(new_components)
return new_components
def _extract(component):
selection = [component]
parent, start, stop = _get_parent_and_start_stop_indices(selection)
if parent is not None:
components = list(getattr(component, "components", ()))
parent.__setitem__(slice(start, stop + 1), components)
return component
def _fuse(components, *, tag=None):
assert _are_contiguous_logical_voice(components)
if all(isinstance(_, _score.Leaf) for _ in components):
return _fuse_leaves(components, tag=tag)
elif all(isinstance(_, _score.Tuplet) for _ in components):
return _fuse_tuplets(components, tag=tag)
else:
raise Exception(f"can only fuse leaves and tuplets (not {components}).")
def _fuse_leaves(leaves, *, tag=None):
assert all(isinstance(_, _score.Leaf) for _ in leaves)
assert _are_contiguous_logical_voice(leaves)
if len(leaves) <= 1:
return leaves
originally_tied = leaves[-1]._has_indicator(_indicators.Tie)
total_preprolated = sum(_._get_preprolated_duration() for _ in leaves)
for leaf in leaves[1:]:
parent = leaf._parent
if parent:
index = parent.index(leaf)
del parent[index]
result = _set_leaf_duration(leaves[0], total_preprolated, tag=tag)
if not originally_tied:
last_leaf = _select.leaf(result, -1)
_bind.detach(_indicators.Tie, last_leaf)
return result
# TODO: remove because unused?
def _fuse_leaves_by_immediate_parent(leaves, *, tag=None):
result = []
parts = _get_leaves_grouped_by_immediate_parents(leaves)
for part in parts:
fused = _fuse(part, tag=tag)
result.append(fused)
return result
def _fuse_tuplets(tuplets, *, tag=None):
assert _are_contiguous_same_parent(tuplets, prototype=_score.Tuplet)
if len(tuplets) == 0:
return None
first = tuplets[0]
first_multiplier = first.multiplier
for tuplet in tuplets[1:]:
if tuplet.multiplier != first_multiplier:
raise ValueError("tuplets must carry same multiplier.")
assert isinstance(first, _score.Tuplet)
new_tuplet = _score.Tuplet(first_multiplier, [], tag=tag)
wrapped = False
if _get.parentage(tuplets[0]).root is not _get.parentage(tuplets[-1]).root:
dummy_container = _score.Container(tuplets)
wrapped = True
swap(tuplets, new_tuplet)
if wrapped:
del dummy_container[:]
return new_tuplet
def _get_leaves_grouped_by_immediate_parents(leaves):
result = []
pairs_generator = itertools.groupby(leaves, lambda _: id(_._parent))
for key, values_generator in pairs_generator:
group = list(values_generator)
result.append(group)
return result
def _give_components_to_empty_container(selection, container):
assert _are_contiguous_same_parent(selection)
assert isinstance(container, _score.Container)
assert not container
components = []
for component in selection:
components.extend(getattr(component, "components", ()))
container._components.extend(components)
_set_parents(container)
def _get_parent_and_start_stop_indices(selection, ignore_before_after_grace=None):
assert _are_contiguous_same_parent(
selection, ignore_before_after_grace=ignore_before_after_grace
)
if selection:
first, last = selection[0], selection[-1]
parent = first._parent
if parent is not None:
first_index = parent.index(first)
last_index = parent.index(last)
return parent, first_index, last_index
return None, None, None
def _give_position_in_parent_to_container(selection, container):
assert _are_contiguous_same_parent(selection)
assert isinstance(container, _score.Container)
parent, start, stop = _get_parent_and_start_stop_indices(selection)
if parent is not None:
parent._components.__setitem__(slice(start, start), [container])
container._set_parent(parent)
for component in selection:
component._set_parent(None)
def _immediately_precedes(component_1, component_2, ignore_before_after_grace=None):
successors = []
current = component_1
# do not include OnBeatGraceContainer here because
# OnBeatGraceContainer is a proper container
grace_prototype = (_score.AfterGraceContainer, _score.BeforeGraceContainer)
while current is not None:
sibling = _iterlib._get_sibling_with_graces(current, 1)
while (
ignore_before_after_grace
and sibling is not None
and isinstance(sibling._parent, grace_prototype)
):
sibling = _iterlib._get_sibling_with_graces(sibling, 1)
if sibling is None:
current = current._parent
else:
descendants = sibling._get_descendants_starting_with()
successors = descendants
break
return component_2 in successors
def _set_leaf_duration(leaf, new_duration, *, tag=None):
new_duration = _duration.Duration(new_duration)
if leaf.multiplier is not None:
multiplier = new_duration.__div__(leaf.written_duration)
leaf.multiplier = _duration.pair(multiplier)
return [leaf]
try:
leaf.written_duration = new_duration
return [leaf]
except _exceptions.AssignabilityError:
pass
components = _makers.make_notes(0, new_duration, tag=tag)
new_leaves = _select.leaves(components)
following_leaf_count = len(new_leaves) - 1
following_leaves = []
for i in range(following_leaf_count):
following_leaf = copy(leaf)
for indicator in _get.indicators(following_leaf):
if i != following_leaf_count - 1:
if getattr(indicator, "time_orientation", _enums.LEFT) != _enums.MIDDLE:
_bind.detach(indicator, following_leaf)
elif (
getattr(indicator, "time_orientation", _enums.LEFT) != _enums.RIGHT
and getattr(indicator, "time_orientation", _enums.LEFT) != _enums.MIDDLE
):
_bind.detach(indicator, following_leaf)
_bind.detach(_score.BeforeGraceContainer, following_leaf)
following_leaves.append(following_leaf)
if following_leaf_count > 0:
for indicator in _get.indicators(leaf):
if getattr(indicator, "time_orientation", _enums.LEFT) == _enums.RIGHT:
_bind.detach(indicator, leaf)
all_leaves = [leaf] + following_leaves
assert len(all_leaves) == len(new_leaves)
for all_leaf, new_leaf in zip(all_leaves, new_leaves):
all_leaf.written_duration = new_leaf.written_duration
logical_tie = _get.logical_tie(leaf)
logical_tie_leaves = list(logical_tie)
for leaf_ in logical_tie:
_bind.detach(_indicators.Tie, leaf_)
_bind.detach(_indicators.RepeatTie, leaf_)
if leaf._parent is not None:
index = leaf._parent.index(leaf)
next_ = index + 1
leaf._parent[next_:next_] = following_leaves
index = logical_tie_leaves.index(leaf)
next_ = index + 1
logical_tie_leaves[next_:next_] = following_leaves
if 1 < len(logical_tie_leaves) and isinstance(leaf, _score.Note | _score.Chord):
_spanners.tie(logical_tie_leaves)
if isinstance(components[0], _score.Leaf):
assert isinstance(all_leaves, list)
return all_leaves
else:
assert isinstance(components[0], _score.Tuplet)
assert len(components) == 1
tuplet = components[0]
multiplier = tuplet.multiplier
tuplet = _score.Tuplet(multiplier, [])
assert isinstance(all_leaves, list)
wrap(all_leaves, tuplet)
return [tuplet]
def _set_parents(container):
for component in container:
component._set_parent(container)
def _split_container_at_index(CONTAINER, i):
"""
Splits container to the left of index ``i``.
Preserves tuplet multiplier when container is a tuplet.
Preserves time signature denominator when container is a measure.
Resizes resizable containers.
Returns split parts.
"""
# partition my components
left_components = CONTAINER[:i]
right_components = CONTAINER[i:]
# instantiate new left and right containers
if isinstance(CONTAINER, _score.Tuplet):
multiplier = CONTAINER.multiplier
left = type(CONTAINER)(multiplier, [])
wrap(left_components, left)
right = type(CONTAINER)(multiplier, [])
wrap(right_components, right)
else:
left = CONTAINER.__copy__()
wrap(left_components, left)
right = CONTAINER.__copy__()
wrap(right_components, right)
# save left and right containers together for iteration
halves = (left, right)
nonempty_halves = [half for half in halves if len(half)]
# incorporate left and right parents in score if possible
selection = [CONTAINER]
parent, start, stop = _get_parent_and_start_stop_indices(selection)
if parent is not None:
parent._components.__setitem__(slice(start, stop + 1), nonempty_halves)
for part in nonempty_halves:
part._set_parent(parent)
else:
left._set_parent(None)
right._set_parent(None)
# return new left and right containers
return halves
def _split_container_by_duration(CONTAINER, duration, *, tag=None):
if CONTAINER.simultaneous:
return _split_simultaneous_by_duration(CONTAINER, duration=duration, tag=tag)
duration = _duration.Duration(duration)
assert 0 <= duration, repr(duration)
if duration == 0:
# TODO: disallow and raise Exception
return [], CONTAINER
# get split point score offset
timespan = _get.timespan(CONTAINER)
global_split_point = timespan.start_offset + duration
# get any duration-crossing descendents
cross_offset = timespan.start_offset + duration
duration_crossing_descendants = []
for descendant in _get.descendants(CONTAINER):
timespan = _get.timespan(descendant)
start_offset = timespan.start_offset
stop_offset = timespan.stop_offset
if start_offset < cross_offset < stop_offset:
duration_crossing_descendants.append(descendant)
# any duration-crossing leaf will be at end of list
bottom = duration_crossing_descendants[-1]
did_split_leaf = False
# if split point necessitates leaf split
if isinstance(bottom, _score.Leaf):
assert isinstance(bottom, _score.Leaf)
original_bottom_parent = bottom._parent
did_split_leaf = True
timespan = _get.timespan(bottom)
split_point_in_bottom = global_split_point - timespan.start_offset
new_leaves = _split_leaf_by_durations(
bottom,
[split_point_in_bottom],
tag=tag,
)
if new_leaves[0]._parent is not original_bottom_parent:
new_leaves_tuplet_wrapper = new_leaves[0]._parent
assert isinstance(new_leaves_tuplet_wrapper, _score.Tuplet)
assert new_leaves_tuplet_wrapper._parent is original_bottom_parent
_split_container_by_duration(
new_leaves_tuplet_wrapper,
split_point_in_bottom,
tag=tag,
)
for leaf in new_leaves:
timespan = _get.timespan(leaf)
if timespan.stop_offset == global_split_point:
leaf_left_of_split = leaf
if timespan.start_offset == global_split_point:
leaf_right_of_split = leaf
duration_crossing_containers = duration_crossing_descendants[:-1]
assert len(duration_crossing_containers)
# if split point falls between leaves
# then find leaf to immediate right of split point
# in order to start upward crawl through duration-crossing containers
else:
duration_crossing_containers = duration_crossing_descendants[:]
for leaf in _iterate.leaves(bottom):
timespan = _get.timespan(leaf)
if timespan.start_offset == global_split_point:
leaf_right_of_split = leaf
leaf_left_of_split = _get.leaf(leaf, -1)
break
else:
raise Exception("can not split empty container {bottom!r}.")
assert leaf_left_of_split is not None
assert leaf_right_of_split is not None
# find component to right of split
# that is also immediate child of last duration-crossing container
for component in leaf_right_of_split._get_parentage():
if component._parent is duration_crossing_containers[-1]:
highest_level_component_right_of_split = component
break
else:
raise ValueError("should not be able to get here.")
# crawl back up through duration-crossing containers and split each
previous = highest_level_component_right_of_split
for container in reversed(duration_crossing_containers):
assert isinstance(container, _score.Container)
index = container.index(previous)
left, right = _split_container_at_index(container, index)
previous = right
# reapply tie here if crawl above killed tie applied to leaves
if did_split_leaf:
if isinstance(leaf_left_of_split, _score.Note):
if (
_get.parentage(leaf_left_of_split).root
is _get.parentage(leaf_right_of_split).root
):
leaves_around_split = (
leaf_left_of_split,
leaf_right_of_split,
)
_attach_tie_to_leaves(leaves_around_split)
# return list-wrapped halves of container
return [left], [right]
def _split_simultaneous_by_duration(CONTAINER, duration, *, tag=None):
assert CONTAINER.simultaneous
left_components, right_components = [], []
for component in CONTAINER[:]:
halves = _split_container_by_duration(component, duration=duration, tag=tag)
left_components_, right_components_ = halves
left_components.extend(left_components_)
right_components.extend(right_components_)
left_container = CONTAINER.__copy__()
right_container = CONTAINER.__copy__()
left_container.extend(left_components)
right_container.extend(right_components)
if _get.parentage(CONTAINER).parent is not None:
containers = [left_container, right_container]
replace(CONTAINER, containers)
# return list-wrapped halves of container
return [left_container], [right_container]
def _split_leaf_by_durations(leaf, durations, *, cyclic=False, tag=None):
durations = [_duration.Duration(_) for _ in durations]
leaf_duration = _get.duration(leaf)
if cyclic:
durations = _sequence.repeat_to_weight(durations, leaf_duration)
if sum(durations) < leaf_duration:
last_duration = leaf_duration - sum(durations)
durations = list(durations)
durations.append(last_duration)
durations = _sequence.truncate(durations, weight=leaf_duration)
originally_tied = leaf._has_indicator(_indicators.Tie)
originally_repeat_tied = leaf._has_indicator(_indicators.RepeatTie)
result_selections = []
# detach grace containers
before_grace_container = leaf._before_grace_container
if before_grace_container is not None:
_bind.detach(before_grace_container, leaf)
after_grace_container = leaf._after_grace_container
if after_grace_container is not None:
_bind.detach(after_grace_container, leaf)
# do other things
leaf_prolation = _get.parentage(leaf).prolation
for duration in durations:
new_leaf = python_copy.copy(leaf)
preprolated_duration = duration / leaf_prolation
selection = _set_leaf_duration(new_leaf, preprolated_duration, tag=tag)
result_selections.append(selection)
result_components = _sequence.flatten(result_selections, depth=-1)
result_leaves = _select.leaves(result_components, grace=False)
assert all(isinstance(_, _score.Component) for _ in result_components)
assert all(isinstance(_, _score.Leaf) for _ in result_leaves)
# strip result leaves of all indicators
for leaf_ in result_leaves:
_bind.detach(object, leaf_)
# replace leaf with flattened result
if _get.parentage(leaf).parent is not None:
replace(leaf, result_components)
# move indicators
first_result_leaf = result_leaves[0]
last_result_leaf = result_leaves[-1]
for indicator in _get.indicators(leaf):
_bind.detach(indicator, leaf)
direction = getattr(indicator, "time_orientation", _enums.LEFT)
if direction is _enums.LEFT:
_bind.attach(indicator, first_result_leaf)
elif direction == _enums.RIGHT:
_bind.attach(indicator, last_result_leaf)
elif direction == _enums.MIDDLE:
_bind.attach(indicator, first_result_leaf)
indicator_copy = python_copy.copy(indicator)
_bind.attach(indicator_copy, last_result_leaf)
else:
raise ValueError(direction)
# reattach grace containers
if before_grace_container is not None:
_bind.attach(before_grace_container, first_result_leaf)
if after_grace_container is not None:
_bind.attach(after_grace_container, last_result_leaf)
# fuse tuplets
if isinstance(result_components[0], _score.Tuplet):
fuse(result_components)
# tie split notes
if isinstance(leaf, _score.Note | _score.Chord) and 1 < len(result_leaves):
_attach_tie_to_leaves(result_leaves)
if originally_repeat_tied and not result_leaves[0]._has_indicator(
_indicators.RepeatTie
):
_bind.attach(_indicators.RepeatTie(), result_leaves[0])
if originally_tied and not result_leaves[-1]._has_indicator(_indicators.Tie):
_bind.attach(_indicators.Tie(), result_leaves[-1])
assert isinstance(result_leaves, list)
assert all(isinstance(_, _score.Leaf) for _ in result_leaves)
return result_leaves
[docs]
def copy(argument, n=1) -> list[_score.Component]:
r"""
Copies argument.
.. container:: example
Copies explicit clefs:
>>> staff = abjad.Staff("c'8 cs'8 d'8 ef'8 e'8 f'8 fs'8 g'8")
>>> clef = abjad.Clef('treble')
>>> abjad.attach(clef, staff[0])
>>> clef = abjad.Clef('bass')
>>> abjad.attach(clef, staff[4])
>>> copied_notes = abjad.mutate.copy(staff[:2])
>>> staff.extend(copied_notes)
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\clef "treble"
c'8
cs'8
d'8
ef'8
\clef "bass"
e'8
f'8
fs'8
g'8
\clef "treble"
c'8
cs'8
}
.. container:: example
Does not copy implicit clefs:
>>> staff = abjad.Staff("c'8 cs'8 d'8 ef'8 e'8 f'8 fs'8 g'8")
>>> clef = abjad.Clef('treble')
>>> abjad.attach(clef, staff[0])
>>> clef = abjad.Clef('bass')
>>> abjad.attach(clef, staff[4])
>>> copied_notes = abjad.mutate.copy(staff[2:4])
>>> staff.extend(copied_notes)
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\clef "treble"
c'8
cs'8
d'8
ef'8
\clef "bass"
e'8
f'8
fs'8
g'8
d'8
ef'8
}
.. container:: example
Copy components one time:
>>> staff = abjad.Staff(r"c'8 d'8 e'8 f'8")
>>> staff.extend(r"g'8 a'8 b'8 c''8")
>>> score = abjad.Score([staff], name="Score")
>>> time_signature = abjad.TimeSignature((2, 4))
>>> abjad.attach(time_signature, staff[0])
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\time 2/4
c'8
d'8
e'8
f'8
g'8
a'8
b'8
c''8
}
>>> selection = staff[2:4]
>>> result = abjad.mutate.copy(selection)
>>> new_staff = abjad.Staff(result)
>>> abjad.show(new_staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(new_staff)
>>> print(string)
\new Staff
{
e'8
f'8
}
>>> staff[2] is new_staff[0]
False
"""
if isinstance(argument, _score.Component):
selection = [argument]
else:
selection = argument
if n == 1:
result = _copy_selection(selection)
if isinstance(argument, _score.Component):
if len(result) == 1:
result = result[0]
return result
else:
result = []
for _ in range(n):
result_ = copy(argument)
result.append(result_)
return result
[docs]
def eject_contents(container: _score.Container) -> list[_score.Component]:
r"""
Ejects ``container`` contents.
.. container:: example
Ejects leaves from container:
>>> container = abjad.Container("c'4 ~ c'4 d'4 ~ d'4")
>>> abjad.show(container) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(container)
>>> print(string)
{
c'4
~
c'4
d'4
~
d'4
}
>>> leaves = abjad.mutate.eject_contents(container)
>>> leaves
[Note("c'4"), Note("c'4"), Note("d'4"), Note("d'4")]
Leaves can be added to a new container:
>>> staff = abjad.Staff(leaves, lilypond_type="RhythmicStaff")
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new RhythmicStaff
{
c'4
~
c'4
d'4
~
d'4
}
Old container is empty:
>>> container
Container()
"""
assert isinstance(container, _score.Container), repr(container)
components = container[:]
container[:] = []
return components
[docs]
def fuse(argument) -> _score.Tuplet | list[_score.Leaf]:
r"""
Fuses ``argument``.
.. container:: example
Fuses in-score leaves:
>>> staff = abjad.Staff("c'8 d'8 e'8 f'8")
>>> abjad.show(staff) # doctest: +SKIP
>>> abjad.mutate.fuse(staff[1:])
[Note("d'4.")]
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
c'8
d'4.
}
.. container:: example
Fuses parent-contiguous tuplets in selection:
>>> tuplet_1 = abjad.Tuplet((2, 3), "c'8 d' e'")
>>> tuplet_2 = abjad.Tuplet((2, 3), "c'16 d' e'")
>>> voice = abjad.Voice([tuplet_1, tuplet_2])
>>> staff = abjad.Staff([voice])
>>> abjad.beam(tuplet_1[:])
>>> abjad.slur(tuplet_2[:])
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\new Voice
{
\tuplet 3/2
{
c'8
[
d'8
e'8
]
}
\tuplet 3/2
{
c'16
(
d'16
e'16
)
}
}
}
>>> tuplets = voice[:]
>>> abjad.mutate.fuse(tuplets)
Tuplet('3:2', "c'8 d'8 e'8 c'16 d'16 e'16")
>>> abjad.show(staff) #doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\new Voice
{
\tuplet 3/2
{
c'8
[
d'8
e'8
]
c'16
(
d'16
e'16
)
}
}
}
Returns new tuplet in selection.
Fuses zero or more parent-contiguous ``tuplets``.
Allows in-score ``tuplets``.
Allows outside-of-score ``tuplets``.
All ``tuplets`` must carry the same multiplier.
All ``tuplets`` must be of the same type.
.. container:: example
REGRESSION. Trims tie from fused note:
>>> staff = abjad.Staff("d'8 ~ d'32 ~ d'16 d'32")
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
d'8
~
d'32
~
d'16
d'32
}
>>> logical_tie = abjad.select.logical_tie(staff[0])
>>> abjad.mutate.fuse(logical_tie)
[Note("d'8..")]
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
d'8..
d'32
}
>>> abjad.get.has_indicator(staff[0], abjad.Tie)
False
"""
_are_contiguous_logical_voice(argument)
if isinstance(argument, _score.Component):
result = _fuse([argument])
else:
result = _fuse(list(argument))
assert isinstance(result, list | _score.Tuplet), repr(result)
return result
[docs]
def logical_tie_to_tuplet(
argument, proportions, *, tag: _tag.Tag | None = None
) -> _score.Tuplet:
r"""
Changes logical tie to tuplet.
.. container:: example
>>> voice = abjad.Voice(r"df'8 c'8 ~ c'16 cqs''4")
>>> staff = abjad.Staff([voice])
>>> score = abjad.Score([staff], name="Score")
>>> abjad.attach(abjad.Dynamic('p'), voice[0])
>>> abjad.attach(abjad.StartHairpin('<'), voice[0])
>>> abjad.attach(abjad.Dynamic('f'), voice[-1])
>>> abjad.override(staff).DynamicLineSpanner.staff_padding = 3
>>> time_signature = abjad.TimeSignature((9, 16))
>>> abjad.attach(time_signature, voice[0])
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
\with
{
\override DynamicLineSpanner.staff-padding = 3
}
{
\new Voice
{
\time 9/16
df'8
\p
\<
c'8
~
c'16
cqs''4
\f
}
}
>>> logical_tie = abjad.select.logical_tie(voice[1])
>>> abjad.mutate.logical_tie_to_tuplet(logical_tie, [2, 1, 1, 1])
Tuplet('5:3', "c'8 c'16 c'16 c'16")
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
\with
{
\override DynamicLineSpanner.staff-padding = 3
}
{
\new Voice
{
\time 9/16
df'8
\p
\<
\tweak text #tuplet-number::calc-fraction-text
\tuplet 5/3
{
c'8
c'16
c'16
c'16
}
cqs''4
\f
}
}
>>> abjad.show(staff) # doctest: +SKIP
.. container:: example
>>> voice = abjad.Voice(r"c'8 ~ c'16 cqs''4")
>>> staff = abjad.Staff([voice])
>>> score = abjad.Score([staff], name="Score")
>>> abjad.hairpin('p < f', voice[:])
>>> abjad.override(staff).DynamicLineSpanner.staff_padding = 3
>>> time_signature = abjad.TimeSignature((7, 16))
>>> abjad.attach(time_signature, voice[0])
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
\with
{
\override DynamicLineSpanner.staff-padding = 3
}
{
\new Voice
{
\time 7/16
c'8
\p
\<
~
c'16
cqs''4
\f
}
}
"""
assert all(isinstance(_, int) for _ in proportions), repr(proportions)
target_duration = sum(_._get_preprolated_duration() for _ in argument)
assert isinstance(target_duration, _duration.Duration)
prolated_duration = target_duration / sum(proportions)
basic_written_duration = prolated_duration.equal_or_greater_power_of_two
written_durations = [_ * basic_written_duration for _ in proportions]
notes: list[_score.Note | _score.Tuplet]
try:
notes = [_score.Note(0, _) for _ in written_durations]
except _exceptions.AssignabilityError:
denominator = target_duration._denominator
note_durations = [_duration.Duration(_, denominator) for _ in proportions]
notes = _makers.make_notes(0, note_durations, tag=tag)
tuplet = _score.Tuplet.from_duration(target_duration, notes, tag=tag)
for leaf in argument:
_bind.detach(_indicators.Tie, leaf)
_bind.detach(_indicators.RepeatTie, leaf)
replace(argument, tuplet)
return tuplet
[docs]
def replace(argument, recipients, *, wrappers: bool = False) -> None:
r"""
Replaces ``argument`` (and contents of ``argument``) with ``recipients``.
.. container:: example
Replaces in-score tuplet (and children of tuplet) with notes. Functions
exactly the same as container setitem:
>>> tuplet_1 = abjad.Tuplet((2, 3), "c'4 d'4 e'4")
>>> tuplet_2 = abjad.Tuplet((2, 3), "d'4 e'4 f'4")
>>> voice = abjad.Voice([tuplet_1, tuplet_2])
>>> leaves = abjad.select.leaves(voice)
>>> abjad.hairpin('p < f', leaves)
>>> abjad.slur(leaves)
>>> abjad.show(voice) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(voice)
>>> print(string)
\new Voice
{
\tuplet 3/2
{
c'4
\p
(
\<
d'4
e'4
}
\tuplet 3/2
{
d'4
e'4
f'4
\f
)
}
}
>>> notes = abjad.makers.make_notes("c' d' e' f' c' d' e' f'", (1, 16))
>>> abjad.mutate.replace([tuplet_1], notes)
>>> abjad.attach(abjad.Dynamic('p'), voice[0])
>>> abjad.attach(abjad.StartHairpin('<'), voice[0])
>>> abjad.show(voice) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(voice)
>>> print(string)
\new Voice
{
c'16
\p
\<
d'16
e'16
f'16
c'16
d'16
e'16
f'16
\tuplet 3/2
{
d'4
e'4
f'4
\f
)
}
}
Preserves both hairpin and slur.
.. container:: example
Copies no wrappers when ``wrappers`` is false:
>>> staff = abjad.Staff("c'2 f'4 g'")
>>> abjad.attach(abjad.Clef('alto'), staff[0])
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\clef "alto"
c'2
f'4
g'4
}
>>> for leaf in staff:
... leaf, abjad.get.effective(leaf, abjad.Clef)
...
(Note("c'2"), Clef(name='alto', hide=False))
(Note("f'4"), Clef(name='alto', hide=False))
(Note("g'4"), Clef(name='alto', hide=False))
>>> chord = abjad.Chord("<d' e'>2")
>>> abjad.mutate.replace(staff[0], chord)
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
<d' e'>2
f'4
g'4
}
>>> for leaf in staff:
... leaf, abjad.get.effective(leaf, abjad.Clef)
...
(Chord("<d' e'>2"), None)
(Note("f'4"), None)
(Note("g'4"), None)
>>> abjad.wf.wellformed(staff)
True
.. container:: example
Set ``wrappers`` to true to copy all wrappers from one leaf to
another leaf (and avoid full-score update). Only works from one
leaf to another leaf:
>>> staff = abjad.Staff("c'2 f'4 g'")
>>> abjad.attach(abjad.Clef('alto'), staff[0])
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\clef "alto"
c'2
f'4
g'4
}
>>> for leaf in staff:
... leaf, abjad.get.effective(leaf, abjad.Clef)
...
(Note("c'2"), Clef(name='alto', hide=False))
(Note("f'4"), Clef(name='alto', hide=False))
(Note("g'4"), Clef(name='alto', hide=False))
>>> chord = abjad.Chord("<d' e'>2")
>>> abjad.mutate.replace(staff[0], chord, wrappers=True)
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\clef "alto"
<d' e'>2
f'4
g'4
}
>>> for leaf in staff:
... leaf, abjad.get.effective(leaf, abjad.Clef)
...
(Chord("<d' e'>2"), Clef(name='alto', hide=False))
(Note("f'4"), Clef(name='alto', hide=False))
(Note("g'4"), Clef(name='alto', hide=False))
>>> abjad.wf.wellformed(staff)
True
.. container:: example
.. todo:: Fix.
Introduces duplicate ties:
>>> staff = abjad.Staff("c'2 ~ c'2")
>>> tied_notes = abjad.makers.make_notes(0, abjad.Duration(5, 8))
>>> abjad.mutate.replace(staff[:1], tied_notes)
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
c'2
~
c'8
c'2
}
"""
if isinstance(argument, _score.Component):
donors = [argument]
else:
donors = argument
assert _are_contiguous_same_parent(donors)
if not isinstance(recipients, list):
if isinstance(recipients, _score.Component):
recipients = [recipients]
else:
recipients = list(recipients)
assert _are_contiguous_same_parent(recipients)
if not donors:
return
wrappers_ = []
if wrappers is True:
if 1 < len(donors) or not isinstance(donors[0], _score.Leaf):
raise Exception(f"set wrappers only with single leaf: {donors!r}.")
if 1 < len(recipients) or not isinstance(recipients[0], _score.Leaf):
raise Exception(f"set wrappers only with single leaf: {recipients!r}.")
donor = donors[0]
wrappers_ = _get.wrappers(donor)
recipient = recipients[0]
parent, start, stop = _get_parent_and_start_stop_indices(donors)
assert parent is not None, repr(donors)
parent.__setitem__(slice(start, stop + 1), recipients)
for wrapper in wrappers_:
# bypass Wrapper._bind_component()
# to avoid full-score update / traversal;
# this works because one-to-one leaf replacement
# including all (persistent) indicators
# doesn't change score structure:
donor._wrappers.remove(wrapper)
wrapper._component = recipient
recipient._wrappers.append(wrapper)
context = wrapper._find_correct_effective_context(
wrapper.component, wrapper.context
)
if context is not None:
context._dependent_wrappers.append(wrapper)
[docs]
def scale(argument, multiplier) -> None:
r"""
Scales ``argument`` by ``multiplier``.
.. container:: example
Scales note duration by dot-generating multiplier:
>>> staff = abjad.Staff("c'8 ( d'8 e'8 f'8 )")
>>> abjad.show(staff) # doctest: +SKIP
>>> abjad.mutate.scale(staff[1], abjad.Fraction(3, 2))
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
c'8
(
d'8.
e'8
f'8
)
}
.. container:: example
Scales tied leaves by dot-generating mutliplier:
>>> staff = abjad.Staff(r"c'8 \accent ~ c'8 d'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
- \accent
~
c'8
d'8
}
>>> logical_tie = abjad.select.logical_tie(staff[0])
>>> logical_tie = abjad.mutate.scale(logical_tie, abjad.Fraction(3, 2))
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\time 3/8
c'8.
- \accent
~
c'8.
d'8
}
.. container:: example
Scales leaves in container by dot-generating multiplier:
>>> container = abjad.Container(r"c'8 ( d'8 e'8 f'8 )")
>>> abjad.show(container) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(container)
>>> print(string)
{
c'8
(
d'8
e'8
f'8
)
}
>>> abjad.mutate.scale(container, abjad.Fraction(3, 2))
>>> abjad.show(container) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(container)
>>> print(string)
{
c'8.
(
d'8.
e'8.
f'8.
)
}
.. container:: example
Scales leaves in tuplet:
>>> staff = abjad.Staff()
>>> score = abjad.Score([staff], name="Score")
>>> tuplet = abjad.Tuplet((4, 5), "c'8 d'8 e'8 f'8 g'8")
>>> staff.append(tuplet)
>>> time_signature = abjad.TimeSignature((4, 8))
>>> leaf = abjad.get.leaf(staff, 0)
>>> abjad.attach(time_signature, leaf)
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\tuplet 5/4
{
\time 4/8
c'8
d'8
e'8
f'8
g'8
}
}
>>> abjad.mutate.scale(tuplet, abjad.Fraction(2))
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\tuplet 5/4
{
\time 4/8
c'4
d'4
e'4
f'4
g'4
}
}
.. container:: example
Scales leaves carrying LilyPond multiplier:
>>> note = abjad.Note("c'8", multiplier=(1, 2))
>>> abjad.show(note) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(note)
>>> print(string)
c'8 * 1/2
>>> abjad.mutate.scale(note, abjad.Fraction(1, 2))
>>> abjad.show(note) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(note)
>>> print(string)
c'16 * 1/2
"""
if hasattr(argument, "_scale"):
argument._scale(multiplier)
else:
assert isinstance(argument, list)
for component in argument:
component._scale(multiplier)
# TODO: add tests of tupletted notes and rests.
# TODO: add examples that show indicator handling.
# TODO: add example showing grace and after grace handling.
[docs]
def split(
argument, durations, *, cyclic: bool = False, tag: _tag.Tag | None = None
) -> list[list[_score.Component]]:
r"""
Splits ``argument`` by ``durations``.
Splits leaves cyclically and ties split notes:
.. container:: example
>>> voice = abjad.Voice("c'1 d'1")
>>> abjad.hairpin("p < f", voice[:])
>>> 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'1
\p
\<
d'1
\f
}
>>> durations = [(3, 4)]
>>> result = abjad.mutate.split(
... voice[:],
... durations,
... cyclic=True,
... )
>>> abjad.show(voice) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(voice)
>>> print(string)
\new Voice
\with
{
\override DynamicLineSpanner.staff-padding = 4
}
{
c'2.
\p
\<
~
c'4
d'2
\f
~
d'2
}
.. container:: example
Splits custom voice and preserves context name:
>>> voice = abjad.Voice(
... "c'4 d' e' f'",
... lilypond_type='CustomVoice',
... name='1',
... )
>>> staff = abjad.Staff([voice])
>>> abjad.hairpin('p < f', voice[:])
>>> abjad.override(staff).DynamicLineSpanner.staff_padding = 3
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
\with
{
\override DynamicLineSpanner.staff-padding = 3
}
{
\context CustomVoice = "1"
{
c'4
\p
\<
d'4
e'4
f'4
\f
}
}
>>> durations = [(1, 8)]
>>> result = abjad.mutate.split(
... staff[:],
... durations,
... cyclic=True,
... )
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
\with
{
\override DynamicLineSpanner.staff-padding = 3
}
{
\context CustomVoice = "1"
{
c'8
\p
\<
~
}
\context CustomVoice = "1"
{
c'8
}
\context CustomVoice = "1"
{
d'8
~
}
\context CustomVoice = "1"
{
d'8
}
\context CustomVoice = "1"
{
e'8
~
}
\context CustomVoice = "1"
{
e'8
}
\context CustomVoice = "1"
{
f'8
\f
~
}
\context CustomVoice = "1"
{
f'8
}
}
>>> for voice in staff:
... voice
...
Voice("c'8", lilypond_type='CustomVoice', name='1')
Voice("c'8", lilypond_type='CustomVoice', name='1')
Voice("d'8", lilypond_type='CustomVoice', name='1')
Voice("d'8", lilypond_type='CustomVoice', name='1')
Voice("e'8", lilypond_type='CustomVoice', name='1')
Voice("e'8", lilypond_type='CustomVoice', name='1')
Voice("f'8", lilypond_type='CustomVoice', name='1')
Voice("f'8", lilypond_type='CustomVoice', name='1')
.. container:: example
Splits parallel container:
>>> voice_1 = abjad.Voice(
... "e''4 ( es'' f'' fs'' )",
... name='Voice_1',
... )
>>> voice_2 = abjad.Voice(
... r"c'4 \p \< cs' d' ds' \f",
... name='Voice_2',
... )
>>> abjad.override(voice_1).Stem.direction = abjad.UP
>>> abjad.override(voice_1).Slur.direction = abjad.UP
>>> container = abjad.Container(
... [voice_1, voice_2],
... simultaneous=True,
... )
>>> abjad.override(voice_2).Stem.direction = abjad.DOWN
>>> staff = abjad.Staff([container])
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
<<
\context Voice = "Voice_1"
\with
{
\override Slur.direction = #up
\override Stem.direction = #up
}
{
e''4
(
es''4
f''4
fs''4
)
}
\context Voice = "Voice_2"
\with
{
\override Stem.direction = #down
}
{
c'4
\p
\<
cs'4
d'4
ds'4
\f
}
>>
}
>>> durations = [(3, 8)]
>>> result = abjad.mutate.split(
... container,
... durations,
... cyclic=False,
... )
>>> abjad.show(staff) # doctest: +SKIP
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
<<
\context Voice = "Voice_1"
\with
{
\override Slur.direction = #up
\override Stem.direction = #up
}
{
e''4
(
es''8
~
}
\context Voice = "Voice_2"
\with
{
\override Stem.direction = #down
}
{
c'4
\p
\<
cs'8
~
}
>>
<<
\context Voice = "Voice_1"
\with
{
\override Slur.direction = #up
\override Stem.direction = #up
}
{
es''8
f''4
fs''4
)
}
\context Voice = "Voice_2"
\with
{
\override Stem.direction = #down
}
{
cs'8
d'4
ds'4
\f
}
>>
}
.. container:: example
Splits leaves with articulations:
>>> staff = abjad.Staff("c'4 d' e' f'")
>>> abjad.attach(abjad.Articulation('^'), staff[0])
>>> abjad.attach(abjad.LaissezVibrer(), staff[1])
>>> abjad.attach(abjad.Articulation('^'), staff[2])
>>> abjad.attach(abjad.LaissezVibrer(), staff[3])
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
c'4
- \marcato
d'4
\laissezVibrer
e'4
- \marcato
f'4
\laissezVibrer
}
>>> durations = [(1, 8)]
>>> result = abjad.mutate.split(
... staff[:],
... durations,
... cyclic=True,
... )
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
c'8
- \marcato
~
c'8
d'8
~
d'8
\laissezVibrer
e'8
- \marcato
~
e'8
f'8
~
f'8
\laissezVibrer
}
.. container:: example
REGRESSION. Preserves tie:
>>> staff = abjad.Staff("d'2 ~ d'")
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
d'2
~
d'2
}
>>> result = abjad.mutate.split(staff[0], [(1, 32)])
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
d'32
~
d'4...
~
d'2
}
.. container:: example
REGRESSION. Leaf independent after-grace leaves unchanged:
>>> music_voice = abjad.Voice("c'4 d' e' f'", name="MusicVoice")
>>> container = abjad.IndependentAfterGraceContainer("af'4 gf'4")
>>> music_voice.insert(3, container)
>>> staff = abjad.Staff([music_voice])
>>> lilypond_file = abjad.LilyPondFile([staff])
>>> abjad.show(lilypond_file) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\context Voice = "MusicVoice"
{
c'4
d'4
\afterGrace
e'4
{
af'4
gf'4
}
f'4
}
}
>>> result = abjad.mutate.split(music_voice[:], [(1, 8)], cyclic=True)
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\context Voice = "MusicVoice"
{
c'8
~
c'8
d'8
~
d'8
e'8
~
\afterGrace
e'8
{
af'4
gf'4
}
f'8
~
f'8
}
}
"""
components = argument
if isinstance(components, _score.Component):
components = [components]
assert all(isinstance(_, _score.Component) for _ in components)
durations = [_duration.Duration(_) for _ in durations]
assert len(durations), repr(durations)
total_component_duration = _get.duration(components)
total_split_duration = sum(durations)
if cyclic:
durations = _sequence.repeat_to_weight(durations, total_component_duration)
durations = list(durations)
elif total_split_duration < total_component_duration:
final_offset = total_component_duration - sum(durations)
durations.append(final_offset)
elif total_component_duration < total_split_duration:
weight = total_component_duration
durations = _sequence.truncate(durations, weight=weight)
durations = list(durations)
# keep copy of durations to partition result components
durations_copy = durations[:]
total_split_duration = sum(durations)
assert total_split_duration == total_component_duration
result, shard = [], []
offset_index = 0
current_shard_duration = _duration.Duration(0)
remaining_components = list(components[:])
advance_to_next_offset = True
# build shards:
# grab next component and next duration each time through loop
while True:
# grab next split point
if advance_to_next_offset:
if not durations:
break
next_split_point = durations.pop(0)
advance_to_next_offset = True
# grab next component from input stack of components
if not remaining_components:
break
current_component = remaining_components.pop(0)
# find where current component endpoint will position us
duration_ = _get.duration(current_component)
candidate_shard_duration = current_shard_duration + duration_
# if current component would fill current shard exactly
if candidate_shard_duration == next_split_point:
shard.append(current_component)
result.append(shard)
shard = []
current_shard_duration = _duration.Duration(0)
offset_index += 1
# if current component would exceed current shard
elif next_split_point < candidate_shard_duration:
local_split_duration = next_split_point
local_split_duration -= current_shard_duration
if isinstance(current_component, _score.Leaf):
leaf_split_durations = [local_split_duration]
duration_ = _get.duration(current_component)
current_duration = duration_
additional_required_duration = current_duration
additional_required_duration -= local_split_duration
split_durations = _sequence.split(
durations,
[additional_required_duration],
cyclic=False,
overhang=True,
)
split_durations = [list(_) for _ in split_durations]
additional_durations = split_durations[0]
leaf_split_durations.extend(additional_durations)
durations = split_durations[-1]
leaf_shards = _split_leaf_by_durations(
current_component,
leaf_split_durations,
cyclic=False,
tag=tag,
)
shard.extend(leaf_shards)
result.append(shard)
offset_index += len(additional_durations)
else:
assert isinstance(current_component, _score.Container)
pair = _split_container_by_duration(
current_component,
local_split_duration,
tag=tag,
)
left_list, right_list = pair
shard.extend(left_list)
result.append(shard)
remaining_components.__setitem__(slice(0, 0), right_list)
shard = []
offset_index += 1
current_shard_duration = _duration.Duration(0)
# if current component would not fill current shard
elif candidate_shard_duration < next_split_point:
shard.append(current_component)
duration_ = _get.duration(current_component)
current_shard_duration += duration_
advance_to_next_offset = False
else:
message = "can not process candidate duration: {!r}."
message = message.format(candidate_shard_duration)
raise ValueError(message)
# append any stub shard
if len(shard):
result.append(shard)
# append any unexamined components
if len(remaining_components):
result.append(remaining_components)
# partition split components according to input durations
result = _sequence.flatten(result, depth=-1)
result = _select.partition_by_durations(result, durations_copy, fill=_enums.EXACT)
# return list of component lists
for list_ in result:
assert isinstance(list_, list), repr(list_)
assert all(isinstance(_, _score.Component) for _ in list_), repr(list_)
return result
[docs]
def swap(argument, container):
r"""
Swaps ``argument`` for empty ``container``.
.. container:: example
Swaps containers for tuplet:
>>> voice = abjad.Voice()
>>> staff = abjad.Staff([voice])
>>> score = abjad.Score([staff], name="Score")
>>> voice.append(abjad.Container("c'4 d'4 e'4"))
>>> voice.append(abjad.Container("d'4 e'4 f'4"))
>>> abjad.attach(abjad.TimeSignature((3, 4)), voice[0][0])
>>> leaves = abjad.select.leaves(voice)
>>> abjad.hairpin('p < f', leaves)
>>> measures = voice[:]
>>> abjad.slur(leaves)
>>> abjad.show(voice) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(voice)
>>> print(string)
\new Voice
{
{
\time 3/4
c'4
\p
(
\<
d'4
e'4
}
{
d'4
e'4
f'4
\f
)
}
}
>>> containers = voice[:]
>>> tuplet = abjad.Tuplet((2, 3), [])
>>> tuplet.denominator = 4
>>> abjad.mutate.swap(containers, tuplet)
>>> abjad.show(voice) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(voice)
>>> print(string)
\new Voice
{
\tuplet 6/4
{
\time 3/4
c'4
\p
(
\<
d'4
e'4
d'4
e'4
f'4
\f
)
}
}
REGRESSION: context indicators survive swap:
>>> prototype = abjad.TimeSignature
>>> for component in abjad.iterate.components(voice):
... time_signature = abjad.get.effective(component, prototype)
... print(component, time_signature)
...
Voice("{ 2/3 c'4 d'4 e'4 d'4 e'4 f'4 }") TimeSignature(pair=(3, 4), hide=False, partial=None)
Tuplet('3:2', "c'4 d'4 e'4 d'4 e'4 f'4") TimeSignature(pair=(3, 4), hide=False, partial=None)
Note("c'4") TimeSignature(pair=(3, 4), hide=False, partial=None)
Note("d'4") TimeSignature(pair=(3, 4), hide=False, partial=None)
Note("e'4") TimeSignature(pair=(3, 4), hide=False, partial=None)
Note("d'4") TimeSignature(pair=(3, 4), hide=False, partial=None)
Note("e'4") TimeSignature(pair=(3, 4), hide=False, partial=None)
Note("f'4") TimeSignature(pair=(3, 4), hide=False, partial=None)
Returns none.
"""
if isinstance(argument, list):
donors = argument
else:
assert isinstance(argument, _score.Component)
donors = [argument]
assert _are_contiguous_same_parent(donors)
assert isinstance(container, _score.Container)
assert not container, repr(container)
_give_components_to_empty_container(donors, container)
_give_position_in_parent_to_container(donors, container)
[docs]
def transpose(argument, interval):
r"""
Transposes notes and chords in ``argument`` by ``interval``.
.. todo:: Move to abjad.pitch package.
.. container:: example
Transposes notes and chords in staff:
>>> staff = abjad.Staff()
>>> score = abjad.Score([staff], name="Score")
>>> staff.extend("c'4 d'4 e'4 r4")
>>> abjad.attach(abjad.TimeSignature((4, 4)), staff[0])
>>> staff.extend("d'4 e'4 <f' a' c''>4")
>>> abjad.attach(abjad.TimeSignature((3, 4)), staff[4])
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\time 4/4
c'4
d'4
e'4
r4
\time 3/4
d'4
e'4
<f' a' c''>4
}
>>> abjad.mutate.transpose(staff, "+m3")
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\time 4/4
ef'4
f'4
g'4
r4
\time 3/4
f'4
g'4
<af' c'' ef''>4
}
Returns none.
"""
named_interval = _pitch.NamedInterval(interval)
for item in _iterate.components(argument, (_score.Note, _score.Chord)):
if isinstance(item, _score.Note):
old_written_pitch = item.note_head.written_pitch
new_written_pitch = old_written_pitch.transpose(named_interval)
item.note_head.written_pitch = new_written_pitch
else:
for note_head in item.note_heads:
old_written_pitch = note_head.written_pitch
new_written_pitch = old_written_pitch.transpose(named_interval)
note_head.written_pitch = new_written_pitch
[docs]
def wrap(argument, container):
r"""
Wraps ``argument`` in empty ``container``.
.. container:: example
Wraps in-score notes in tuplet:
>>> staff = abjad.Staff("c'8 [ ( d' e' ] ) c' [ ( d' e' ] )")
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
c'8
[
(
d'8
e'8
)
]
c'8
[
(
d'8
e'8
)
]
}
>>> tuplet = abjad.Tuplet((2, 3), [])
>>> abjad.mutate.wrap(staff[-3:], tuplet)
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
c'8
[
(
d'8
e'8
)
]
\tuplet 3/2
{
c'8
[
(
d'8
e'8
)
]
}
}
.. container:: example
Wraps outside-score notes in tuplet:
>>> notes = abjad.makers.make_notes([0, 2, 4], [(1, 8)])
>>> tuplet = abjad.Tuplet((2, 3), [])
>>> abjad.mutate.wrap(notes, tuplet)
>>> abjad.show(tuplet) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(tuplet)
>>> print(string)
\tuplet 3/2
{
c'8
d'8
e'8
}
(This usage merely substitutes for the tuplet initializer.)
.. container:: example
Wraps leaves in container:
>>> notes = [abjad.Note(n, (1, 8)) for n in range(8)]
>>> staff = abjad.Staff(notes)
>>> score = abjad.Score([staff], name="Score")
>>> abjad.attach(abjad.TimeSignature((4, 8)), staff[0])
>>> container = abjad.Container()
>>> abjad.mutate.wrap(staff[:4], container)
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
{
\time 4/8
c'8
cs'8
d'8
ef'8
}
e'8
f'8
fs'8
g'8
}
.. container:: example
Wraps each leaf in tuplet:
>>> notes = [abjad.Note(n, (1, 1)) for n in range(4)]
>>> staff = abjad.Staff(notes)
>>> for note in staff:
... tuplet = abjad.Tuplet((2, 3))
... abjad.mutate.wrap(note, tuplet)
...
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
\tweak edge-height #'(0.7 . 0)
\tuplet 3/2
{
c'1
}
\tweak edge-height #'(0.7 . 0)
\tuplet 3/2
{
cs'1
}
\tweak edge-height #'(0.7 . 0)
\tuplet 3/2
{
d'1
}
\tweak edge-height #'(0.7 . 0)
\tuplet 3/2
{
ef'1
}
}
.. container:: example
Raises exception on nonempty ``container``:
>>> import pytest
>>> staff = abjad.Staff("c'8 [ ( d' e' ] ) c' [ ( d' e' ] )")
>>> tuplet = abjad.Tuplet((2, 3), "g'8 a' fs'")
>>> abjad.mutate.wrap(staff[-3:], tuplet)
Traceback (most recent call last):
...
Exception: must be empty container: Tuplet('3:2', "g'8 a'8 fs'8").
.. container:: example
REGRESSION. Contexted indicators (like time signature) survive
wrap:
>>> staff = abjad.Staff("c'4 d' e' f'")
>>> score = abjad.Score([staff], name="Score")
>>> leaves = abjad.select.leaves(staff)
>>> abjad.attach(abjad.TimeSignature((3, 8)), leaves[0])
>>> container = abjad.Container()
>>> abjad.mutate.wrap(leaves, container)
>>> abjad.show(staff) # doctest: +SKIP
.. docs::
>>> string = abjad.lilypond(staff)
>>> print(string)
\new Staff
{
{
\time 3/8
c'4
d'4
e'4
f'4
}
}
>>> prototype = abjad.TimeSignature
>>> for component in abjad.iterate.components(staff):
... time_signature = abjad.get.effective(component, prototype)
... print(component, time_signature)
...
Staff("{ c'4 d'4 e'4 f'4 }") TimeSignature(pair=(3, 8), hide=False, partial=None)
Container("c'4 d'4 e'4 f'4") TimeSignature(pair=(3, 8), hide=False, partial=None)
Note("c'4") TimeSignature(pair=(3, 8), hide=False, partial=None)
Note("d'4") TimeSignature(pair=(3, 8), hide=False, partial=None)
Note("e'4") TimeSignature(pair=(3, 8), hide=False, partial=None)
Note("f'4") TimeSignature(pair=(3, 8), hide=False, partial=None)
Returns none.
"""
if not isinstance(container, _score.Container) or 0 < len(container):
raise Exception(f"must be empty container: {container!r}.")
if isinstance(argument, _score.Component):
selection = [argument]
else:
selection = argument
parent, start, stop = _get_parent_and_start_stop_indices(
selection, ignore_before_after_grace=True
)
if not _are_contiguous_logical_voice(selection, ignore_before_after_grace=True):
message = "must be contiguous components in same logical voice:\n"
message += f" {selection!r}."
raise Exception(message)
container._components = list(selection)
_set_parents(container)
if parent is not None:
parent._components.insert(start, container)
container._set_parent(parent)