Source code for abjad.label

import dataclasses
import typing

from . import _getlib, _iterlib
from . import bind as _bind
from . import cyclictuple as _cyclictuple
from . import duration as _duration
from . import enums as _enums
from . import indicators as _indicators
from . import iterate as _iterate
from . import overrides as _overrides
from . import pcollections as _pcollections
from . import pitch as _pitch
from . import score as _score
from . import select as _select
from . import setclass as _setclass
from . import tweaks as _tweaks
from . import verticalmoment as _verticalmoment


def _attach(label, leaf, *, deactivate=False, direction=None, tag=None):
    _bind.attach(label, leaf, deactivate=deactivate, direction=direction, tag=tag)


def _color_leaf(leaf, color, *, deactivate=False, tag=None):
    if isinstance(leaf, _score.Skip):
        color = color[1:]
        comment = _indicators.LilyPondLiteral(f"% {color}", site="before")
        _attach(comment, leaf, deactivate=deactivate, tag=tag)
    else:
        assert color.startswith("#")
        string = rf"\abjad-color-music #'{color[1:]}"
        literal = _indicators.LilyPondLiteral(string, site="before")
        _attach(literal, leaf)
    return leaf


pc_number_to_color = {
    0: "#(x11-color 'red)",
    1: "#(x11-color 'MediumBlue)",
    2: "#(x11-color 'orange)",
    3: "#(x11-color 'LightSlateBlue)",
    4: "#(x11-color 'ForestGreen)",
    5: "#(x11-color 'MediumOrchid)",
    6: "#(x11-color 'firebrick)",
    7: "#(x11-color 'DeepPink)",
    8: "#(x11-color 'DarkOrange)",
    9: "#(x11-color 'IndianRed)",
    10: "#(x11-color 'CadetBlue)",
    11: "#(x11-color 'SeaGreen)",
    12: "#(x11-color 'LimeGreen)",
}


