import builtins
import collections
import copy
import fractions
import itertools
import math
import sys
import typing
from . import cyclictuple as _cyclictuple
from . import enums as _enums
from . import math as _math
def _partition_sequence_cyclically_by_weights_at_least(
sequence, weights, overhang=False
):
l_copy = list(sequence)
result = []
current_part = []
target_weight_index = 0
len_weights = len(weights)
while l_copy:
target_weight = weights[target_weight_index % len_weights]
item = l_copy.pop(0)
current_part.append(item)
if target_weight <= _math.weight(current_part):
result.append(current_part)
current_part = []
target_weight_index += 1
assert not l_copy
if current_part:
if overhang:
result.append(current_part)
result = [type(sequence)(_) for _ in result]
return result
def _partition_sequence_cyclically_by_weights_at_most(
sequence, weights, overhang=False
):
result = []
current_part = []
current_target_weight_index = 0
current_target_weight = weights[current_target_weight_index]
l_copy = list(sequence)
while l_copy:
current_target_weight = weights[current_target_weight_index % len(weights)]
item = l_copy.pop(0)
current_part_weight = _math.weight(current_part)
candidate_part_weight = current_part_weight + _math.weight([item])
if candidate_part_weight < current_target_weight:
current_part.append(item)
elif candidate_part_weight == current_target_weight:
current_part.append(item)
result.append(current_part)
current_part = []
current_target_weight_index += 1
elif current_target_weight < candidate_part_weight:
if current_part:
l_copy.insert(0, item)
result.append(current_part)
current_part = []
current_target_weight_index += 1
else:
raise Exception("elements in sequence too big.")
else:
raise ValueError("candidate and target rates must compare.")
if current_part:
if overhang:
result.append(current_part)
result = [type(sequence)(_) for _ in result]
return result
def _partition_sequence_once_by_weights_at_least(sequence, weights, overhang=False):
result = []
current_part = []
l_copy = list(sequence)
for num_weight, target_weight in enumerate(weights):
while True:
try:
item = l_copy.pop(0)
except IndexError:
if num_weight + 1 == len(weights):
if current_part:
result.append(current_part)
break
raise Exception("too few elements in sequence.")
current_part.append(item)
if target_weight <= _math.weight(current_part):
result.append(current_part)
current_part = []
break
if l_copy:
if overhang:
result.append(l_copy)
result = [type(sequence)(_) for _ in result]
return result
def _partition_sequence_once_by_weights_at_most(sequence, weights, overhang=False):
l_copy = list(sequence)
result = []
current_part = []
for target_weight in weights:
while True:
try:
item = l_copy.pop(0)
except IndexError:
raise Exception("too few elements in sequence.")
current_weight = _math.weight(current_part)
candidate_weight = current_weight + _math.weight([item])
if candidate_weight < target_weight:
current_part.append(item)
elif candidate_weight == target_weight:
current_part.append(item)
result.append(current_part)
current_part = []
break
elif target_weight < candidate_weight:
if current_part:
result.append(current_part)
current_part = []
l_copy.insert(0, item)
break
else:
raise Exception("elements in sequence too big.")
else:
raise ValueError("candidate and target weights must compare.")
if overhang:
left_over = current_part + l_copy
if left_over:
result.append(left_over)
result = [type(sequence)(_) for _ in result]
return result
[docs]
def partition_by_counts(
sequence,
counts,
*,
cyclic: bool = False,
enchain: bool = False,
overhang: bool | _enums.Comparison = False,
reversed_: bool = False,
) -> list:
r"""
Partitions ``sequence`` by ``counts``.
Partitions sequence once by counts without overhang:
.. container:: example
>>> sequence = list(range(16))
>>> sequence = abjad.sequence.partition_by_counts(
... sequence,
... [3],
... cyclic=False,
... overhang=False,
... )
>>> sequence
[[0, 1, 2]]
>>> for part in sequence:
... part
[0, 1, 2]
.. container:: example
Partitions sequence once by counts without overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [4, 3],
... cyclic=False,
... overhang=False,
... )
>>> for part in parts:
... part
[0, 1, 2, 3]
[4, 5, 6]
.. container:: example
Partitions sequence cyclically by counts without overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [3],
... cyclic=True,
... overhang=False,
... )
>>> for part in parts:
... part
[0, 1, 2]
[3, 4, 5]
[6, 7, 8]
[9, 10, 11]
[12, 13, 14]
.. container:: example
Partitions sequence cyclically by counts without overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [4, 3],
... cyclic=True,
... overhang=False,
... )
>>> for part in parts:
... part
[0, 1, 2, 3]
[4, 5, 6]
[7, 8, 9, 10]
[11, 12, 13]
.. container:: example
Partitions sequence once by counts with overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [3],
... cyclic=False,
... overhang=True,
... )
>>> for part in parts:
... part
[0, 1, 2]
[3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15]
.. container:: example
Partitions sequence once by counts with overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [4, 3],
... cyclic=False,
... overhang=True,
... )
>>> for part in parts:
... part
[0, 1, 2, 3]
[4, 5, 6]
[7, 8, 9, 10, 11, 12, 13, 14, 15]
.. container:: example
Partitions sequence cyclically by counts with overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [3],
... cyclic=True,
... overhang=True,
... )
>>> for part in parts:
... part
[0, 1, 2]
[3, 4, 5]
[6, 7, 8]
[9, 10, 11]
[12, 13, 14]
[15]
.. container:: example
Partitions sequence cyclically by counts with overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [4, 3],
... cyclic=True,
... overhang=True,
... )
>>> for part in parts:
... part
[0, 1, 2, 3]
[4, 5, 6]
[7, 8, 9, 10]
[11, 12, 13]
[14, 15]
.. container:: example
Reverse-partitions sequence once by counts without overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [3],
... cyclic=False,
... overhang=False,
... reversed_=True,
... )
>>> for part in parts:
... part
[13, 14, 15]
.. container:: example
Reverse-partitions sequence once by counts without overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [4, 3],
... cyclic=False,
... overhang=False,
... reversed_=True,
... )
>>> for part in parts:
... part
[9, 10, 11]
[12, 13, 14, 15]
.. container:: example
Reverse-partitions sequence cyclically by counts without overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [3],
... cyclic=True,
... overhang=False,
... reversed_=True,
... )
>>> for part in parts:
... part
[1, 2, 3]
[4, 5, 6]
[7, 8, 9]
[10, 11, 12]
[13, 14, 15]
.. container:: example
Reverse-partitions sequence cyclically by counts without overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [4, 3],
... cyclic=True,
... overhang=False,
... reversed_=True,
... )
>>> for part in parts:
... part
[2, 3, 4]
[5, 6, 7, 8]
[9, 10, 11]
[12, 13, 14, 15]
.. container:: example
Reverse-partitions sequence once by counts with overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [3],
... cyclic=False,
... overhang=True,
... reversed_=True,
... )
>>> for part in parts:
... part
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12]
[13, 14, 15]
.. container:: example
Reverse-partitions sequence once by counts with overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [4, 3],
... cyclic=False,
... overhang=True,
... reversed_=True,
... )
>>> for part in parts:
... part
[0, 1, 2, 3, 4, 5, 6, 7, 8]
[9, 10, 11]
[12, 13, 14, 15]
.. container:: example
Reverse-partitions sequence cyclically by counts with overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [3],
... cyclic=True,
... overhang=True,
... reversed_=True,
... )
>>> for part in parts:
... part
[0]
[1, 2, 3]
[4, 5, 6]
[7, 8, 9]
[10, 11, 12]
[13, 14, 15]
.. container:: example
Reverse-partitions sequence cyclically by counts with overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [4, 3],
... cyclic=True,
... overhang=True,
... reversed_=True,
... )
>>> for part in parts:
... part
[0, 1]
[2, 3, 4]
[5, 6, 7, 8]
[9, 10, 11]
[12, 13, 14, 15]
.. container:: example
Partitions sequence once by counts and asserts that sequence partitions
exactly (with no overhang):
>>> sequence = list(range(10))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [2, 3, 5],
... cyclic=False,
... overhang=abjad.EXACT,
... )
>>> for part in parts:
... part
[0, 1]
[2, 3, 4]
[5, 6, 7, 8, 9]
.. container:: example
Partitions sequence cyclically by counts and asserts that sequence partitions
exactly. Exact partitioning means partitioning with no overhang:
>>> sequence = list(range(10))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [2],
... cyclic=True,
... overhang=abjad.EXACT,
... )
>>> for part in parts:
... part
[0, 1]
[2, 3]
[4, 5]
[6, 7]
[8, 9]
.. container:: example
Partitions string:
>>> sequence = list("some text")
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [3],
... cyclic=False,
... overhang=True,
... )
>>> for part in parts:
... part
['s', 'o', 'm']
['e', ' ', 't', 'e', 'x', 't']
.. container:: example
Partitions sequence cyclically into enchained parts by counts; truncates
overhang:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [2, 6],
... cyclic=True,
... enchain=True,
... overhang=False,
... )
>>> for part in parts:
... part
[0, 1]
[1, 2, 3, 4, 5, 6]
[6, 7]
[7, 8, 9, 10, 11, 12]
[12, 13]
.. container:: example
Partitions sequence cyclically into enchained parts by counts; returns
overhang at end:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [2, 6],
... cyclic=True,
... enchain=True,
... overhang=True,
... )
>>> for part in parts:
... part
[0, 1]
[1, 2, 3, 4, 5, 6]
[6, 7]
[7, 8, 9, 10, 11, 12]
[12, 13]
[13, 14, 15]
.. container:: example
REGRESSION: partitions sequence cyclically into enchained parts by counts;
does not return false 1-element part at end:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(
... sequence,
... [5],
... cyclic=True,
... enchain=True,
... overhang=True,
... )
>>> for part in parts:
... part
[0, 1, 2, 3, 4]
[4, 5, 6, 7, 8]
[8, 9, 10, 11, 12]
[12, 13, 14, 15]
.. container:: example
Edge case: empty counts nests sequence and ignores keywords:
>>> sequence = list(range(16))
>>> parts = abjad.sequence.partition_by_counts(sequence, [])
>>> for part in parts:
... part
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15]
Returns list of ``sequence`` types.
"""
assert isinstance(cyclic, bool), repr(cyclic)
assert isinstance(enchain, bool), repr(enchain)
assert overhang in (True, False, _enums.EXACT), repr(overhang)
assert isinstance(reversed_, bool), repr(reversed_)
if not all(isinstance(_, int) and 0 <= _ for _ in counts):
raise Exception(f"must be nonnegative integers: {counts!r}.")
if reversed_:
sequence = type(sequence)(reversed(sequence))
if counts:
counts = _cyclictuple.CyclicTuple(counts)
else:
return type(sequence)([sequence])
result = []
i, start = 0, 0
while True:
count = counts[i]
stop = start + count
part = sequence[start:stop]
if len(sequence) < stop:
if enchain and len(part) == 1:
part = None
break
result.append(part)
start = stop
i += 1
if not cyclic and len(counts) <= i:
part = sequence[start:]
break
if enchain:
start -= 1
if part:
if overhang is True:
result.append(part)
elif overhang is _enums.EXACT and len(part) == count:
result.append(part)
elif overhang is _enums.EXACT and len(part) != count:
raise Exception("sequence does not partition exactly.")
if reversed_:
result_ = []
for part in reversed(result):
part_type = type(part)
part = reversed(part)
part = part_type(part)
result_.append(part)
result = result_
return result
[docs]
def partition_by_ratio_of_lengths(sequence, ratio: tuple[int, ...]) -> list:
r"""
Partitions ``sequence`` by ``ratio`` of lengths.
Partitions sequence by ``1:1:1`` ratio:
.. container:: example
>>> numbers = list(range(10))
>>> ratio = (1, 1, 1)
>>> for part in abjad.sequence.partition_by_ratio_of_lengths(numbers, ratio):
... part
[0, 1, 2]
[3, 4, 5, 6]
[7, 8, 9]
.. container:: example
Partitions sequence by ``1:1:2`` ratio:
>>> numbers = list(range(10))
>>> ratio = (1, 1, 2)
>>> for part in abjad.sequence.partition_by_ratio_of_lengths(numbers, ratio):
... part
[0, 1, 2]
[3, 4]
[5, 6, 7, 8, 9]
Returns list of ``sequence`` types.
"""
assert isinstance(ratio, tuple), repr(ratio)
length = len(sequence)
counts = _math.partition_integer_by_ratio(length, ratio)
parts = partition_by_counts(sequence, counts, cyclic=False, overhang=_enums.EXACT)
return parts
[docs]
def partition_by_ratio_of_weights(sequence, weights: typing.Sequence[int]) -> list:
"""
Partitions ``sequence`` by ratio of ``weights``.
.. container:: example
>>> ratio = (1, 1, 1)
>>> sequence = list(10 * [1])
>>> sequence = abjad.sequence.partition_by_ratio_of_weights(sequence, ratio)
>>> for item in sequence:
... item
...
[1, 1, 1]
[1, 1, 1, 1]
[1, 1, 1]
.. container:: example
>>> ratio = (1, 1, 1, 1)
>>> sequence = list(10 * [1])
>>> sequence = abjad.sequence.partition_by_ratio_of_weights(sequence, ratio)
>>> for item in sequence:
... item
...
[1, 1, 1]
[1, 1]
[1, 1, 1]
[1, 1]
.. container:: example
>>> ratio = (2, 2, 3)
>>> sequence = list(10 * [1])
>>> sequence = abjad.sequence.partition_by_ratio_of_weights(sequence, ratio)
>>> for item in sequence:
... item
...
[1, 1, 1]
[1, 1, 1]
[1, 1, 1, 1]
.. container:: example
>>> ratio = (3, 2, 2)
>>> sequence = list(10 * [1])
>>> sequence = abjad.sequence.partition_by_ratio_of_weights(sequence, ratio)
>>> for item in sequence:
... item
...
[1, 1, 1, 1]
[1, 1, 1]
[1, 1, 1]
.. container:: example
>>> ratio = (1, 1)
>>> items = [1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2]
>>> sequence = list(items)
>>> sequence = abjad.sequence.partition_by_ratio_of_weights(sequence, ratio)
>>> for item in sequence:
... item
...
[1, 1, 1, 1, 1, 1, 2, 2]
[2, 2, 2, 2]
.. container:: example
>>> ratio = (1, 1, 1)
>>> items = [1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2]
>>> sequence = list(items)
>>> sequence = abjad.sequence.partition_by_ratio_of_weights(sequence, ratio)
>>> for item in sequence:
... item
...
[1, 1, 1, 1, 1, 1]
[2, 2, 2]
[2, 2, 2]
.. container:: example
>>> ratio = (1, 1, 1)
>>> sequence = list([5, 5])
>>> sequence = abjad.sequence.partition_by_ratio_of_weights(sequence, ratio)
>>> for item in sequence:
... item
...
[5]
[5]
[]
.. container:: example
>>> ratio = (1, 1, 1, 1)
>>> sequence = list([5, 5])
>>> sequence = abjad.sequence.partition_by_ratio_of_weights(sequence, ratio)
>>> for item in sequence:
... item
...
[5]
[]
[5]
[]
.. container:: example
>>> ratio = (2, 2, 3)
>>> sequence = list([5, 5])
>>> sequence = abjad.sequence.partition_by_ratio_of_weights(sequence, ratio)
>>> for item in sequence:
... item
...
[5]
[5]
[]
.. container:: example
>>> ratio = (3, 2, 2)
>>> sequence = list([5, 5])
>>> sequence = abjad.sequence.partition_by_ratio_of_weights(sequence, ratio)
>>> for item in sequence:
... item
...
[5]
[5]
[]
Rounded weight-proportions of sequences returned equal to rounded ``weights``.
Returns list of ``sequence`` types.
"""
assert all(isinstance(_, int | float | fractions.Fraction) for _ in sequence)
sequence_weight = _math.weight(sequence)
partitioned_weights = _math.partition_integer_by_ratio(sequence_weight, weights)
cumulative_weights = _math.cumulative_sums(partitioned_weights, start=None)
items = []
sublist: list[typing.Any] = []
items.append(sublist)
current_cumulative_weight = cumulative_weights.pop(0)
for item in sequence:
sublist.append(item)
while current_cumulative_weight <= _math.weight(flatten(items, depth=-1)):
try:
current_cumulative_weight = cumulative_weights.pop(0)
sublist = []
items.append(sublist)
except IndexError:
break
result = [type(sequence)(_) for _ in items]
return result
[docs]
def partition_by_weights(
sequence,
weights: typing.Sequence[int | fractions.Fraction],
*,
cyclic: bool = False,
overhang: bool = False,
allow_part_weights: _enums.Comparison = _enums.EXACT,
) -> list:
r"""
Partitions ``sequence`` by ``weights`` exactly.
>>> sequence = [3, 3, 3, 3, 4, 4, 4, 4, 5]
.. container:: example
Partitions sequence once by weights with overhang:
>>> for item in abjad.sequence.partition_by_weights(
... sequence,
... [3, 9],
... cyclic=False,
... overhang=False,
... ):
... item
...
[3]
[3, 3, 3]
.. container:: example
Partitions sequence once by weights. Allows overhang:
>>> for item in abjad.sequence.partition_by_weights(
... sequence,
... [3, 9],
... cyclic=False,
... overhang=True,
... ):
... item
...
[3]
[3, 3, 3]
[4, 4, 4, 4, 5]
.. container:: example
Partitions sequence cyclically by weights:
>>> for item in abjad.sequence.partition_by_weights(
... sequence,
... [12],
... cyclic=True,
... overhang=False,
... ):
... item
...
[3, 3, 3, 3]
[4, 4, 4]
.. container:: example
Partitions sequence cyclically by weights. Allows overhang:
>>> for item in abjad.sequence.partition_by_weights(
... sequence,
... [12],
... cyclic=True,
... overhang=True,
... ):
... item
...
[3, 3, 3, 3]
[4, 4, 4]
[4, 5]
>>> sequence = list([3, 3, 3, 3, 4, 4, 4, 4, 5, 5])
.. container:: example
Partitions sequence once by weights. Allows part weights to be just less than
specified:
>>> for item in abjad.sequence.partition_by_weights(
... sequence,
... [10, 4],
... cyclic=False,
... overhang=False,
... allow_part_weights=abjad.LESS,
... ):
... item
...
[3, 3, 3]
[3]
.. container:: example
Partitions sequence once by weights. Allows part weights to be just less than
specified. Allows overhang:
>>> for item in abjad.sequence.partition_by_weights(
... sequence,
... [10, 4],
... cyclic=False,
... overhang=True,
... allow_part_weights=abjad.LESS,
... ):
... item
...
[3, 3, 3]
[3]
[4, 4, 4, 4, 5, 5]
.. container:: example
Partitions sequence cyclically by weights. Allows part weights to be just
less than specified:
>>> for item in abjad.sequence.partition_by_weights(
... sequence,
... [10, 5],
... cyclic=True,
... overhang=False,
... allow_part_weights=abjad.LESS,
... ):
... item
...
[3, 3, 3]
[3]
[4, 4]
[4]
[4, 5]
[5]
.. container:: example
Partitions sequence cyclically by weights. Allows part weights to be just
less than specified. Allows overhang:
>>> for item in abjad.sequence.partition_by_weights(
... sequence,
... [10, 5],
... cyclic=True,
... overhang=True,
... allow_part_weights=abjad.LESS,
... ):
... item
...
[3, 3, 3]
[3]
[4, 4]
[4]
[4, 5]
[5]
>>> sequence = list([3, 3, 3, 3, 4, 4, 4, 4, 5, 5])
.. container:: example
Partitions sequence once by weights. Allow part weights to be just more than
specified:
>>> for item in abjad.sequence.partition_by_weights(
... sequence,
... [10, 4],
... cyclic=False,
... overhang=False,
... allow_part_weights=abjad.MORE,
... ):
... item
...
[3, 3, 3, 3]
[4]
.. container:: example
Partitions sequence once by weights. Allows part weights to be just more than
specified. Allows overhang:
>>> for item in abjad.sequence.partition_by_weights(
... sequence,
... [10, 4],
... cyclic=False,
... overhang=True,
... allow_part_weights=abjad.MORE,
... ):
... item
...
[3, 3, 3, 3]
[4]
[4, 4, 4, 5, 5]
.. container:: example
Partitions sequence cyclically by weights. Allows part weights to be just
more than specified:
>>> for item in abjad.sequence.partition_by_weights(
... sequence,
... [10, 4],
... cyclic=True,
... overhang=False,
... allow_part_weights=abjad.MORE,
... ):
... item
...
[3, 3, 3, 3]
[4]
[4, 4, 4]
[5]
.. container:: example
Partitions sequence cyclically by weights. Allows part weights to be just
more than specified. Allows overhang:
>>> for item in abjad.sequence.partition_by_weights(
... sequence,
... [10, 4],
... cyclic=True,
... overhang=True,
... allow_part_weights=abjad.MORE,
... ):
... item
...
[3, 3, 3, 3]
[4]
[4, 4, 4]
[5]
[5]
Returns list of ``sequence`` types.
"""
if allow_part_weights is _enums.EXACT:
candidate = type(sequence)(sequence)
candidate = split(candidate, weights, cyclic=cyclic, overhang=overhang)
flattened_candidate = flatten(candidate, depth=-1)
if flattened_candidate == sequence[: len(flattened_candidate)]:
return candidate
else:
raise Exception("can not partition exactly.")
elif allow_part_weights is _enums.MORE:
if not cyclic:
result = _partition_sequence_once_by_weights_at_least(
sequence, weights, overhang=overhang
)
else:
result = _partition_sequence_cyclically_by_weights_at_least(
sequence, weights, overhang=overhang
)
elif allow_part_weights is _enums.LESS:
if not cyclic:
result = _partition_sequence_once_by_weights_at_most(
sequence, weights, overhang=overhang
)
else:
result = _partition_sequence_cyclically_by_weights_at_most(
sequence, weights, overhang=overhang
)
else:
message = "allow_part_weights must be ordinal constant: {!r}."
message = message.format(allow_part_weights)
raise ValueError(message)
return result
[docs]
def split(
sequence,
weights: typing.Sequence[int | fractions.Fraction],
*,
cyclic: bool = False,
overhang: bool = False,
) -> list:
r"""
Splits ``sequence`` by ``weights``.
.. container:: example
Splits sequence cyclically by weights with overhang:
>>> sequence = list([10, -10, 10, -10])
>>> for part in abjad.sequence.split(
... sequence,
... (3, 15, 3),
... cyclic=True,
... overhang=True,
... ):
... part
...
[3]
[7, -8]
[-2, 1]
[3]
[6, -9]
[-1]
.. container:: example
Splits sequence once by weights with overhang:
>>> for part in abjad.sequence.split(
... sequence,
... (3, 15, 3),
... cyclic=False,
... overhang=True,
... ):
... part
...
[3]
[7, -8]
[-2, 1]
[9, -10]
.. container:: example
Splits sequence once by weights without overhang:
>>> for part in abjad.sequence.split(
... sequence,
... (3, 15, 3),
... cyclic=False,
... overhang=False,
... ):
... part
...
[3]
[7, -8]
[-2, 1]
.. container:: example
REGRESSION. Splits sequence of durations cyclically by weights
with overhang; then expresses durations as pairs with denominator:
>>> sequence = list([
... abjad.Duration(20, 2),
... abjad.Duration(-20, 2),
... abjad.Duration(20, 2),
... abjad.Duration(-20, 2),
... ])
>>> for part in abjad.sequence.split(
... sequence,
... (3, 15, 3),
... cyclic=True,
... overhang=True,
... ):
... [abjad.duration.with_denominator(_, 2) for _ in part]
...
[(6, 2)]
[(14, 2), (-16, 2)]
[(-4, 2), (2, 2)]
[(6, 2)]
[(12, 2), (-18, 2)]
[(-2, 2)]
Returns list of ``sequence`` types.
"""
result = []
current_index = 0
current_piece: list[typing.Any] = []
if cyclic:
weights = repeat_to_weight(
weights, _math.weight(sequence), allow_total=_enums.LESS
)
for weight in weights:
current_piece_weight = _math.weight(current_piece)
while current_piece_weight < weight:
current_piece.append(sequence[current_index])
current_index += 1
current_piece_weight = _math.weight(current_piece)
if current_piece_weight == weight:
current_piece_ = type(sequence)(current_piece)
result.append(current_piece_)
current_piece = []
elif weight < current_piece_weight:
overage = current_piece_weight - weight
current_last_element = current_piece.pop(-1)
needed = abs(current_last_element) - overage
needed *= _math.sign(current_last_element)
current_piece.append(needed)
current_piece_ = type(sequence)(current_piece)
result.append(current_piece_)
overage *= _math.sign(current_last_element)
current_piece = [overage]
if overhang:
last_piece = current_piece
last_piece.extend(sequence[current_index:])
if last_piece:
last_piece_ = type(sequence)(last_piece)
result.append(last_piece_)
return result
[docs]
def filter(sequence, predicate: typing.Callable | None = None):
"""
Filters ``sequence`` by callable ``predicate``.
.. container:: example
By length:
>>> sequence = [[1], [2, 3, [4]], [5], [6, 7, [8]]]
>>> abjad.sequence.filter(sequence, lambda _: len(_) == 1)
[[1], [5]]
By duration:
>>> staff = abjad.Staff("c'4. d'8 e'4. f'8 g'2")
>>> sequence = list(staff)
>>> abjad.sequence.filter(
... sequence, lambda _: _.written_duration == abjad.Duration(1, 8)
... )
[Note("d'8"), Note("f'8")]
Returns ``sequence`` type.
"""
if predicate is None:
return sequence[:]
items = []
for item in sequence:
if predicate(item):
items.append(item)
return type(sequence)(items)
# creates an iterator that can generate a flattened list,
# descending down into child elements to a depth given in the arguments.
# note that depth < 0 is effectively equivalent to infinity.
def _flatten_helper(sequence, classes, depth):
if not isinstance(sequence, classes):
yield sequence
elif depth == 0:
for item in sequence:
yield item
else:
for item in sequence:
# flatten an iterable by one level
depth_ = depth - 1
for item_ in _flatten_helper(item, classes, depth_):
yield item_
[docs]
def flatten(
sequence, *, classes: typing.Sequence[typing.Type] | None = None, depth: int = 1
):
r"""
Flattens ``sequence``.
.. container:: example
Flattens sequence:
>>> sequence = [1, [2, 3, [4]], 5, [6, 7, [8]]]
>>> abjad.sequence.flatten(sequence)
[1, 2, 3, [4], 5, 6, 7, [8]]
Flattens sequence to depth 2:
>>> sequence = [1, [2, 3, [4]], 5, [6, 7, [8]]]
>>> abjad.sequence.flatten(sequence, depth=2)
[1, 2, 3, 4, 5, 6, 7, 8]
Flattens sequence to depth -1:
>>> sequence = [1, [2, 3, [4]], 5, [6, 7, [8]]]
>>> abjad.sequence.flatten(sequence, depth=-1)
[1, 2, 3, 4, 5, 6, 7, 8]
Flattens tuples in sequence only:
>>> sequence = ["ab", "cd", ("ef", "gh"), ("ij", "kl")]
>>> abjad.sequence.flatten(sequence, classes=(tuple, list))
['ab', 'cd', 'ef', 'gh', 'ij', 'kl']
Returns ``sequence`` type.
"""
if classes is None:
classes = (collections.abc.Sequence,)
items = _flatten_helper(sequence, classes, depth)
return type(sequence)(items)
[docs]
def group_by(sequence, predicate: typing.Callable | None = None) -> list:
"""
Groups ``sequence`` items by value of items.
.. container:: example
>>> items = [0, 0, -1, -1, 2, 3, -5, 1, 1, 5, -5]
>>> sequence = list(items)
>>> for item in abjad.sequence.group_by(sequence):
... item
...
[0, 0]
[-1, -1]
[2]
[3]
[-5]
[1, 1]
[5]
[-5]
.. container:: example
>>> staff = abjad.Staff("c'8 d' d' e' e' e'")
>>> predicate = lambda _: abjad.PitchSet([_])
>>> sequence = list(staff)
>>> for item in abjad.sequence.group_by(sequence, predicate):
... item
...
[Note("c'8")]
[Note("d'8"), Note("d'8")]
[Note("e'8"), Note("e'8"), Note("e'8")]
Returns list of ``sequence`` types.
"""
items = []
if predicate is None:
pairs = itertools.groupby(sequence, lambda _: _)
for count, group in pairs:
item = type(sequence)(group)
items.append(item)
else:
pairs = itertools.groupby(sequence, predicate)
for count, group in pairs:
item = type(sequence)(group)
items.append(item)
return items
[docs]
def has_duplicates(sequence):
"""
Is true when ``sequence`` has duplicates.
.. container::
>>> abjad.sequence.has_duplicates([0, 1, 2, 3, 4])
False
>>> abjad.sequence.has_duplicates([0, 1, 2, 3, 3])
True
>>> abjad.sequence.has_duplicates([0, 1, 0, 1, 0])
True
"""
return len(set(sequence)) < len(sequence)
[docs]
def is_decreasing(sequence, *, strict: bool = True) -> bool:
"""
Is true when ``sequence`` is decreasing.
Is true when sequence is strictly decreasing:
.. container:: example
>>> abjad.sequence.is_decreasing([5, 4, 3, 2, 1, 0], strict=True)
True
>>> abjad.sequence.is_decreasing([3, 3, 3, 2, 1, 0], strict=True)
False
>>> abjad.sequence.is_decreasing([3, 3, 3, 3, 3, 3], strict=True)
False
>>> abjad.sequence.is_decreasing([], strict=True)
True
.. container:: example
Is true when sequence decreases monotonically:
>>> abjad.sequence.is_decreasing([5, 4, 3, 2, 1, 0], strict=False)
True
>>> abjad.sequence.is_decreasing([3, 3, 3, 2, 1, 0], strict=False)
True
>>> abjad.sequence.is_decreasing([3, 3, 3, 3, 3, 3], strict=False)
True
>>> abjad.sequence.is_decreasing([], strict=False)
True
"""
if strict:
try:
previous = None
for current in sequence:
if previous is not None:
if not current < previous:
return False
previous = current
return True
except TypeError:
return False
else:
try:
previous = None
for current in sequence:
if previous is not None:
if not current <= previous:
return False
previous = current
return True
except TypeError:
return False
[docs]
def is_increasing(sequence, *, strict: bool = True) -> bool:
"""
Is true when ``sequence`` is increasing.
Is true when sequence is strictly increasing:
.. container:: example
>>> abjad.sequence.is_increasing([0, 1, 2, 3, 4, 5], strict=True)
True
>>> abjad.sequence.is_increasing([0, 1, 2, 3, 3, 3], strict=True)
False
>>> abjad.sequence.is_increasing([3, 3, 3, 3, 3, 3], strict=True)
False
>>> abjad.sequence.is_increasing([], strict=True)
True
.. container:: example
Is true when sequence increases monotonically:
>>> abjad.sequence.is_increasing([0, 1, 2, 3, 4, 5], strict=False)
True
>>> abjad.sequence.is_increasing([0, 1, 2, 3, 3, 3], strict=False)
True
>>> abjad.sequence.is_increasing([3, 3, 3, 3, 3, 3], strict=False)
True
>>> abjad.sequence.is_increasing([], strict=False)
True
"""
if strict:
try:
previous = None
for current in sequence:
if previous is not None:
if not previous < current:
return False
previous = current
return True
except TypeError:
return False
else:
try:
previous = None
for current in sequence:
if previous is not None:
if not previous <= current:
return False
previous = current
return True
except TypeError:
return False
[docs]
def is_permutation(sequence) -> bool:
"""
Is true when ``sequence`` is a permutation.
.. container:: example
>>> abjad.sequence.is_permutation([4, 5, 0, 3, 2, 1])
True
>>> abjad.sequence.is_permutation([1, 1, 5, 3, 2, 1])
False
"""
return tuple(sorted(sequence)) == tuple(range(len(sequence)))
[docs]
def is_repetition_free(sequence) -> bool:
"""
Is true when ``sequence`` is repetition-free.
.. container:: example
>>> abjad.sequence.is_repetition_free([0, 1, 2, 6, 7, 8])
True
>>> abjad.sequence.is_repetition_free([])
True
>>> abjad.sequence.is_repetition_free([0, 1, 2, 2, 7, 8])
False
"""
try:
for left, right in nwise(sequence):
if left == right:
return False
return True
except TypeError:
return False
[docs]
def join(sequence):
"""
Join subsequences in ``sequence``.
.. container:: example
>>> items = [(1, 2, 3), (), (4, 5), (), (6,)]
>>> sequence = list(items)
>>> sequence
[(1, 2, 3), (), (4, 5), (), (6,)]
>>> abjad.sequence.join(sequence)
[(1, 2, 3, 4, 5, 6)]
Returns ``sequence`` type.
"""
if not sequence:
return type(sequence)()
item = sequence[0]
for item_ in sequence[1:]:
item += item_
return type(sequence)([item])
[docs]
def nwise(
sequence, n: int = 2, *, cyclic: bool = False, wrapped: bool = False
) -> typing.Iterator:
"""
Iterates ``sequence`` ``n`` items at a time.
Iterates items 2 at a time:
.. container:: example
>>> sequence = list(range(10))
>>> for item in abjad.sequence.nwise(sequence):
... item
...
[0, 1]
[1, 2]
[2, 3]
[3, 4]
[4, 5]
[5, 6]
[6, 7]
[7, 8]
[8, 9]
Iterates items 3 at a time:
>>> sequence = list(range(10))
>>> for item in abjad.sequence.nwise(sequence, n=3):
... item
...
[0, 1, 2]
[1, 2, 3]
[2, 3, 4]
[3, 4, 5]
[4, 5, 6]
[5, 6, 7]
[6, 7, 8]
[7, 8, 9]
Iterates items 2 at a time. Wraps around at end:
>>> sequence = list(range(10))
>>> for item in abjad.sequence.nwise(sequence, n=2, wrapped=True):
... item
...
[0, 1]
[1, 2]
[2, 3]
[3, 4]
[4, 5]
[5, 6]
[6, 7]
[7, 8]
[8, 9]
[9, 0]
Iterates items 3 at a time. Wraps around at end:
>>> sequence = list(range(10))
>>> for item in abjad.sequence.nwise(sequence, n=3, wrapped=True):
... item
...
[0, 1, 2]
[1, 2, 3]
[2, 3, 4]
[3, 4, 5]
[4, 5, 6]
[5, 6, 7]
[6, 7, 8]
[7, 8, 9]
[8, 9, 0]
[9, 0, 1]
Iterates items 2 at a time. Cycles indefinitely:
>>> sequence = list(range(10))
>>> pairs = abjad.sequence.nwise(sequence, n=2, cyclic=True)
>>> for _ in range(15):
... next(pairs)
...
[0, 1]
[1, 2]
[2, 3]
[3, 4]
[4, 5]
[5, 6]
[6, 7]
[7, 8]
[8, 9]
[9, 0]
[0, 1]
[1, 2]
[2, 3]
[3, 4]
[4, 5]
Returns infinite generator.
Iterates items 3 at a time. Cycles indefinitely:
>>> sequence = list(range(10))
>>> triples = abjad.sequence.nwise(sequence, n=3, cyclic=True)
>>> for _ in range(15):
... next(triples)
...
[0, 1, 2]
[1, 2, 3]
[2, 3, 4]
[3, 4, 5]
[4, 5, 6]
[5, 6, 7]
[6, 7, 8]
[7, 8, 9]
[8, 9, 0]
[9, 0, 1]
[0, 1, 2]
[1, 2, 3]
[2, 3, 4]
[3, 4, 5]
[4, 5, 6]
Returns infinite generator.
Iterates items 1 at a time:
>>> sequence = list(range(10))
>>> for item in abjad.sequence.nwise(sequence, n=1):
... item
...
[0]
[1]
[2]
[3]
[4]
[5]
[6]
[7]
[8]
[9]
Ignores ``wrapped`` when ``cyclic=True``.
"""
if cyclic:
item_buffer = []
long_enough = False
for item in sequence:
item_buffer.append(item)
if not long_enough:
if n <= len(item_buffer):
long_enough = True
if long_enough:
yield type(sequence)(item_buffer[-n:])
len_sequence = len(item_buffer)
current = len_sequence - n + 1
while True:
output = []
for local_offset in range(n):
index = (current + local_offset) % len_sequence
output.append(item_buffer[index])
yield type(sequence)(output)
current += 1
current %= len_sequence
elif wrapped:
first_n_minus_1: list[typing.Any] = []
item_buffer = []
for item in sequence:
item_buffer.append(item)
if len(item_buffer) == n:
yield type(sequence)(item_buffer)
item_buffer.pop(0)
if len(first_n_minus_1) < n - 1:
first_n_minus_1.append(item)
item_buffer = item_buffer + first_n_minus_1
if item_buffer:
for x in range(n - 1):
stop = x + n
yield type(sequence)(item_buffer[x:stop])
else:
item_buffer = []
for item in sequence:
item_buffer.append(item)
if len(item_buffer) == n:
yield type(sequence)(item_buffer)
item_buffer.pop(0)
[docs]
def permute(sequence, permutation: typing.Sequence[int]):
r"""
Permutes ``sequence`` by ``permutation``.
.. container:: example
>>> abjad.sequence.permute([10, 11, 12, 13, 14, 15], [5, 4, 0, 1, 2, 3])
[15, 14, 10, 11, 12, 13]
>>> abjad.sequence.permute([11, 12, 13, 14], [1, 0, 3, 2])
[12, 11, 14, 13]
.. container:: exception
Raises exception when lengths do not match:
>>> abjad.sequence.permute([1, 2, 3, 4, 5, 6], [3, 0, 1, 2])
Traceback (most recent call last):
...
ValueError: permutation [3, 0, 1, 2] must match length of [1, 2, 3, 4, 5, 6].
Returns ``sequence`` type.
"""
permutation = type(sequence)(permutation)
if not is_permutation(permutation):
raise ValueError(f"must be permutation: {permutation!r}.")
if len(permutation) != len(sequence):
message = f"permutation {permutation!r} must match length of {sequence !r}."
raise ValueError(message)
result = []
for i, item in enumerate(sequence):
j = permutation[i]
item_ = sequence[j]
result.append(item_)
return type(sequence)(result)
[docs]
def remove(sequence, pattern):
"""
Removes ``sequence`` items at indices specified by ``pattern``.
.. container:: example
>>> sequence = list(range(15))
>>> abjad.sequence.remove(sequence, abjad.index([2, 3]))
[0, 1, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14]
Removes elements and indices -2 and -3:
>>> abjad.sequence.remove(sequence, abjad.index([-2, -3]))
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 14]
>>> abjad.sequence.remove(sequence, abjad.index([2, 3], 4))
[0, 1, 4, 5, 8, 9, 12, 13]
>>> abjad.sequence.remove(sequence, abjad.index([-2, -3], 4))
[2, 3, 6, 7, 10, 11, 14]
>>> abjad.sequence.remove(sequence, abjad.index([]))
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14]
>>> abjad.sequence.remove(sequence, abjad.index([97, 98, 99]))
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14]
Removes no elements:
>>> abjad.sequence.remove(sequence, abjad.index([-97, -98, -99]))
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14]
Returns ``sequence`` type.
"""
items = []
length = len(sequence)
indices = pattern.indices
period = pattern.period or length
if indices is None:
indices = range(length)
new_indices = []
for i in indices:
if length < abs(i):
continue
if i < 0:
i = length + i
i = i % period
new_indices.append(i)
indices = new_indices
indices.sort()
for i, item in enumerate(sequence):
if i % period not in indices:
items.append(item)
return type(sequence)(items)
[docs]
def remove_repeats(sequence):
"""
Removes repeats from ``sequence``.
.. container:: example
>>> abjad.sequence.remove_repeats([31, 31, 35, 35, 31, 31, 31, 31, 35])
[31, 35, 31, 35]
Returns ``sequence`` type.
"""
items = [sequence[0]]
for item in sequence[1:]:
if item != items[-1]:
items.append(item)
return type(sequence)(items)
[docs]
def repeat(sequence, n: int = 1) -> list:
r"""
Repeats ``sequence``, to a total of ``n`` copies.
.. container:: example
>>> abjad.sequence.repeat([1, 2, 3], n=0)
[]
>>> abjad.sequence.repeat([1, 2, 3], n=1)
[[1, 2, 3]]
>>> abjad.sequence.repeat([1, 2, 3], n=2)
[[1, 2, 3], [1, 2, 3]]
Returns list of ``sequence`` types.
"""
sequences = []
for i in range(n):
sequences.append(sequence[:])
return sequences
[docs]
def repeat_to_length(sequence, length: int = 0, *, start: int = 0):
"""
Repeats ``sequence`` to ``length``.
.. container:: example
Repeats list to length 11:
>>> abjad.sequence.repeat_to_length([0, 1, 2, 3, 4], 11)
[0, 1, 2, 3, 4, 0, 1, 2, 3, 4, 0]
>>> abjad.sequence.repeat_to_length([0, 1, 2, 3, 4], 11, start=2)
[2, 3, 4, 0, 1, 2, 3, 4, 0, 1, 2]
>>> abjad.sequence.repeat_to_length([0, -1, -2, -3, -4], 11)
[0, -1, -2, -3, -4, 0, -1, -2, -3, -4, 0]
>>> abjad.sequence.repeat_to_length([0, -1, -2, -3, -4], 0)
[]
>>> abjad.sequence.repeat_to_length([1, 2, 3], 10, start=100)
[2, 3, 1, 2, 3, 1, 2, 3, 1, 2]
Returns ``sequence`` type.
"""
assert _math.is_nonnegative_integer(length), repr(length)
assert len(sequence), repr(sequence)
items = []
start %= len(sequence)
stop_index = start + length
repetitions = int(math.ceil(float(stop_index) / len(sequence)))
for i in range(repetitions):
for item in sequence:
items.append(item)
return type(sequence)(items[start:stop_index])
[docs]
def repeat_to_weight(
sequence,
weight: int | fractions.Fraction,
*,
allow_total: _enums.Comparison = _enums.EXACT,
):
"""
Repeats ``sequence`` to ``weight``.
.. container:: example
Repeats sequence to weight of 23 exactly:
>>> abjad.sequence.repeat_to_weight([5, -5, -5], 23)
[5, -5, -5, 5, -3]
Repeats sequence to weight of 23 more:
>>> abjad.sequence.repeat_to_weight([5, -5, -5], 23, allow_total=abjad.MORE)
[5, -5, -5, 5, -5]
Repeats sequence to weight of 23 or less:
>>> abjad.sequence.repeat_to_weight([5, -5, -5], 23, allow_total=abjad.LESS)
[5, -5, -5, 5]
>>> sequence = [abjad.Duration(3, 16)]
>>> weight = abjad.Duration(5, 4)
>>> sequence = abjad.sequence.repeat_to_weight(sequence, weight)
>>> sum(sequence)
Duration(5, 4)
>>> [abjad.duration.with_denominator(_, 16) for _ in sequence]
[(3, 16), (3, 16), (3, 16), (3, 16), (3, 16), (3, 16), (2, 16)]
Returns ``sequence`` type.
"""
assert 0 <= weight
if allow_total is _enums.EXACT:
sequence_weight = _math.weight(sequence)
complete_repetitions = int(math.ceil(float(weight) / float(sequence_weight)))
items = list(sequence)
items = complete_repetitions * items
overage = complete_repetitions * sequence_weight - weight
for item in reversed(items):
if 0 < overage:
element_weight = abs(item)
candidate_overage = overage - element_weight
if 0 <= candidate_overage:
overage = candidate_overage
items.pop()
else:
absolute_amount_to_keep = element_weight - overage
assert 0 < absolute_amount_to_keep
signed_amount_to_keep = absolute_amount_to_keep
signed_amount_to_keep *= _math.sign(item)
items.pop()
items.append(signed_amount_to_keep)
break
else:
break
elif allow_total is _enums.LESS:
items = [sequence[0]]
i = 1
while _math.weight(items) < weight:
items.append(sequence[i % len(sequence)])
i += 1
if weight < _math.weight(items):
items = items[:-1]
return type(sequence)(items)
elif allow_total is _enums.MORE:
items = [sequence[0]]
i = 1
while _math.weight(items) < weight:
items.append(sequence[i % len(sequence)])
i += 1
return type(sequence)(items)
else:
raise ValueError(f"is not an ordinal value constant: {allow_total!r}.")
return type(sequence)(items)
[docs]
def replace(sequence, old, new):
"""
Replaces every ``old``-valued item in ``sequence`` with ``new``.
.. container:: example
>>> sequence = [0, 2, 3, 0, 2, 3, 0, 2, 3]
>>> abjad.sequence.replace(sequence, 0, 1)
[1, 2, 3, 1, 2, 3, 1, 2, 3]
Returns ``sequence`` type.
"""
items = []
for item in sequence:
if item == old:
new_copy = copy.copy(new)
items.append(new_copy)
else:
items.append(item)
return type(sequence)(items)
[docs]
def replace_at(sequence, indices, new_material):
"""
Replaces items at ``indices`` with ``new_material``.
.. container:: example
Replaces items at indices 0, 2, 4, 6:
>>> sequence = list(range(16))
>>> abjad.sequence.replace_at(
... sequence,
... ([0], 2),
... (['A', 'B', 'C', 'D'], None),
... )
['A', 1, 'B', 3, 'C', 5, 'D', 7, 8, 9, 10, 11, 12, 13, 14, 15]
Replaces elements at indices 0, 1, 8, 13:
>>> sequence = list(range(16))
>>> abjad.sequence.replace_at(
... sequence,
... ([0, 1, 8, 13], None),
... (['A', 'B', 'C', 'D'], None),
... )
['A', 'B', 2, 3, 4, 5, 6, 7, 'C', 9, 10, 11, 12, 'D', 14, 15]
Replaces every item at even index:
>>> sequence = list(range(16))
>>> abjad.sequence.replace_at(
... sequence,
... ([0], 2),
... (['*'], 1),
... )
['*', 1, '*', 3, '*', 5, '*', 7, '*', 9, '*', 11, '*', 13, '*', 15]
Replaces every element at an index congruent to 0 (mod 6) with ``'A'``; replaces
every element at an index congruent to 2 (mod 6) with ``'B'``:
>>> sequence = list(range(16))
>>> abjad.sequence.replace_at(
... sequence,
... ([0], 2),
... (['A', 'B'], 3),
... )
['A', 1, 'B', 3, 4, 5, 'A', 7, 'B', 9, 10, 11, 'A', 13, 'B', 15]
Returns ``sequence`` type.
"""
assert isinstance(indices, collections.abc.Sequence)
assert len(indices) == 2
index_values, index_period = indices
assert isinstance(index_values, collections.abc.Sequence)
index_values = set(index_values)
assert isinstance(index_period, int | type(None))
assert isinstance(new_material, collections.abc.Sequence)
assert len(new_material) == 2
material_values, material_period = new_material
assert isinstance(material_values, collections.abc.Sequence)
material_values = list(material_values)
assert isinstance(material_period, int | type(None))
maxsize = sys.maxsize
if index_period is None:
index_period = maxsize
if material_period is None:
material_period = maxsize
items = []
material_index = 0
for index, item in enumerate(sequence):
if index % index_period in index_values:
try:
cyclic_material_index = material_index % material_period
material_value = material_values[cyclic_material_index]
items.append(material_value)
except IndexError:
items.append(item)
material_index += 1
else:
items.append(item)
return type(sequence)(items)
[docs]
def retain_pattern(sequence, pattern):
"""
Retains ``sequence`` items at indices matching ``pattern``.
.. container:: example
>>> sequence = list(range(10))
>>> abjad.sequence.retain_pattern(sequence, abjad.index_all())
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
>>> abjad.sequence.retain_pattern(sequence, abjad.index([2, 3]))
[2, 3]
>>> abjad.sequence.retain_pattern(sequence, abjad.index([-2, -3]))
[7, 8]
>>> abjad.sequence.retain_pattern(sequence, abjad.index([2, 3], 4))
[2, 3, 6, 7]
>>> abjad.sequence.retain_pattern(sequence, abjad.index([-2, -3], 4))
[0, 3, 4, 7, 8]
>>> abjad.sequence.retain_pattern(sequence, abjad.index([97, 98, 99]))
[]
>>> abjad.sequence.retain_pattern(sequence, abjad.index([-97, -98, -99]))
[]
Returns ``sequence`` type.
"""
length = len(sequence)
items = []
for i, item in enumerate(sequence):
if pattern.matches_index(i, length):
items.append(item)
return type(sequence)(items)
[docs]
def reverse(sequence, *, recurse: bool = False):
r"""
Reverses ``sequence``.
.. container:: example
Reverses sequence:
>>> sequence = [[1, 2], 3, [4, 5]]
>>> abjad.sequence.reverse(sequence)
[[4, 5], 3, [1, 2]]
Reverses recursively:
>>> segment_1 = abjad.PitchClassSegment([1, 2])
>>> pitch = abjad.NumberedPitch(3)
>>> segment_2 = abjad.PitchClassSegment([4, 5])
>>> sequence = list([segment_1, pitch, segment_2])
>>> for item in abjad.sequence.reverse(sequence, recurse=True):
... item
...
PitchClassSegment([5, 4])
NumberedPitch(3)
PitchClassSegment([2, 1])
Returns ``sequence`` type.
"""
if not recurse:
return type(sequence)(reversed(sequence))
def _reverse_helper(item):
if isinstance(item, collections.abc.Iterable):
subitems_ = [_reverse_helper(_) for _ in reversed(item)]
return type(item)(subitems_)
else:
return item
items = _reverse_helper(sequence[:])
return type(sequence)(items)
[docs]
def rotate(sequence, n: int = 0):
r"""
Rotates ``sequence`` by index ``n``.
.. container:: example
Rotates sequence to the right:
>>> sequence = list(range(10))
>>> abjad.sequence.rotate(sequence, n=4)
[6, 7, 8, 9, 0, 1, 2, 3, 4, 5]
Rotates sequence to the left:
>>> sequence = list(range(10))
>>> abjad.sequence.rotate(sequence, n=-3)
[3, 4, 5, 6, 7, 8, 9, 0, 1, 2]
Rotates sequence to neither the right nor the left:
>>> sequence = list(range(10))
>>> abjad.sequence.rotate(sequence, n=0)
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
Returns ``sequence`` type.
"""
n = n or 0
items = []
if len(sequence):
n = n % len(sequence)
for item in sequence[-n : len(sequence)] + sequence[:-n]:
items.append(item)
return type(sequence)(items)
[docs]
def sum_by_sign(sequence, *, sign: typing.Sequence[int] = (-1, 0, 1)):
"""
Sums consecutive ``sequence`` items by ``sign``.
>>> sequence = [0, 0, -1, -1, 2, 3, -5, 1, 2, 5, -5, -6]
>>> abjad.sequence.sum_by_sign(sequence)
[0, -2, 5, -5, 8, -11]
>>> abjad.sequence.sum_by_sign(sequence, sign=[-1])
[0, 0, -2, 2, 3, -5, 1, 2, 5, -11]
>>> abjad.sequence.sum_by_sign(sequence, sign=[0])
[0, -1, -1, 2, 3, -5, 1, 2, 5, -5, -6]
>>> abjad.sequence.sum_by_sign(sequence, sign=[1])
[0, 0, -1, -1, 5, -5, 8, -5, -6]
>>> abjad.sequence.sum_by_sign(sequence, sign=[-1, 0])
[0, -2, 2, 3, -5, 1, 2, 5, -11]
>>> abjad.sequence.sum_by_sign(sequence, sign=[-1, 1])
[0, 0, -2, 5, -5, 8, -11]
>>> abjad.sequence.sum_by_sign(sequence, sign=[0, 1])
[0, -1, -1, 5, -5, 8, -5, -6]
>>> abjad.sequence.sum_by_sign(sequence, sign=[-1, 0, 1])
[0, -2, 5, -5, 8, -11]
Sums consecutive negative elements when ``-1`` in ``sign``.
Sums consecutive zero-valued elements when ``0`` in ``sign``.
Sums consecutive positive elements when ``1`` in ``sign``.
Returns ``sequence`` type.
"""
items = []
generator = itertools.groupby(sequence, _math.sign)
for current_sign, group in generator:
if current_sign in sign:
items.append(sum(list(group)))
else:
for item in group:
items.append(item)
return type(sequence)(items)
[docs]
def truncate(
sequence,
*,
sum_: int | fractions.Fraction | None = None,
weight: int | fractions.Fraction | None = None,
):
"""
Truncates ``sequence``.
>>> sequence = [-1, 2, -3, 4, -5, 6, -7, 8, -9, 10]
.. container:: example
Truncates sequence to weights ranging from 1 to 10:
>>> for weight in range(1, 11):
... result = abjad.sequence.truncate(sequence, weight=weight)
... print(weight, result)
...
1 [-1]
2 [-1, 1]
3 [-1, 2]
4 [-1, 2, -1]
5 [-1, 2, -2]
6 [-1, 2, -3]
7 [-1, 2, -3, 1]
8 [-1, 2, -3, 2]
9 [-1, 2, -3, 3]
10 [-1, 2, -3, 4]
Truncates sequence to sums ranging from 1 to 10:
>>> for sum_ in range(1, 11):
... result = abjad.sequence.truncate(sequence, sum_=sum_)
... print(sum_, result)
...
1 [-1, 2]
2 [-1, 2, -3, 4]
3 [-1, 2, -3, 4, -5, 6]
4 [-1, 2, -3, 4, -5, 6, -7, 8]
5 [-1, 2, -3, 4, -5, 6, -7, 8, -9, 10]
6 [-1, 2, -3, 4, -5, 6, -7, 8, -9, 10]
7 [-1, 2, -3, 4, -5, 6, -7, 8, -9, 10]
8 [-1, 2, -3, 4, -5, 6, -7, 8, -9, 10]
9 [-1, 2, -3, 4, -5, 6, -7, 8, -9, 10]
10 [-1, 2, -3, 4, -5, 6, -7, 8, -9, 10]
Truncates sequence to zero weight:
>>> abjad.sequence.truncate(sequence, weight=0)
[]
Truncates sequence to zero sum:
>>> abjad.sequence.truncate(sequence, sum_=0)
[]
Ignores ``sum`` when ``weight`` and ``sum`` are both set.
Raises value error on negative ``sum``.
Returns ``sequence`` type.
"""
if weight is not None:
assert 0 <= weight, repr(weight)
items = []
if 0 < weight:
total = 0
for item in sequence:
total += abs(item)
if total < weight:
items.append(item)
else:
sign = _math.sign(item)
trimmed_part = weight - _math.weight(items)
trimmed_part *= sign
items.append(trimmed_part)
break
elif sum_ is not None:
assert 0 <= sum_, repr(sum_)
items = []
if 0 < sum_:
total = 0
for item in sequence:
total += item
if total < sum_:
items.append(item)
else:
items.append(sum_ - sum(items))
break
return type(sequence)(items)
[docs]
def weight(sequence) -> typing.Any:
"""
Gets weight of ``sequence``.
.. container:: example
>>> abjad.sequence.weight([])
0
>>> abjad.sequence.weight([1])
1
>>> abjad.sequence.weight([1, 2, 3])
6
>>> abjad.sequence.weight([1, 2, -3])
6
>>> abjad.sequence.weight([-1, -2, -3])
6
>>> abjad.sequence.weight([-1, 2, -3, 4, -5, 6, -7, 8, -9, 10])
55
>>> abjad.sequence.weight([[1, -7, -7], [1, -8 -8]])
32
"""
weights = []
for item in sequence:
if isinstance(item, collections.abc.Iterable):
weights.append(weight(item))
else:
weights.append(abs(item))
return sum(weights)
[docs]
def zip(sequences, *, cyclic: bool = False, truncate: bool = True) -> list[tuple]:
"""
Zips ``sequences``.
Zips cyclically:
.. container:: example
>>> sequence = [[1, 2, 3], ['a', 'b']]
>>> for item in abjad.sequence.zip(sequence, cyclic=True):
... item
...
(1, 'a')
(2, 'b')
(3, 'a')
>>> sequence = [[10, 11, 12], [20, 21], [30, 31, 32, 33]]
>>> for item in abjad.sequence.zip(sequence, cyclic=True):
... item
...
(10, 20, 30)
(11, 21, 31)
(12, 20, 32)
(10, 21, 33)
Zips without truncation:
>>> sequence = [[1, 2, 3, 4], [11, 12, 13], [21, 22, 23]]
>>> for item in abjad.sequence.zip(sequence, truncate=False):
... item
...
(1, 11, 21)
(2, 12, 22)
(3, 13, 23)
(4,)
.. container:: example
Zips strictly:
>>> sequences = [[1, 2, 3, 4], [11, 12, 13], [21, 22, 23]]
>>> for triple in abjad.sequence.zip(sequences):
... triple
...
(1, 11, 21)
(2, 12, 22)
(3, 13, 23)
Returns list of tuples.
"""
for item in sequences:
if not isinstance(item, collections.abc.Iterable):
raise Exception(f"must be iterable: {item!r}.")
items: list[typing.Any] = []
if cyclic:
if not min(len(_) for _ in sequences):
return items
maximum_length = max([len(_) for _ in sequences])
for i in range(maximum_length):
part = []
for item in sequences:
index = i % len(item)
element = item[index]
part.append(element)
items.append(part)
elif not truncate:
maximum_length = max([len(_) for _ in sequences])
for i in range(maximum_length):
part = []
for item in sequences:
try:
part.append(item[i])
except IndexError:
pass
items.append(part)
elif truncate:
for item in builtins.zip(*sequences):
items.append(item)
result = [tuple(_) for _ in items]
return result