[docs] def color_container(container, color="#red") -> None: r""" Colors contents of ``container``. .. container:: example >>> staff = abjad.Staff("c'8 d'8") >>> score = abjad.Score([staff], name="Score") >>> abjad.attach(abjad.TimeSignature((2, 8)), staff[0]) >>> abjad.label.color_container(staff, "#red") >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override Accidental.color = #red \override Beam.color = #red \override Dots.color = #red \override NoteHead.color = #red \override Rest.color = #red \override Stem.color = #red \override TupletBracket.color = #red \override TupletNumber.color = #red } { \time 2/8 c'8 d'8 } """ _overrides.override(container).Accidental.color = color _overrides.override(container).Beam.color = color _overrides.override(container).Dots.color = color _overrides.override(container).NoteHead.color = color _overrides.override(container).Rest.color = color _overrides.override(container).Stem.color = color _overrides.override(container).TupletBracket.color = color _overrides.override(container).TupletNumber.color = color
[docs] def color_leaves(argument, color="#red", *, deactivate=False, tag=None) -> None: r""" Colors leaves in ``argument``. .. container:: example >>> staff = abjad.Staff("cs'8. r8. s8. <c' cs' a'>8.") >>> abjad.label.color_leaves(staff, "#red") >>> lilypond_file = abjad.LilyPondFile([r'\include "abjad.ily"', staff]) >>> abjad.show(lilypond_file) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \abjad-color-music #'red cs'8. \abjad-color-music #'red r8. % red s8. \abjad-color-music #'red <c' cs' a'>8. } """ if isinstance(color, str): for leaf in _iterate.leaves(argument): _color_leaf(leaf, color, deactivate=deactivate, tag=tag) else: assert isinstance(color, typing.Sequence) colors = _cyclictuple.CyclicTuple(color) for i, item in enumerate(argument): color = colors[i] color_leaves(item, color, deactivate=deactivate, tag=tag)
[docs] def color_note_heads(argument, color_map=pc_number_to_color) -> None: r""" Colors note note-heads. .. container:: example >>> chord = abjad.Chord([12, 14, 18, 21, 23], (1, 4)) >>> pitches = [[-12, -10, 4], [-2, 8, 11, 17], [19, 27, 30, 33, 37]] >>> colors = ["#red", "#blue", "#green"] >>> color_map = abjad.ColorMap(colors=colors, pitch_iterables=pitches) >>> abjad.label.color_note_heads(chord, color_map) >>> abjad.show(chord) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(chord) >>> print(string) < \tweak Accidental.color #red \tweak color #red c'' \tweak Accidental.color #red \tweak color #red d'' \tweak Accidental.color #green \tweak color #green fs'' \tweak Accidental.color #green \tweak color #green a'' \tweak Accidental.color #blue \tweak color #blue b'' >4 .. container:: example Colors note note-head: >>> note = abjad.Note("c'4") >>> abjad.label.color_note_heads(note, color_map) >>> abjad.show(note) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(note) >>> print(string) \tweak Accidental.color #red \tweak color #red c'4 Colors nothing: >>> staff = abjad.Staff() >>> abjad.label.color_note_heads(staff, color_map) Colors note-heads: >>> string = "c'8 cs'8 d'8 ds'8 e'8 f'8 fs'8 g'8 gs'8 a'8 as'8 b'8 c''8" >>> staff = abjad.Staff(string) >>> abjad.label.color_note_heads(staff) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { \tweak Accidental.color #(x11-color 'red) \tweak color #(x11-color 'red) c'8 \tweak Accidental.color #(x11-color 'MediumBlue) \tweak color #(x11-color 'MediumBlue) cs'8 \tweak Accidental.color #(x11-color 'orange) \tweak color #(x11-color 'orange) d'8 \tweak Accidental.color #(x11-color 'LightSlateBlue) \tweak color #(x11-color 'LightSlateBlue) ds'8 \tweak Accidental.color #(x11-color 'ForestGreen) \tweak color #(x11-color 'ForestGreen) e'8 \tweak Accidental.color #(x11-color 'MediumOrchid) \tweak color #(x11-color 'MediumOrchid) f'8 \tweak Accidental.color #(x11-color 'firebrick) \tweak color #(x11-color 'firebrick) fs'8 \tweak Accidental.color #(x11-color 'DeepPink) \tweak color #(x11-color 'DeepPink) g'8 \tweak Accidental.color #(x11-color 'DarkOrange) \tweak color #(x11-color 'DarkOrange) gs'8 \tweak Accidental.color #(x11-color 'IndianRed) \tweak color #(x11-color 'IndianRed) a'8 \tweak Accidental.color #(x11-color 'CadetBlue) \tweak color #(x11-color 'CadetBlue) as'8 \tweak Accidental.color #(x11-color 'SeaGreen) \tweak color #(x11-color 'SeaGreen) b'8 \tweak Accidental.color #(x11-color 'red) \tweak color #(x11-color 'red) c''8 } """ color_map = color_map or pc_number_to_color for leaf in _iterate.leaves(argument): if isinstance(leaf, _score.Chord): for note_head in leaf.note_heads: number = note_head.written_pitch.number pc = _pitch.NumberedPitchClass(number) color = color_map.get(pc, None) if color is not None: _tweaks.tweak(note_head, rf"\tweak Accidental.color {color}") _tweaks.tweak(note_head, rf"\tweak color {color}") elif isinstance(leaf, _score.Note): note_head = leaf.note_head number = note_head.written_pitch.number pc = _pitch.NumberedPitchClass(number) color = color_map[pc.number] if color is not None: _tweaks.tweak(leaf.note_head, rf"\tweak Accidental.color {color}") _tweaks.tweak(leaf.note_head, rf"\tweak color {color}")
[docs] def remove_markup(argument) -> None: r""" Removes markup from leaves in ``argument``. .. container:: example >>> staff = abjad.Staff("c'8 d'8 e'8 f'8") >>> abjad.label.with_pitches(staff) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'8 ^ \markup { c' } d'8 ^ \markup { d' } e'8 ^ \markup { e' } f'8 ^ \markup { f' } } >>> abjad.label.remove_markup(staff) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'8 d'8 e'8 f'8 } """ for leaf in _iterate.leaves(argument): _bind.detach(_indicators.Markup, leaf)
[docs] def vertical_moments( argument, direction=_enums.UP, prototype=None, *, deactivate=False, tag=None ): r''' Labels vertical moments. .. container:: example Labels indices: >>> staff_group = abjad.StaffGroup([]) >>> staff = abjad.Staff("c'8 d'4 e'16 f'16") >>> staff_group.append(staff) >>> staff = abjad.Staff(r"""\clef "alto" g4 f4""") >>> staff_group.append(staff) >>> staff = abjad.Staff(r"""\clef "bass" c,2""") >>> staff_group.append(staff) >>> abjad.label.vertical_moments(staff_group) >>> abjad.show(staff_group) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff_group) >>> print(string) \new StaffGroup << \new Staff { c'8 ^ \markup \tiny 0 d'4 ^ \markup \tiny 1 e'16 ^ \markup \tiny 3 f'16 ^ \markup \tiny 4 } \new Staff { \clef "alto" g4 f4 ^ \markup \tiny 2 } \new Staff { \clef "bass" c,2 } >> Labels pitch numbers: >>> staff_group = abjad.StaffGroup([]) >>> staff = abjad.Staff("c'8 d'4 e'16 f'16") >>> staff_group.append(staff) >>> staff = abjad.Staff(r"""\clef "alto" g4 f4""") >>> staff_group.append(staff) >>> staff = abjad.Staff(r"""\clef "bass" c,2""") >>> staff_group.append(staff) >>> abjad.label.vertical_moments( ... staff_group, ... prototype=abjad.NumberedPitch, ... ) >>> abjad.show(staff_group) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff_group) >>> print(string) \new StaffGroup << \new Staff { c'8 ^ \markup \tiny \column { 0 -5 -24 } d'4 ^ \markup \tiny \column { 2 -5 -24 } e'16 ^ \markup \tiny \column { 4 -7 -24 } f'16 ^ \markup \tiny \column { 5 -7 -24 } } \new Staff { \clef "alto" g4 f4 ^ \markup \tiny \column { 2 -7 -24 } } \new Staff { \clef "bass" c,2 } >> .. container:: example Labels pitch-class numbers: >>> staff_group = abjad.StaffGroup([]) >>> staff = abjad.Staff("c'8 d'4 e'16 f'16") >>> staff_group.append(staff) >>> staff = abjad.Staff(r"""\clef "alto" g4 f4""") >>> staff_group.append(staff) >>> staff = abjad.Staff(r"""\clef "bass" c,2""") >>> staff_group.append(staff) >>> prototype = abjad.NumberedPitchClass >>> abjad.label.vertical_moments(staff_group, prototype=prototype) >>> abjad.show(staff_group) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff_group) >>> print(string) \new StaffGroup << \new Staff { c'8 ^ \markup \tiny \column { 7 0 } d'4 ^ \markup \tiny \column { 7 2 0 } e'16 ^ \markup \tiny \column { 5 4 0 } f'16 ^ \markup \tiny \column { 5 0 } } \new Staff { \clef "alto" g4 f4 ^ \markup \tiny \column { 5 2 0 } } \new Staff { \clef "bass" c,2 } >> .. container:: example Labels interval numbers: >>> staff_group = abjad.StaffGroup([]) >>> staff = abjad.Staff("c'8 d'4 e'16 f'16") >>> staff_group.append(staff) >>> staff = abjad.Staff(r"""\clef "alto" g4 f4""") >>> staff_group.append(staff) >>> staff = abjad.Staff(r"""\clef "bass" c,2""") >>> staff_group.append(staff) >>> prototype = abjad.NumberedInterval >>> abjad.label.vertical_moments(staff_group, prototype=prototype) >>> abjad.show(staff_group) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff_group) >>> print(string) \new StaffGroup << \new Staff { c'8 ^ \markup \tiny \column { 15 12 } d'4 ^ \markup \tiny \column { 16 12 } e'16 ^ \markup \tiny \column { 17 11 } f'16 ^ \markup \tiny \column { 18 11 } } \new Staff { \clef "alto" g4 f4 ^ \markup \tiny \column { 16 11 } } \new Staff { \clef "bass" c,2 } >> .. container:: example Labels interval-class numbers: >>> staff_group = abjad.StaffGroup([]) >>> staff = abjad.Staff("c'8 d'4 e'16 f'16") >>> staff_group.append(staff) >>> staff = abjad.Staff(r"""\clef "alto" g4 f4""") >>> staff_group.append(staff) >>> staff = abjad.Staff(r"""\clef "bass" c,2""") >>> staff_group.append(staff) >>> prototype = abjad.NumberedIntervalClass >>> abjad.label.vertical_moments(staff_group, prototype=prototype) >>> abjad.show(staff_group) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff_group) >>> print(string) \new StaffGroup << \new Staff { c'8 ^ \markup \tiny \column { 12 7 } d'4 ^ \markup \tiny \column { 2 7 } e'16 ^ \markup \tiny \column { 4 5 } f'16 ^ \markup \tiny \column { 5 5 } } \new Staff { \clef "alto" g4 f4 ^ \markup \tiny \column { 2 5 } } \new Staff { \clef "bass" c,2 } >> .. container:: example Labels interval-class vectors: >>> staff_group = abjad.StaffGroup([]) >>> staff = abjad.Staff("c'8 d'4 e'16 f'16") >>> staff_group.append(staff) >>> staff = abjad.Staff(r"""\clef "alto" g4 f4""") >>> staff_group.append(staff) >>> staff = abjad.Staff(r"""\clef "bass" c,2""") >>> staff_group.append(staff) >>> prototype = abjad.pcollections.make_interval_class_vector >>> abjad.label.vertical_moments(staff_group, prototype=prototype) >>> abjad.show(staff_group) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff_group) >>> print(string) \new StaffGroup << \new Staff { c'8 ^ \markup \tiny \tiny 1000020 d'4 ^ \markup \tiny \tiny 0010020 e'16 ^ \markup \tiny \tiny 0100110 f'16 ^ \markup \tiny \tiny 1000020 } \new Staff { \clef "alto" g4 f4 ^ \markup \tiny \tiny 0011010 } \new Staff { \clef "bass" c,2 } >> .. container:: example Labels set-classes: >>> staff_group = abjad.StaffGroup([]) >>> staff = abjad.Staff("c'8 d'4 e'16 f'16") >>> staff_group.append(staff) >>> staff = abjad.Staff(r"""\clef "alto" g4 f4""") >>> staff_group.append(staff) >>> staff = abjad.Staff(r"""\clef "bass" c,2""") >>> staff_group.append(staff) >>> prototype = abjad.SetClass() >>> abjad.label.vertical_moments(staff_group, prototype=prototype) >>> abjad.show(staff_group) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff_group) >>> print(string) \new StaffGroup << \new Staff { c'8 ^ \markup \tiny \line { "SC(2-5){0, 5}" } d'4 ^ \markup \tiny \line { "SC(3-9){0, 2, 7}" } e'16 ^ \markup \tiny \line { "SC(3-4){0, 1, 5}" } f'16 ^ \markup \tiny \line { "SC(2-5){0, 5}" } } \new Staff { \clef "alto" g4 f4 ^ \markup \tiny \line { "SC(3-7){0, 2, 5}" } } \new Staff { \clef "bass" c,2 } >> Set ``prototype`` to one of the classes shown above. Returns none. ''' prototype = prototype or int vertical_moments = _verticalmoment.iterate_vertical_moments(argument) for index, vertical_moment in enumerate(vertical_moments): label, string = None, None if prototype is int: string = str(index) elif prototype is _pitch.NumberedPitch: leaves = vertical_moment.leaves generator = _iterate.pitches(leaves) pitches = _pcollections.PitchSegment(generator) if not pitches: continue pitch_numbers = [str(pitch.number) for pitch in pitches] string = rf'\column {{ {" ".join(pitch_numbers)} }}' elif prototype is _pitch.NumberedPitchClass: leaves = vertical_moment.leaves generator = _iterate.pitches(leaves) pitches = _pcollections.PitchSegment(generator) if not pitches: continue pitch_classes = [pitch.pitch_class.number for pitch in pitches] pitch_classes = list(set(pitch_classes)) pitch_classes.sort() pitch_classes.reverse() numbers = [str(_) for _ in pitch_classes] string = rf'\column {{ {" ".join(numbers)} }}' elif prototype is _pitch.NumberedInterval: leaves = vertical_moment.leaves notes = [_ for _ in leaves if isinstance(_, _score.Note)] if not notes: continue notes.sort(key=lambda x: x.written_pitch.number) notes.reverse() bass_note = notes[-1] upper_notes = notes[:-1] named_intervals = [] for upper_note in upper_notes: named_interval = _pitch.NamedInterval.from_pitch_carriers( bass_note.written_pitch, upper_note.written_pitch ) named_intervals.append(named_interval) numbers = [str(x.number) for x in named_intervals] string = rf'\column {{ {" ".join(numbers)} }}' elif prototype is _pitch.NumberedIntervalClass: leaves = vertical_moment.leaves notes = [_ for _ in leaves if isinstance(_, _score.Note)] if not notes: continue notes.sort(key=lambda _: _.written_pitch.number) notes.reverse() bass_note = notes[-1] upper_notes = notes[:-1] numbers = [] for upper_note in upper_notes: interval = _pitch.NamedInterval.from_pitch_carriers( bass_note.written_pitch, upper_note.written_pitch ) interval_class = _pitch.NumberedIntervalClass(interval) number = interval_class.number numbers.append(number) string = " ".join([str(_) for _ in numbers]) string = rf"\column {{ {string} }}" elif prototype is _setclass.SetClass or isinstance( prototype, _setclass.SetClass ): if prototype is _setclass.SetClass: prototype = prototype() assert isinstance(prototype, _setclass.SetClass) leaves = vertical_moment.leaves generator = _iterate.pitches(leaves) pitch_class_set = _pcollections.PitchClassSet(generator) if not pitch_class_set: continue set_class = _setclass.SetClass.from_pitches( pitch_class_set, lex_rank=prototype.lex_rank, transposition_only=prototype.transposition_only, ) string = str(set_class) string = rf'\line {{ "{string}" }}' elif callable(prototype): leaves = vertical_moment.leaves generator = _iterate.pitches(leaves) string = prototype(generator) else: raise TypeError(f"unknown prototype {prototype!r}.") assert string is not None label = _indicators.Markup(rf"\markup \tiny {string}") if direction is _enums.UP: leaf = vertical_moment.start_leaves[0] else: leaf = vertical_moment.start_leaves[-1] _attach(label, leaf, deactivate=deactivate, direction=direction, tag=tag)
[docs] def with_durations( argument, *, denominator=None, direction=_enums.UP, in_seconds: bool = False ): r""" Labels logical ties in ``argument`` with durations. .. container:: example Labels logical tie durations: >>> staff = abjad.Staff(r"c'4. d'8 ~ d'4. e'16 [ ef'16 ]") >>> abjad.label.with_durations(staff) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4. ^ \markup \fraction 3 8 d'8 ^ \markup \fraction 1 2 ~ d'4. e'16 ^ \markup \fraction 1 16 [ ef'16 ^ \markup \fraction 1 16 ] } .. container:: example Labels logical ties with preferred denominator: >>> staff = abjad.Staff(r"c'4. d'8 ~ d'4. e'16 [ ef'16 ]") >>> abjad.label.with_durations(staff, denominator=16) >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff { c'4. ^ \markup \fraction 6 16 d'8 ^ \markup \fraction 8 16 ~ d'4. e'16 ^ \markup \fraction 1 16 [ ef'16 ^ \markup \fraction 1 16 ] } Returns none. """ for logical_tie in _iterate.logical_ties(argument): duration = _getlib._get_duration(logical_tie, in_seconds=in_seconds) pair = duration.pair if denominator is not None: pair = _duration.with_denominator(duration, denominator) n, d = pair label = _indicators.Markup(rf"\markup \fraction {n} {d}") _attach(label, logical_tie.head, direction=direction)
[docs] def with_indices(argument, direction=_enums.UP, prototype=None) -> None: r""" Labels logical ties in ``argument`` with indices. Labels logical tie indices: .. container:: example >>> staff = abjad.Staff("<c' bf'>8 <g' a'>4 af'8 ~ af'8 gf'8 ~ gf'4") >>> abjad.label.with_indices(staff) >>> abjad.override(staff).TextScript.staff_padding = 2 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 2 } { <c' bf'>8 ^ \markup 0 <g' a'>4 ^ \markup 1 af'8 ^ \markup 2 ~ af'8 gf'8 ^ \markup 3 ~ gf'4 } .. container:: example Labels note indices: >>> staff = abjad.Staff("<c' bf'>8 <g' a'>4 af'8 ~ af'8 gf'8 ~ gf'4") >>> abjad.label.with_indices(staff, prototype=abjad.Note) >>> abjad.override(staff).TextScript.staff_padding = 2 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 2 } { <c' bf'>8 <g' a'>4 af'8 ^ \markup 0 ~ af'8 ^ \markup 1 gf'8 ^ \markup 2 ~ gf'4 ^ \markup 3 } .. container:: example Labels chord indices: >>> staff = abjad.Staff("<c' bf'>8 <g' a'>4 af'8 ~ af'8 gf'8 ~ gf'4") >>> abjad.label.with_indices(staff, prototype=abjad.Chord) >>> abjad.override(staff).TextScript.staff_padding = 2 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 2 } { <c' bf'>8 ^ \markup 0 <g' a'>4 ^ \markup 1 af'8 ~ af'8 gf'8 ~ gf'4 } .. container:: example Labels leaf indices: >>> staff = abjad.Staff("<c' bf'>8 <g' a'>4 af'8 ~ af'8 gf'8 ~ gf'4") >>> abjad.label.with_indices(staff, prototype=abjad.Leaf) >>> abjad.override(staff).TextScript.staff_padding = 2 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 2 } { <c' bf'>8 ^ \markup 0 <g' a'>4 ^ \markup 1 af'8 ^ \markup 2 ~ af'8 ^ \markup 3 gf'8 ^ \markup 4 ~ gf'4 ^ \markup 5 } .. container:: example Labels tuplet indices: >>> tuplet = abjad.Tuplet((2, 3), "c'8 [ d'8 e'8 ]") >>> tuplets = abjad.mutate.copy(tuplet, 4) >>> staff = abjad.Staff(tuplets) >>> abjad.label.with_indices(staff, prototype=abjad.Tuplet) >>> abjad.override(staff).TextScript.staff_padding = 2 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 2 } { \tuplet 3/2 { c'8 ^ \markup 0 [ d'8 e'8 ] } \tuplet 3/2 { c'8 ^ \markup 1 [ d'8 e'8 ] } \tuplet 3/2 { c'8 ^ \markup 2 [ d'8 e'8 ] } \tuplet 3/2 { c'8 ^ \markup 3 [ d'8 e'8 ] } } """ if prototype is None: generator = _iterate.logical_ties(argument) else: generator = _iterate.components(argument, prototype=prototype) items = list(generator) for index, item in enumerate(items): label = _indicators.Markup(rf"\markup {index}") leaves = _select.leaves(item) first_leaf = leaves[0] _attach(label, first_leaf, direction=direction)
[docs] def with_intervals(argument, direction=_enums.UP, prototype=None) -> None: r""" Labels consecutive notes in ``argument`` with intervals. Labels consecutive notes with interval names: .. container:: example >>> pitch_numbers = [0, 25, 11, -4, -14, -13, 9, 10] >>> notes = abjad.makers.make_notes(pitch_numbers, [(1, 4)]) >>> staff = abjad.Staff(notes) >>> abjad.label.with_intervals(staff, prototype=None) >>> abjad.override(staff).TextScript.staff_padding = 4 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 4 } { c'4 ^ \markup +A15 cs'''4 ^ \markup -M9 b'4 ^ \markup -A9 af4 ^ \markup -m7 bf,4 ^ \markup +A1 b,4 ^ \markup +m14 a'4 ^ \markup +m2 bf'4 } .. container:: example Labels consecutive notes with interval-class names: >>> pitch_numbers = [0, 25, 11, -4, -14, -13, 9, 10] >>> notes = abjad.makers.make_notes(pitch_numbers, [(1, 4)]) >>> staff = abjad.Staff(notes) >>> prototype = abjad.NamedIntervalClass >>> abjad.label.with_intervals(staff, prototype=prototype) >>> abjad.override(staff).TextScript.staff_padding = 4 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 4 } { c'4 ^ \markup +A1 cs'''4 ^ \markup -M2 b'4 ^ \markup -A2 af4 ^ \markup -m7 bf,4 ^ \markup +A1 b,4 ^ \markup +m7 a'4 ^ \markup +m2 bf'4 } .. container:: example Labels consecutive notes with interval numbers: >>> pitch_numbers = [0, 25, 11, -4, -14, -13, 9, 10] >>> notes = abjad.makers.make_notes(pitch_numbers, [(1, 4)]) >>> staff = abjad.Staff(notes) >>> prototype = abjad.NumberedInterval >>> abjad.label.with_intervals(staff, prototype=prototype) >>> abjad.override(staff).TextScript.staff_padding = 4 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 4 } { c'4 ^ \markup +25 cs'''4 ^ \markup -14 b'4 ^ \markup -15 af4 ^ \markup -10 bf,4 ^ \markup +1 b,4 ^ \markup +22 a'4 ^ \markup +1 bf'4 } .. container:: example Labels consecutive notes with interval-class numbers: >>> pitch_numbers = [0, 25, 11, -4, -14, -13, 9, 10] >>> notes = abjad.makers.make_notes(pitch_numbers, [(1, 4)]) >>> staff = abjad.Staff(notes) >>> prototype = abjad.NumberedIntervalClass >>> abjad.label.with_intervals(staff, prototype=prototype) >>> abjad.override(staff).TextScript.staff_padding = 4 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 4 } { c'4 ^ \markup +1 cs'''4 ^ \markup -2 b'4 ^ \markup -3 af4 ^ \markup -10 bf,4 ^ \markup +1 b,4 ^ \markup +10 a'4 ^ \markup +1 bf'4 } .. container:: example Labels consecutive notes with inversion-equivalent interval-class numbers: >>> pitch_numbers = [0, 25, 11, -4, -14, -13, 9, 10] >>> notes = abjad.makers.make_notes(pitch_numbers, [(1, 4)]) >>> staff = abjad.Staff(notes) >>> prototype = abjad.NumberedInversionEquivalentIntervalClass >>> abjad.label.with_intervals(staff, prototype=prototype) >>> abjad.override(staff).TextScript.staff_padding = 4 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 4 } { c'4 ^ \markup 1 cs'''4 ^ \markup 2 b'4 ^ \markup 3 af4 ^ \markup 2 bf,4 ^ \markup 1 b,4 ^ \markup 2 a'4 ^ \markup 1 bf'4 } """ prototype = prototype or _pitch.NamedInterval for note in _iterate.leaves(argument, _score.Note): label = None next_leaf = _iterlib._get_leaf(note, 1) if isinstance(next_leaf, _score.Note): interval = _pitch.NamedInterval.from_pitch_carriers(note, next_leaf) interval = prototype(interval) if hasattr(interval, "name"): label = _indicators.Markup(rf"\markup {interval.name}") elif isinstance(interval, _pitch.NumberedInversionEquivalentIntervalClass): label = _indicators.Markup(rf"\markup {interval.number}") elif isinstance( interval, _pitch.NumberedIntervalClass | _pitch.NumberedInterval ): label = _indicators.Markup(rf"\markup {interval.signed_string}") if label is not None: _attach(label, note, direction=direction)
[docs] def with_pitches(argument, direction=_enums.UP, locale=None, prototype=None): r""" Labels logical ties in ``argument`` with pitches. .. container:: example Labels logical ties with pitch names: >>> staff = abjad.Staff("<a d' fs'>4 g'4 ~ g'8 r8 fs''4") >>> abjad.label.with_pitches(staff, prototype=None) >>> abjad.override(staff).TextScript.staff_padding = 4 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 4 } { <a d' fs'>4 ^ \markup \column { "fs'" "d'" "a" } g'4 ^ \markup { g' } ~ g'8 r8 fs''4 ^ \markup { fs'' } } .. container:: example Labels logical ties with American pitch names: >>> staff = abjad.Staff("<a d' fs'>4 g'4 ~ g'8 r8 fs''4") >>> abjad.label.with_pitches(staff, locale="us", prototype=None) >>> abjad.override(staff).TextScript.staff_padding = 4 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 4 } { <a d' fs'>4 ^ \markup \column { "F#4" "D4" "A3" } g'4 ^ \markup { G4 } ~ g'8 r8 fs''4 ^ \markup { "F#5" } } .. container:: example Labels logical ties with pitch numbers: >>> staff = abjad.Staff("<a d' fs'>4 g'4 ~ g'8 r8 fs''4") >>> prototype = abjad.NumberedPitch >>> abjad.label.with_pitches(staff, prototype=prototype) >>> abjad.override(staff).TextScript.staff_padding = 4 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 4 } { <a d' fs'>4 ^ \markup \column { 6 2 -3 } g'4 ^ \markup 7 ~ g'8 r8 fs''4 ^ \markup 18 } .. container:: example Labels logical ties with pitch-class numbers: >>> staff = abjad.Staff("<a d' fs'>4 g'4 ~ g'8 r8 fs''4") >>> prototype = abjad.NumberedPitchClass >>> abjad.label.with_pitches(staff, prototype=prototype) >>> abjad.override(staff).TextScript.staff_padding = 4 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 4 } { <a d' fs'>4 ^ \markup \column { 6 2 9 } g'4 ^ \markup 7 ~ g'8 r8 fs''4 ^ \markup 6 } .. container:: example Labels logical ties with pitch names (filtered by selection): >>> voice = abjad.Voice("df''4 c''4 f'4 fs'4 d''4 ds''4") >>> string = 'Horizontal_bracket_engraver' >>> voice.consists_commands.append(string) >>> selections = [voice[:2], voice[-2:]] >>> for selection in selections: ... abjad.horizontal_bracket(selection) ... >>> abjad.label.with_pitches(selections) >>> abjad.override(voice).HorizontalBracket.staff_padding = 3 >>> abjad.override(voice).TextScript.staff_padding = 2 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \consists Horizontal_bracket_engraver \override HorizontalBracket.staff-padding = 3 \override TextScript.staff-padding = 2 } { df''4 ^ \markup { df'' } \startGroup c''4 ^ \markup { c'' } \stopGroup f'4 fs'4 d''4 ^ \markup { d'' } \startGroup ds''4 ^ \markup { ds'' } \stopGroup } .. container:: example Labels logical ties with pitch numbers (filtered by selection): >>> voice = abjad.Voice("df''4 c''4 f'4 fs'4 d''4 ds''4") >>> voice.consists_commands.append('Horizontal_bracket_engraver') >>> selections = [voice[:2], voice[-2:]] >>> for selection in selections: ... abjad.horizontal_bracket(selection) ... >>> prototype = abjad.NumberedPitch >>> abjad.label.with_pitches(selections, prototype=prototype) >>> abjad.override(voice).HorizontalBracket.staff_padding = 3 >>> abjad.override(voice).TextScript.staff_padding = 2 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \consists Horizontal_bracket_engraver \override HorizontalBracket.staff-padding = 3 \override TextScript.staff-padding = 2 } { df''4 ^ \markup 13 \startGroup c''4 ^ \markup 12 \stopGroup f'4 fs'4 d''4 ^ \markup 14 \startGroup ds''4 ^ \markup 15 \stopGroup } .. container:: example Labels logical ties with pitch-class numbers (filtered by selection): >>> voice = abjad.Voice("df''4 c''4 f'4 fs'4 d''4 ds''4") >>> voice.consists_commands.append('Horizontal_bracket_engraver') >>> selections = [voice[:2], voice[-2:]] >>> for selection in selections: ... abjad.horizontal_bracket(selection) ... >>> prototype = abjad.NumberedPitchClass >>> abjad.label.with_pitches(selections, prototype=prototype) >>> abjad.override(voice).HorizontalBracket.staff_padding = 3 >>> abjad.override(voice).TextScript.staff_padding = 2 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \consists Horizontal_bracket_engraver \override HorizontalBracket.staff-padding = 3 \override TextScript.staff-padding = 2 } { df''4 ^ \markup 1 \startGroup c''4 ^ \markup 0 \stopGroup f'4 fs'4 d''4 ^ \markup 2 \startGroup ds''4 ^ \markup 3 \stopGroup } Returns none. """ prototype = prototype or _pitch.NamedPitch logical_ties = _iterate.logical_ties(argument) for logical_tie in logical_ties: leaf = logical_tie.head label = None if prototype is _pitch.NamedPitch: if isinstance(leaf, _score.Note): string = leaf.written_pitch.get_name(locale=locale) if "#" in string: string = '"' + string + '"' label = _indicators.Markup(rf"\markup {{ {string} }}") elif isinstance(leaf, _score.Chord): pitches = leaf.written_pitches pitches = reversed(pitches) names = [] for pitch in pitches: name = pitch.get_name(locale=locale) name = '"' + name + '"' names.append(name) string = " ".join(names) label = _indicators.Markup(rf"\markup \column {{ {string} }}") elif prototype is _pitch.NumberedPitch: if isinstance(leaf, _score.Note): pitch = leaf.written_pitch.number label = _indicators.Markup(rf"\markup {pitch}") elif isinstance(leaf, _score.Chord): pitches = leaf.written_pitches pitches = reversed(pitches) pitches = [str(_.number) for _ in pitches] string = " ".join(pitches) label = _indicators.Markup(rf"\markup \column {{ {string} }}") elif prototype is _pitch.NumberedPitchClass: if isinstance(leaf, _score.Note): pitch = leaf.written_pitch.pitch_class.number label = _indicators.Markup(rf"\markup {pitch}") elif isinstance(leaf, _score.Chord): pitches = leaf.written_pitches pitches = reversed(pitches) pitches = [str(_.pitch_class.number) for _ in pitches] string = " ".join(pitches) label = _indicators.Markup(rf"\markup \column {{ {string} }}") if label is not None: _attach(label, leaf, direction=direction)
[docs] def with_set_classes(argument, direction=_enums.UP, prototype=None): r""" Labels selections ``argument`` with set-classes. .. container:: example Labels selections with Forte-ranked transposition-inversion set-classes: >>> string = "df''8 c''8 bf'8 a'8 f'4. fs'8 g'8 b'8 d''2." >>> voice = abjad.Voice(string) >>> voice.consists_commands.append('Horizontal_bracket_engraver') >>> selections = [voice[:4], voice[-4:]] >>> for selection in selections: ... abjad.horizontal_bracket(selection) ... >>> abjad.label.with_set_classes(selections) >>> abjad.override(voice).HorizontalBracket.staff_padding = 3 >>> abjad.override(voice).TextScript.staff_padding = 2 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \consists Horizontal_bracket_engraver \override HorizontalBracket.staff-padding = 3 \override TextScript.staff-padding = 2 } { df''8 ^ \markup \tiny \line { "SC(4-3){0, 1, 3, 4}" } \startGroup c''8 bf'8 a'8 \stopGroup f'4. fs'8 ^ \markup \tiny \line { "SC(4-20){0, 1, 5, 8}" } \startGroup g'8 b'8 d''2. \stopGroup } .. container:: example Labels selections with lex-ranked transposition-inversion set-classes: >>> string = "df''8 c''8 bf'8 a'8 f'4. fs'8 g'8 b'8 d''2." >>> voice = abjad.Voice(string) >>> voice.consists_commands.append('Horizontal_bracket_engraver') >>> selections = [voice[:4], voice[-4:]] >>> for selection in selections: ... abjad.horizontal_bracket(selection) ... >>> prototype = abjad.SetClass(lex_rank=True) >>> abjad.label.with_set_classes(selections, prototype=prototype) >>> abjad.override(voice).HorizontalBracket.staff_padding = 3 >>> abjad.override(voice).TextScript.staff_padding = 2 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \consists Horizontal_bracket_engraver \override HorizontalBracket.staff-padding = 3 \override TextScript.staff-padding = 2 } { df''8 ^ \markup \tiny \line { "SC(4-6){0, 1, 3, 4}" } \startGroup c''8 bf'8 a'8 \stopGroup f'4. fs'8 ^ \markup \tiny \line { "SC(4-16){0, 1, 5, 8}" } \startGroup g'8 b'8 d''2. \stopGroup } .. container:: example Labels selections with transposition-only set-classes: >>> string = "df''8 c''8 bf'8 a'8 f'4. fs'8 g'8 b'8 d''2." >>> voice = abjad.Voice(string) >>> voice.consists_commands.append('Horizontal_bracket_engraver') >>> selections = [voice[:4], voice[-4:]] >>> for selection in selections: ... abjad.horizontal_bracket(selection) ... >>> prototype = abjad.SetClass(lex_rank=True, transposition_only=True) >>> abjad.label.with_set_classes(selections, prototype=prototype) >>> abjad.override(voice).HorizontalBracket.staff_padding = 3 >>> abjad.override(voice).TextScript.staff_padding = 2 >>> abjad.show(voice) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(voice) >>> print(string) \new Voice \with { \consists Horizontal_bracket_engraver \override HorizontalBracket.staff-padding = 3 \override TextScript.staff-padding = 2 } { df''8 ^ \markup \tiny \line { "SC(4-6){0, 1, 3, 4}" } \startGroup c''8 bf'8 a'8 \stopGroup f'4. fs'8 ^ \markup \tiny \line { "SC(4-16){0, 1, 5, 8}" } \startGroup g'8 b'8 d''2. \stopGroup } Returns none. """ prototype = prototype or _setclass.SetClass() if prototype is _setclass.SetClass: prototype = prototype() assert isinstance(prototype, _setclass.SetClass), repr(prototype) for selection in argument: generator = _iterate.pitches(selection) pitch_class_set = _pcollections.PitchClassSet(generator) if not pitch_class_set: continue set_class = _setclass.SetClass.from_pitches( pitch_class_set, lex_rank=prototype.lex_rank, transposition_only=prototype.transposition_only, ) string = str(set_class) label = _indicators.Markup(rf'\markup \tiny \line {{ "{string}" }}') leaf = selection[0] _attach(label, leaf, direction=direction)
[docs] def with_start_offsets( argument, brackets=None, clock_time=None, direction=None, global_offset=None, markup_command=None, ): r""" Labels logical ties in ``argument`` with start offsets. .. container:: example Labels logical tie start offsets: >>> string = r"\times 2/3 { c'4 d'4 e'4 ~ } e'4 ef'4" >>> staff = abjad.Staff(string) >>> abjad.label.with_start_offsets(staff, direction=abjad.UP) Duration(1, 1) >>> abjad.override(staff).TextScript.staff_padding = 4 >>> abjad.override(staff).TupletBracket.staff_padding = 0 >>> abjad.show(staff) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(staff) >>> print(string) \new Staff \with { \override TextScript.staff-padding = 4 \override TupletBracket.staff-padding = 0 } { \tuplet 3/2 { c'4 ^ \markup { 0 } d'4 ^ \markup { 1/6 } e'4 ^ \markup { 1/3 } ~ } e'4 ef'4 ^ \markup { 3/4 } } .. container:: example Labels logical tie start offsets with clock time: >>> staff = abjad.Staff(r"c'2 d' e' f'") >>> score = abjad.Score([staff]) >>> mark = abjad.MetronomeMark(abjad.Duration(1, 4), 60) >>> abjad.attach(mark, staff[0]) >>> abjad.label.with_start_offsets(staff, clock_time=True) Duration(8, 1) >>> abjad.override(staff).TextScript.staff_padding = 4 >>> abjad.override(staff).TupletBracket.staff_padding = 0 >>> abjad.show(score) # doctest: +SKIP .. docs:: >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff \with { \override TextScript.staff-padding = 4 \override TupletBracket.staff-padding = 0 } { \tempo 4=60 c'2 ^ \markup { 0'00'' } d'2 ^ \markup { 0'02'' } e'2 ^ \markup { 0'04'' } f'2 ^ \markup { 0'06'' } } >> .. container:: example Labels logical tie start offsets with clock time and custom markup command. No PDF shown here because command is custom: >>> staff = abjad.Staff(r"c'2 d' e' f'") >>> score = abjad.Score([staff]) >>> mark = abjad.MetronomeMark(abjad.Duration(1, 4), 60) >>> abjad.attach(mark, staff[0]) >>> abjad.label.with_start_offsets( ... staff, ... clock_time=True, ... markup_command=r'\dark_cyan_markup', ... ) Duration(8, 1) >>> abjad.override(staff).TextScript.staff_padding = 4 >>> abjad.override(staff).TupletBracket.staff_padding = 0 >>> string = abjad.lilypond(score) >>> print(string) \new Score << \new Staff \with { \override TextScript.staff-padding = 4 \override TupletBracket.staff-padding = 0 } { \tempo 4=60 c'2 ^ \dark_cyan_markup { 0'00'' } d'2 ^ \dark_cyan_markup { 0'02'' } e'2 ^ \dark_cyan_markup { 0'04'' } f'2 ^ \dark_cyan_markup { 0'06'' } } >> Returns total duration. """ direction = direction or _enums.UP if global_offset is not None: assert isinstance(global_offset, _duration.Duration) for logical_tie in _iterate.logical_ties(argument): if clock_time: timespan = logical_tie.head._get_timespan(in_seconds=True) start_offset = timespan.start_offset if global_offset is not None: start_offset += global_offset string = start_offset.to_clock_string() else: timespan = logical_tie.head._get_timespan() start_offset = timespan.start_offset if global_offset is not None: start_offset += global_offset string = str(start_offset) if brackets: string = f"[{string}]" if markup_command is not None: label = _indicators.Markup(rf"{markup_command} {{ {string} }}") else: label = _indicators.Markup(rf"\markup {{ {string} }}") _attach(label, logical_tie.head, direction=direction) total_duration = _duration.Duration(timespan.stop_offset) if global_offset is not None: total_duration += global_offset return total_duration
[docs] @dataclasses.dataclass(slots=True) class ColorMap: """ Color map. .. container:: example Maps pitch-classes to red, green and blue: >>> color_map = abjad.ColorMap( ... colors=["#red", "#green", "#blue"], ... pitch_iterables=[ ... [-8, 2, 10, 21], ... [0, 11, 32, 41], ... [15, 25, 42, 43], ... ], ... ) >>> color_map ColorMap(colors=['#red', '#green', '#blue'], pitch_iterables=[[-8, 2, 10, 21], [0, 11, 32, 41], [15, 25, 42, 43]]) """ colors: typing.Any = None pitch_iterables: typing.Any = None _color_dictionary: dict = dataclasses.field(compare=False, init=False, repr=False)
[docs] def __post_init__(self): self.pitch_iterables = self.pitch_iterables or [] self.colors = self.colors or [] assert len(self.pitch_iterables) == len(self.colors) self._color_dictionary = {} self._initialize_color_dictionary()
[docs] def __getitem__(self, pitch_class) -> str: """ Gets ``pitch_class`` color. .. container:: example >>> color_map = abjad.ColorMap( ... colors=["#red", "#green", "#blue"], ... pitch_iterables=[ ... [-8, 2, 10, 21], ... [0, 11, 32, 41], ... [15, 25, 42, 43], ... ], ... ) >>> color_map[11] '#green' """ pitch_class = _pitch.NumberedPitchClass(pitch_class) return self._color_dictionary[pitch_class.number]
def _initialize_color_dictionary(self): for pitch_iterable, color in zip(self.pitch_iterables, self.colors): for pitch in pitch_iterable: pc = _pitch.NumberedPitchClass(pitch) keys = set(self._color_dictionary.keys()) if pc.number in keys: print(pc, list(self._color_dictionary.keys())) raise KeyError("duplicated pitch-class in color map: {pc!r}.") self._color_dictionary[pc.number] = color
[docs] def __hash__(self): """ Makes hash. """ return hash(repr(self))
@property def is_twelve_tone_complete(self) -> bool: """ Is true when color map contains all 12-ET pitch-classes. .. container:: example >>> color_map = abjad.ColorMap( ... colors=["#red", "#green", "#blue"], ... pitch_iterables=[ ... [-8, 2, 10, 21], ... [0, 11, 32, 41], ... [15, 25, 42, 43], ... ], ... ) >>> color_map.is_twelve_tone_complete True """ pcs = range(12) return set(pcs).issubset(set(self._color_dictionary.keys())) @property def is_twenty_four_tone_complete(self) -> bool: """ Is true when color map contains all 24-ET pitch-classes. .. container:: example >>> color_map = abjad.ColorMap( ... colors=["#red", "#green", "#blue"], ... pitch_iterables=[ ... [-8, 2, 10, 21], ... [0, 11, 32, 41], ... [15, 25, 42, 43], ... ], ... ) >>> color_map.is_twenty_four_tone_complete False """ pcs = [x / 2.0 for x in range(24)] pcs = [int(x) if int(x) == x else x for x in pcs] return set(pcs).issubset(set(self._color_dictionary.keys())) @property def pairs(self) -> list[tuple[int, str]]: """ Gets pairs. .. container:: example >>> color_map = abjad.ColorMap( ... colors=["#red", "#green", "#blue"], ... pitch_iterables=[ ... [-8, 2, 10, 21], ... [0, 11, 32, 41], ... [15, 25, 42, 43], ... ], ... ) >>> for pair in color_map.pairs: ... pair ... (0, '#green') (1, '#blue') (2, '#red') (3, '#blue') (4, '#red') (5, '#green') (6, '#blue') (7, '#blue') (8, '#green') (9, '#red') (10, '#red') (11, '#green') """ items = list(self._color_dictionary.items()) return list(sorted(items))
[docs] def get(self, key, alternative=None) -> str: """ Gets ``key`` from color map. .. container:: example >>> color_map = abjad.ColorMap( ... colors=["#red", "#green", "#blue"], ... pitch_iterables=[ ... [-8, 2, 10, 21], ... [0, 11, 32, 41], ... [15, 25, 42, 43], ... ], ... ) >>> color_map.get(11) '#green' Returns ``alternative`` when ``key`` is not found. """ try: return self[key] except (KeyError, TypeError, ValueError): return alternative