music21.duration

The duration module contains Duration objects (among other objects and functions). Duration objects are a fundamental component of Note and all Music21Objects, such as TimeSignature objects.

Containers such as Stream and Score also have durations which are equal to the position of the ending of the last object in the Stream.

Music21 Durations are almost always measured in Quarter Notes, so an eighth note has a duration of 0.5. Different Duration-like objects support objects such as grace notes which take no duration on the page, have a short (but real) duration when played, and have a duration-type representation when performed.

Example usage:

>>> d = duration.Duration()
>>> d.quarterLength = 0.5
>>> d.type
'eighth'
>>> d.type = 'whole'
>>> d.quarterLength
4.0
>>> d.quarterLength = 0.166666666
>>> d.type
'16th'
>>> d.tuplets[0].numberNotesActual
3
>>> d.tuplets[0].numberNotesNormal
2

Duration

class music21.duration.Duration(typeOrDuration: str | OffsetQLIn | DurationTuple | None = None, /, *, type: str | None = None, dots: int | None = 0, quarterLength: OffsetQLIn | None = None, durationTuple: DurationTuple | None = None, components: Iterable[DurationTuple] | None = None, client: base.Music21Object | None = None, **keywords)

Durations are one of the most important objects in music21. A Duration represents a span of musical time measurable in terms of quarter notes (or in advanced usage other units). For instance, “57 quarter notes” or “dotted half tied to quintuplet sixteenth note” or simply “quarter note.”

A Duration object is made of one or more immutable DurationTuple objects stored on the components list. A Duration created by setting quarterLength sets the attribute expressionIsInferred to True, which indicates that callers (such as splitElementsToCompleteTuplets()) can express this Duration using another combination of components that sums to the quarterLength. Otherwise, expressionIsInferred is set to False, indicating that components are not allowed to mutate.

Multiple DurationTuples in a single Duration may be used to express tied notes, or may be used to split duration across barlines or beam groups. Some Duration objects are not expressible as a single notation unit.

Duration objects are not Music21Objects.

If a single argument is passed to Duration() and it is a string, then it is assumed to be a type, such as ‘half’, ‘eighth’, or ‘16th’, etc. If that single argument is a number then it is assumed to be a quarterLength (2 for half notes, 0.5 for eighth notes, 0.75 for dotted eighth notes, 0.333333333 for a triplet eighth, etc.). If one or more named arguments are passed then the Duration() is configured according to those arguments. Supported arguments are ‘type’, ‘dots’, ‘quarterLength’, or ‘components’.

Example 1: a triplet eighth configured by quarterLength:

>>> d = duration.Duration(0.333333333)
>>> d.type
'eighth'
>>> d.tuplets
(<music21.duration.Tuplet 3/2/eighth>,)

Example 2: A Duration made up of multiple music21.duration.DurationTuple objects automatically configured by the specified quarterLength.

>>> d2 = duration.Duration(0.625)
>>> d2.type
'complex'
>>> d2.components
(DurationTuple(type='eighth', dots=0, quarterLength=0.5),
 DurationTuple(type='32nd', dots=0, quarterLength=0.125))
>>> d2.expressionIsInferred
True

Example 3: A Duration configured by keywords.

>>> d3 = duration.Duration(type='half', dots=2)
>>> d3.quarterLength
3.5
>>> d3.expressionIsInferred
False

Duration bases

Duration read-only properties

Duration.fullName

Return the most complete representation of this Duration, providing dots, type, tuplet, and quarter length representation.

>>> d = duration.Duration(quarterLength=1.5)
>>> d.fullName
'Dotted Quarter'
>>> d = duration.Duration(type='half')
>>> d.fullName
'Half'
>>> d = duration.Duration(quarterLength=1.25)
>>> d.fullName
'Quarter tied to 16th (1 1/4 total QL)'
>>> d = duration.Duration(quarterLength=0.333333)
>>> d.fullName
'Eighth Triplet (1/3 QL)'
>>> d = duration.Duration(quarterLength=0.666666)
>>> d.fullName
'Quarter Triplet (2/3 QL)'
>>> d = duration.Duration(quarterLength=0.571428)
>>> d.fullName
'Quarter Septuplet (4/7 QL)'
>>> d = duration.Duration(quarterLength=0)
>>> d.fullName
'Zero Duration (0 total QL)'
Duration.isComplex

Returns True if this Duration has more than one DurationTuple object on the component list. That is to say if it’s a single Duration that need multiple tied noteheads to represent.

>>> aDur = duration.Duration()
>>> aDur.quarterLength = 1.375
>>> aDur.isComplex
True
>>> len(aDur.components)
2
>>> aDur.components
(DurationTuple(type='quarter', dots=0, quarterLength=1.0),
 DurationTuple(type='16th', dots=1, quarterLength=0.375))
>>> cDur = duration.Duration()
>>> cDur.quarterLength = 0.25
>>> cDur.isComplex
False
>>> len(cDur.components)
1
Duration.ordinal

Get the ordinal value of the Duration, where whole is 4, half is 5, etc.

>>> d = duration.Duration()
>>> d.quarterLength = 2.0
>>> d.ordinal
5
>>> d.type = '16th'
>>> d.ordinal
8

Complex values have an ordinal of the string ‘complex’. This might change to NaN in a later version.

>>> d.quarterLength = 2.5
>>> d.ordinal
'complex'

Zero durations have an ordinal of None

>>> d2 = duration.Duration(0.0)
>>> print(d2.ordinal)
None
Duration.quarterLengthNoTuplets

Returns the quarter length of the duration without taking into account triplets.

Does not cache.

>>> d = duration.Duration(1/3)
>>> d.quarterLengthNoTuplets
0.5

Read-only properties inherited from ProtoM21Object:

Duration read/write properties

Duration.components

Returns or sets a tuple of the component DurationTuples of this Duration object

>>> d = duration.Duration(1.0)
>>> d.components
(DurationTuple(type='quarter', dots=0, quarterLength=1.0),)

Tuplets do not have the tuplet in their components.

>>> d = duration.Duration(1/3)
>>> d.components
(DurationTuple(type='eighth', dots=0, quarterLength=0.5),)

With a complex duration it becomes clearer why multiple components are needed. Here is a duration that cannot be expressed as a single note.

>>> d = duration.Duration(1.25)
>>> d.type
'complex'
>>> d.components
(DurationTuple(type='quarter', dots=0, quarterLength=1.0),
 DurationTuple(type='16th', dots=0, quarterLength=0.25))

But it can be expressed another way and will output in that way in MusicXML and other readers:

>>> component0 = duration.DurationTuple(type='eighth', dots=0, quarterLength=0.5)
>>> component1 = duration.DurationTuple(type='eighth', dots=1, quarterLength=0.75)
>>> d.components = [component0, component1]

It is allowed but not advised to set components that do not add up to the current (pre-tuplet) quarterLength. In which case the quarterLength will be adjusted:

>>> d.components = [component0]
>>> d
<music21.duration.Duration 0.5>
>>> d.type
'eighth'
Duration.dotGroups

Dot groups are a convenience for transcribing medieval music. They represent dotted-dotted notes (written one above another). For instance a half note with dotGroups = (1, 1) represents a dotted half note that is itself dotted. Worth 9 eighth notes (dotted-half tied to dotted-quarter). It is not the same as a double-dotted half note, which is only worth 7 eighth notes.

>>> a = duration.Duration()
>>> a.type = 'half'
>>> a.dotGroups
(0,)
>>> a.dots = 1
>>> a.dotGroups = (1, 1)
>>> a.quarterLength
4.5
Duration.dots

Returns or sets the number of dots in the Duration if it is a simple Duration.

For returning only the number of dots on the first component is returned for complex durations. (Previously it could return None if it was not a simple duration which led to some terribly difficult to find errors.)

>>> a = duration.Duration()
>>> a.type = 'quarter'
>>> a.dots = 1
>>> a.quarterLength
1.5
>>> a.dots = 2
>>> a.quarterLength
1.75

If a duration is complex then setting dots has the effect of setting the number of dots to value on every component.

>>> DT = duration.durationTupleFromTypeDots
>>> complexDuration = duration.Duration()
>>> complexDuration.addDurationTuple(DT('half', 0))
>>> complexDuration.addDurationTuple(DT('eighth', 2))
>>> complexDuration.type
'complex'
>>> complexDuration.quarterLength
2.875

In a complex duration, the number of dots comes from the first component:

>>> complexDuration.dots
0

But if set, applies to all components:

>>> complexDuration.dots = 1
>>> complexDuration.components
(DurationTuple(type='half', dots=1, quarterLength=3.0),
 DurationTuple(type='eighth', dots=1, quarterLength=0.75))
>>> complexDuration.quarterLength
3.75

Dots can go pretty high.

>>> d = duration.Duration('half')
>>> d.quarterLength
2.0
>>> d.dots = 5
>>> d.quarterLength
3.9375
>>> d.dots = 10
>>> d.quarterLength
3.998046875

Infinite dots… (an Easter egg…)

>>> from math import inf
>>> d.type = 'half'
>>> d.dots = inf
>>> d.quarterLength
4.0
>>> d.dots
0
>>> d.type
'whole'
Duration.linked

Gets or sets the .linked property – if linked (default) then type, dots, tuplets are always coherent with quarterLength. If not, then they are separate.

>>> d = duration.Duration(0.5)
>>> d.linked
True

Linked durations change other values when one changes:

>>> d.type = '16th'
>>> d.quarterLength
0.25

Unlinked values do not:

>>> d.linked = False
>>> d.type = 'half'
>>> d.quarterLength
0.25
Duration.quarterLength

Returns the quarter note length or Sets the quarter note length to the specified value. May be expressed as a float or Fraction.

Currently (if the value is different from what is already stored) this wipes out any existing components, not preserving their type. So if you’ve set up Duration(1.5) as 3-eighth notes, setting Duration to 1.75 will NOT dot the last eighth note, but instead give you a single double-dotted half note.

>>> a = duration.Duration()
>>> a.quarterLength = 3.5
>>> a.quarterLength
3.5
>>> for thisUnit in a.components:
...    print(duration.unitSpec(thisUnit))
(3.5, 'half', 2, None, None, None)
>>> a.quarterLength = 2.5
>>> a.quarterLength
2.5
>>> for thisUnit in a.components:
...    print(duration.unitSpec(thisUnit))
(2.0, 'half', 0, None, None, None)
(0.5, 'eighth', 0, None, None, None)

Note that integer values of quarter lengths get silently converted to floats (internally opFracs):

>>> b = duration.Duration()
>>> b.quarterLength = 5
>>> b.quarterLength
5.0
>>> b.type  # complex because 5qL cannot be expressed as a single note.
'complex'

Float values will be converted to fractions if they are inexpressible exactly as floats:

>>> b = duration.Duration()
>>> b.quarterLength = 1/3
>>> b.quarterLength
Fraction(1, 3)
Duration.tuplets

Return a tuple of Tuplet objects. Setting tuplets will inform the client (Note) that the duration has changed.

Duration.type

Get or set the type of the Duration.

>>> a = duration.Duration()
>>> a.type = 'half'
>>> a.quarterLength
2.0
>>> a.type= '16th'
>>> a.quarterLength
0.25

Duration methods

Duration.__eq__(other)

Two durations are the same if their type, dots, tuplets, and quarterLength are all the same.

>>> aDur = duration.Duration('quarter')
>>> bDur = duration.Duration('16th')
>>> cDur = duration.Duration('16th')
>>> aDur == bDur
False
>>> aDur != bDur
True
>>> cDur == bDur
True
>>> dDur = duration.Duration(0.0)
>>> eDur = duration.Duration(0.0)
>>> dDur == eDur
True
>>> tupDur1 = duration.Duration(2 / 3)
>>> tupDur2 = duration.Duration(2 / 3)
>>> tupDur1 == tupDur2
True
>>> graceDur1 = tupDur1.getGraceDuration()
>>> graceDur1 == tupDur1
False
>>> graceDur2 = tupDur2.getGraceDuration()
>>> graceDur1 == graceDur2
True

Link status must be the same:

>>> tupDur1.linked = False
>>> tupDur1 == tupDur2
False
Duration.addDurationTuple(dur: DurationTuple | Duration | str | int | float | Fraction, *, _skipInform=False)

Add a DurationTuple or a Duration’s components to this Duration. Does not simplify the Duration. For instance, adding two quarter notes results in two tied quarter notes, not one half note. See consolidate below for more info on how to do that.

>>> a = duration.Duration('quarter')
>>> b = duration.durationTupleFromTypeDots('quarter', 0)
>>> a.addDurationTuple(b)
>>> a.quarterLength
2.0
>>> a.type
'complex'
Duration.aggregateTupletMultiplier() float | Fraction

Returns the multiple of all the tuplet multipliers as an opFrac.

This method is needed for MusicXML time-modification among other places.

No tuplets…

>>> complexDur = duration.Duration('eighth')
>>> complexDur.aggregateTupletMultiplier()
1.0

With tuplets:

>>> complexDur.appendTuplet(duration.Tuplet())
>>> complexDur.aggregateTupletMultiplier()
Fraction(2, 3)

Nested tuplets are possible…

>>> tup2 = duration.Tuplet()
>>> tup2.setRatio(5, 4)
>>> complexDur.appendTuplet(tup2)
>>> complexDur.aggregateTupletMultiplier()
Fraction(8, 15)
Duration.appendTuplet(newTuplet: Tuplet) None

Adds a new Tuplet to a Duration, sets the Tuplet’s .frozen state to True, and then informs the client (Note) that the duration has changed.

>>> tup = duration.Tuplet(3, 2)
>>> d = duration.Duration(1.0)
>>> d.appendTuplet(tup)
>>> d.quarterLength
Fraction(2, 3)
>>> t2 = duration.Tuplet(5, 4)
>>> d.appendTuplet(t2)
>>> d.quarterLength
Fraction(8, 15)
>>> tup.frozen
True
Duration.augmentOrDiminish(amountToScale, retainComponents=False)

Given a number greater than zero, creates a new Duration object after multiplying the current quarterLength of the duration by the number and resets the components for the duration (by default).

Returns a new duration that has the new length.

>>> aDur = duration.Duration()
>>> aDur.quarterLength = 1.5  # dotted quarter
>>> cDur = aDur.augmentOrDiminish(2)
>>> cDur.quarterLength
3.0
>>> cDur.type
'half'
>>> cDur.dots
1

aDur is not changed:

>>> aDur
<music21.duration.Duration 1.5>

A complex duration that cannot be expressed as a single notehead (component)

>>> bDur = duration.Duration()
>>> bDur.quarterLength = 2.125  # requires components
>>> bDur.quarterLength
2.125
>>> len(bDur.components)
2
>>> bDur.components
(DurationTuple(type='half', dots=0, quarterLength=2.0),
 DurationTuple(type='32nd', dots=0, quarterLength=0.125))

By default, when augmenting or diminishing, we will delete any unusual components or tuplets:

>>> dDur = duration.Duration(1.5)
>>> dDur.appendTuplet(duration.Tuplet(3, 2))
>>> dDur
<music21.duration.Duration 1.0>
>>> dDur.dots
1
>>> dDur.tuplets
(<music21.duration.Tuplet 3/2>,)
>>> eDur = dDur.augmentOrDiminish(2)
>>> eDur
<music21.duration.Duration 2.0>
>>> eDur.dots
0
>>> eDur.tuplets
()
>>> eRetain = dDur.augmentOrDiminish(2, retainComponents=True)
>>> eRetain
<music21.duration.Duration 2.0>
>>> eRetain.dots
1
>>> eRetain.tuplets
(<music21.duration.Tuplet 3/2>,)
>>> fDur = duration.Duration(1.0)
>>> fDur.addDurationTuple(duration.DurationTuple('quarter', 0, 1.0))
>>> fDur
<music21.duration.Duration 2.0>
>>> fDur.components
(DurationTuple(type='quarter', dots=0, quarterLength=1.0),
 DurationTuple(type='quarter', dots=0, quarterLength=1.0))
>>> gDur = fDur.augmentOrDiminish(0.5)
>>> gDur.components
(DurationTuple(type='quarter', dots=0, quarterLength=1.0),)
>>> gRetain = fDur.augmentOrDiminish(0.5, retainComponents=True)
>>> gRetain.components
(DurationTuple(type='eighth', dots=0, quarterLength=0.5),
 DurationTuple(type='eighth', dots=0, quarterLength=0.5))

Negative values raise ValueError:

>>> fDur.augmentOrDiminish(-1)
Traceback (most recent call last):
ValueError: amountToScale must be greater than zero
Duration.clear() None

Permit all components to be removed. This is needed for resetting to zero duration.

>>> a = duration.Duration()
>>> a.quarterLength = 6
>>> a.type
'whole'
>>> a.components
(DurationTuple(type='whole', dots=1, quarterLength=6.0),)
>>> a.clear()
>>> a.dots
0
>>> a.components
()
>>> a.type
'zero'
>>> a.quarterLength
0.0
Duration.componentIndexAtQtrPosition(quarterPosition)

returns the index number of the duration component sounding at the given quarter position.

Note that for 0 and the last value, the object is returned.

>>> components = []
>>> components.append(duration.Duration('quarter'))
>>> components.append(duration.Duration('quarter'))
>>> components.append(duration.Duration('quarter'))
>>> a = duration.Duration()
>>> a.components = components
>>> a.quarterLength
3.0
>>> a.componentIndexAtQtrPosition(0.5)
0
>>> a.componentIndexAtQtrPosition(1.5)
1
>>> a.componentIndexAtQtrPosition(2.5)
2

this is odd behavior:

e.g. given d1, d2, d3 as 3 quarter notes and self.components = [d1, d2, d3]

then

self.componentIndexAtQtrPosition(1.5) == d2 self.componentIndexAtQtrPosition(2.0) == d3 self.componentIndexAtQtrPosition(2.5) == d3

Currently sometimes returns the component itself. Changing in v7.

Duration.componentStartTime(componentIndex: int) float

For a valid component index value, this returns the quarter note offset at which that component would start.

This does not handle fractional arguments.

>>> components = []
>>> qdt = duration.DurationTuple('quarter', 0, 1.0)
>>> components.append(qdt)
>>> components.append(qdt)
>>> components.append(qdt)
>>> a = duration.Duration()
>>> a.components = components
>>> a.quarterLength
3.0
>>> a.componentStartTime(0)
0.0
>>> a.componentStartTime(1)
1.0
>>> a.componentStartTime(3)
Traceback (most recent call last):
IndexError: invalid component index value 3 submitted;
            value must be an integer between 0 and 2
Duration.consolidate()

Given a Duration with multiple components, consolidate into a single Duration. This can only be based on quarterLength; this is destructive: information is lost from components.

This cannot be done for all Durations, as DurationTuples cannot express all durations

>>> a = duration.Duration(1)
>>> a.addDurationTuple(duration.DurationTuple('half', 0, 2.0))
>>> a.addDurationTuple(duration.DurationTuple('quarter', 0, 1.0))
>>> a.quarterLength
4.0
>>> len(a.components)
3
>>> a.type
'complex'

After consolidate:

>>> a.consolidate()
>>> a.quarterLength
4.0
>>> len(a.components)
1

It gains a type!

>>> a.type
'whole'

If the type cannot be expressed then the type is inexpressible

>>> a = duration.Duration(1)
>>> a.addDurationTuple(duration.DurationTuple('half', 0, 2.0))
>>> a.addDurationTuple(duration.DurationTuple('half', 0, 2.0))
>>> a.quarterLength
5.0
>>> len(a.components)
3
>>> a.type
'complex'

After consolidate:

>>> a.consolidate()
>>> a.quarterLength
5.0
>>> len(a.components)
1
>>> a.components
(DurationTuple(type='inexpressible', dots=0, quarterLength=5.0),)

It gains a type!

>>> a.type
'inexpressible'

For an ‘inexpressible’ duration, the opposite of consolidate is to set the duration’s quarterLength to itself. It won’t necessarily return to the original components, but it will usually create something that can be notated.

>>> a.quarterLength = a.quarterLength
>>> a.type
'complex'
>>> a.components
(DurationTuple(type='whole', dots=0, quarterLength=4.0),
 DurationTuple(type='quarter', dots=0, quarterLength=1.0))
Duration.currentComponents()

Advanced Method:

returns the current components WITHOUT running the component updater.

Needed by some internal methods. Components are made on the fly.

>>> d = duration.Duration(1.25)
>>> d.currentComponents()
()

Like in quantum physics, an observation affects the state:

>>> d.type
'complex'
>>> d.currentComponents()
(DurationTuple(type='quarter', dots=0, quarterLength=1.0),
 DurationTuple(type='16th', dots=0, quarterLength=0.25))

Generally, just look at .components

>>> d = duration.Duration(1.25)
>>> d.components
(DurationTuple(type='quarter', dots=0, quarterLength=1.0),
 DurationTuple(type='16th', dots=0, quarterLength=0.25))
Duration.getGraceDuration(appoggiatura=False) GraceDuration | AppoggiaturaDuration

Return a deepcopy of this Duration as a GraceDuration instance with the same types.

>>> d = duration.Duration(1.25)
>>> d
<music21.duration.Duration 1.25>
>>> d.components
(DurationTuple(type='quarter', dots=0, quarterLength=1.0),
 DurationTuple(type='16th', dots=0, quarterLength=0.25))
>>> gd = d.getGraceDuration()
>>> gd
<music21.duration.GraceDuration unlinked type:complex quarterLength:0.0>
>>> gd.quarterLength
0.0
>>> gd.components
(DurationTuple(type='quarter', dots=0, quarterLength=0.0),
 DurationTuple(type='16th', dots=0, quarterLength=0.0))

d is unchanged.

>>> d.quarterLength
1.25
Duration.informClient() bool

A method that tells the client that something has changed.

Call informSites({‘changedAttribute’: ‘duration’, ‘quarterLength’: quarterLength}) on any call that changes the quarterLength, so that the client can make a change.

Returns False if there was no need to inform the client (like nothing has changed) or if .client is None. Otherwise returns True.

Duration.sliceComponentAtPosition(quarterPosition)

Given a quarter position within a component, divide that component into two components.

>>> d = duration.Duration()
>>> d.clear()  # need to remove default
>>> components = []
>>> d.addDurationTuple(duration.Duration('quarter'))
>>> d.addDurationTuple(duration.Duration('quarter'))
>>> d.addDurationTuple(duration.Duration('quarter'))
>>> d.quarterLength
3.0
>>> d.sliceComponentAtPosition(0.5)
>>> d.quarterLength
3.0
>>> len(d.components)
4
>>> d.components[0].type
'eighth'
>>> d.components[1].type
'eighth'
>>> d.components[2].type
'quarter'
Duration.splitDotGroups(*, inPlace=False)

splits a dotGroup-duration (of 1 component) into a new duration of two components. Returns a new duration

Probably does not handle properly tuplets of dot-groups. Never seen one, so probably okay.

>>> d1 = duration.Duration(type='half')
>>> d1.dotGroups = (1, 1)
>>> d1.quarterLength
4.5
>>> d2 = d1.splitDotGroups()
>>> d2.components
(DurationTuple(type='half', dots=1, quarterLength=3.0),
 DurationTuple(type='quarter', dots=1, quarterLength=1.5))
>>> d2.quarterLength
4.5

Here’s how a system that does not support dotGroups can still display the notes accurately. N.B. MusicXML does this automatically, so no need.

>>> n1 = note.Note()
>>> n1.duration = d1
>>> n1.duration = n1.duration.splitDotGroups()
>>> n1.duration.components
(DurationTuple(type='half', dots=1, quarterLength=3.0),
 DurationTuple(type='quarter', dots=1, quarterLength=1.5))
>>> s1 = stream.Stream()
>>> s1.append(meter.TimeSignature('9/8'))
>>> s1.append(n1)
>>> s1.show('lily.png')
.. image:: images/duration_splitDotGroups.*
>>> n2 = note.Note()
>>> n2.duration.type = 'quarter'
>>> n2.duration.dotGroups = (1, 1)
>>> n2.quarterLength
2.25
>>> n2.show()  # generates a dotted-quarter tied to dotted-eighth
>>> n2.duration.splitDotGroups(inPlace=True)
>>> n2.duration.dotGroups
(1,)
>>> n2.duration.components
(DurationTuple(type='quarter', dots=1, quarterLength=1.5),
 DurationTuple(type='eighth', dots=1, quarterLength=0.75))
>>> n2 = note.Note()
>>> n2.duration.type = 'quarter'
>>> n2.duration.dotGroups = (1, 1, 1)
>>> n2.quarterLength
3.375
>>> dSplit = n2.duration.splitDotGroups()
>>> dSplit.quarterLength
3.375
>>> dSplit.components
(DurationTuple(type='quarter', dots=1, quarterLength=1.5),
 DurationTuple(type='eighth', dots=1, quarterLength=0.75),
 DurationTuple(type='eighth', dots=1, quarterLength=0.75),
 DurationTuple(type='16th', dots=1, quarterLength=0.375))

Does NOT handle tuplets etc.

Methods inherited from ProtoM21Object:

Duration instance variables

Duration.client

A duration’s “client” is the object that holds this duration as a property. It is informed whenever the duration changes.

>>> n = note.Note('C#5', type='whole')
>>> d = n.duration
>>> d.client is n
True
Duration.expressionIsInferred

Boolean indicating whether this duration was created from a number rather than a type and thus can be changed to another expression. For instance the duration of 0.5 is generally an eighth note, but in the middle of a triplet group might be better written as a dotted-eighth triplet. If expressionIsInferred is True then music21 can change it according to complex. If False, then the type, dots, and tuplets are considered immutable.

>>> d = duration.Duration(0.5)
>>> d.expressionIsInferred
True
>>> d = duration.Duration('eighth')
>>> d.expressionIsInferred
False

Tuplet

class music21.duration.Tuplet(numberNotesActual: int = 3, numberNotesNormal: int = 2, durationActual: DurationTuple | Duration | str | tuple[str, int] | None = None, durationNormal: DurationTuple | Duration | str | tuple[str, int] | None = None, *, tupletId: int = 0, nestedLevel: int = 1, type: Literal['start', 'stop', 'startStop', False, None] = None, bracket: Literal[True, False, 'slur'] = True, placement: Literal['above', 'below'] = 'above', tupletActualShow: Literal['number', 'type', 'both', None] = 'number', tupletNormalShow: Literal['number', 'type', 'both', None] = None, frozen: bool = False, **keywords)

A tuplet object is a representation of a musical tuplet (like a triplet). It expresses a ratio that modifies duration values and are stored in Duration objects in a “tuple” (immutable list; since there can be nested tuplets) in the duration’s .tuplets property.

The primary representation uses two pairs of note numbers and durations.

The first pair of note numbers and durations describes the representation within the tuplet, or the value presented by the context. This is called “actual.” In a standard 8th note triplet this would be 3, eighth, meaning that a complete collection of this tuplet will be visually represented as three eighth notes. These attributes are numberNotesActual, durationActual.

The second pair of note numbers and durations describes the space that would have been occupied in a normal context. This is called “normal.” In a standard 8th note triplet this would be 2, eighth, meaning that a complete collection of notes under this tuplet will occupy the space of two eighth notes. These attributes are numberNotesNormal, durationNormal.

If duration values are not provided then durationActual and durationNormal are left as None – meaning that it is unspecified what the duration that completes the tuplet is. And this tuplet just represents a Ratio. PRIOR TO v4 durationActual and durationNormal were assumed to be eighths.

If only one duration, either durationActual or durationNormal, is provided, both are set to the same value.

Note that this is a duration modifier, or a generator of ratios to scale quarterLength values in Duration objects.

>>> myTup = duration.Tuplet(numberNotesActual=5, numberNotesNormal=4)
>>> print(myTup.tupletMultiplier())
4/5

We know that it is 5 in the place of 4, but 5 what in the place of 4 what?

>>> myTup.durationActual is None
True
>>> myTup
<music21.duration.Tuplet 5/4>

But we can change that:

>>> myTup.setDurationType('eighth')
>>> myTup.durationActual
DurationTuple(type='eighth', dots=0, quarterLength=0.5)
>>> myTup
<music21.duration.Tuplet 5/4/eighth>

In this case, the tupletMultiplier is a float because it can be expressed as a binary number:

>>> myTup2 = duration.Tuplet(8, 5)
>>> tm = myTup2.tupletMultiplier()
>>> tm
0.625

Here, six sixteenth notes occupy the space of four sixteenth notes.

>>> myTup2 = duration.Tuplet(6, 4, '16th')
>>> print(myTup2.durationActual.type)
16th
>>> print(myTup2.durationNormal.type)
16th
>>> print(myTup2.tupletMultiplier())
2/3

Tuplets may be frozen, in which case they become immutable. Tuplets which are attached to Durations are automatically frozen. Otherwise a tuplet could change without the attached duration knowing about it, which would be a real problem.

>>> myTup.frozen = True
>>> myTup.tupletActual = [3, 2]
Traceback (most recent call last):
music21.duration.TupletException: A frozen tuplet (or one attached to a duration)
    has immutable length.
>>> myHalf = duration.Duration('half')
>>> myHalf.appendTuplet(myTup2)
>>> myTup2.tupletActual = [5, 4]
Traceback (most recent call last):
music21.duration.TupletException: A frozen tuplet (or one attached to a duration)
    has immutable length.

Note that if you want to create a note with a simple Tuplet attached to it, you can just change the quarterLength of the note:

>>> myNote = note.Note('C#4')
>>> myNote.duration.quarterLength = 0.8
>>> myNote.duration.quarterLength
Fraction(4, 5)
>>> myNote.duration.fullName
'Quarter Quintuplet (4/5 QL)'
>>> myNote.duration.tuplets
(<music21.duration.Tuplet 5/4/quarter>,)

Tuplet bases

Tuplet read-only properties

Tuplet.fullName

Return the most complete representation of this tuplet in a readable form.

>>> tup = duration.Tuplet(numberNotesActual=5, numberNotesNormal=2)
>>> tup.fullName
'Quintuplet'
>>> tup = duration.Tuplet(numberNotesActual=3, numberNotesNormal=2)
>>> tup.fullName
'Triplet'
>>> tup = duration.Tuplet(numberNotesActual=17, numberNotesNormal=14)
>>> tup.fullName
'Tuplet of 17/14ths'

Read-only properties inherited from ProtoM21Object:

Tuplet read/write properties

Tuplet.durationActual

durationActual is a DurationTuple that represents the notes that are actually present and counted in a tuplet. For instance, in a 7 dotted-eighth in the place of 2 double-dotted quarter notes tuplet, the duration actual would be…

>>> d = duration.Tuplet(7, 2)
>>> print(d.durationActual)
None
>>> d.durationActual = duration.Duration('eighth', dots=1)

Notice that the Duration object gets converted to a DurationTuple

>>> d.durationActual
DurationTuple(type='eighth', dots=1, quarterLength=0.75)
>>> d.durationActual = 'quarter'
>>> d.durationActual
DurationTuple(type='quarter', dots=0, quarterLength=1.0)
Tuplet.durationNormal

durationNormal is a DurationTuple that represents the notes that would be present in the space normally (if there were no tuplets). For instance, in a 7 dotted-eighth in the place of 2 double-dotted quarter notes tuplet, the durationNormal would be…

>>> d = duration.Tuplet(7, 2)
>>> print(d.durationNormal)
None
>>> d.durationNormal = duration.Duration('quarter', dots=2)

Notice that the Duration object gets converted to a DurationTuple

>>> d.durationNormal
DurationTuple(type='quarter', dots=2, quarterLength=1.75)
>>> d.durationNormal = 'half'
>>> d.durationNormal
DurationTuple(type='half', dots=0, quarterLength=2.0)
Tuplet.tupletActual

Get or set a two element list of number notes actual and duration actual.

Tuplet.tupletNormal

Get or set a two element list of number notes actual and duration normal.

Tuplet methods

Tuplet.__eq__(other) bool

Two Tuplets are equal if their numbers are equal and durations are equal.

Visual details (type, bracket, placement, tupletActualShow, etc.) do not matter.

>>> triplet1 = duration.Tuplet(3, 2)
>>> triplet2 = duration.Tuplet(3, 2)
>>> triplet1 == triplet2
True
>>> quadruplet = duration.Tuplet(4, 3)
>>> triplet1 == quadruplet
False
>>> triplet3 = duration.Tuplet(3, 2, 'half')
>>> triplet1 == triplet3
False
Tuplet.augmentOrDiminish(amountToScale: int | float)

Given a number greater than zero, multiplies the current quarterLength of the duration by the number and resets the components for the duration (by default). Or if inPlace is set to False, returns a new duration that has the new length.

# TODO: add inPlace setting.

>>> a = duration.Tuplet()
>>> a.setRatio(6, 2)
>>> a.tupletMultiplier()
Fraction(1, 3)
>>> a.setDurationType('eighth')
>>> a.durationActual
DurationTuple(type='eighth', dots=0, quarterLength=0.5)
>>> c = a.augmentOrDiminish(0.5)
>>> c.durationActual
DurationTuple(type='16th', dots=0, quarterLength=0.25)
>>> c.tupletMultiplier()
Fraction(1, 3)

Raises ValueError if amountToScale is negative.

>>> a.augmentOrDiminish(-1)
Traceback (most recent call last):
ValueError: amountToScale must be greater than zero
Tuplet.setDurationType(durType: str | int | float | Fraction, dots=0)

Set both durationActual and durationNormal from either a string type or a quarterLength. optional dots can add dots to a string Type (or I suppose a quarterLength…but why?)

>>> a = duration.Tuplet()
>>> a.tupletMultiplier()
Fraction(2, 3)
>>> a.totalTupletLength()
1.0
>>> a.setDurationType('half')
>>> a.durationNormal
DurationTuple(type='half', dots=0, quarterLength=2.0)
>>> a.tupletMultiplier()
Fraction(2, 3)
>>> a.totalTupletLength()
4.0
>>> a.setDurationType('half', dots=1)
>>> a.durationNormal
DurationTuple(type='half', dots=1, quarterLength=3.0)
>>> a.totalTupletLength()
6.0
>>> a.setDurationType(2.0)
>>> a.totalTupletLength()
4.0
>>> a.setDurationType(4.0)
>>> a.totalTupletLength()
8.0
Tuplet.setRatio(actual, normal)

Set the ratio of actual divisions to represented in normal divisions. A triplet is 3 actual in the time of 2 normal.

>>> a = duration.Tuplet()
>>> a.tupletMultiplier()
Fraction(2, 3)
>>> a.setRatio(6, 2)
>>> a.numberNotesActual
6
>>> a.numberNotesNormal
2
>>> a.tupletMultiplier()
Fraction(1, 3)

One way of expressing 6/4-ish triplets without numbers:

>>> a = duration.Tuplet()
>>> a.setRatio(3, 1)
>>> a.durationActual = duration.durationTupleFromTypeDots('quarter', 0)
>>> a.durationNormal = duration.durationTupleFromTypeDots('half', 0)
>>> a.tupletMultiplier()
Fraction(2, 3)
>>> a.totalTupletLength()
2.0
Tuplet.totalTupletLength() float | Fraction

The total duration in quarter length of the tuplet as defined, assuming that enough notes existed to fill all entire tuplet as defined.

For instance, 3 quarters in the place of 2 quarters = 2.0 5 half notes in the place of a 2 dotted half notes = 6.0 (In the end it’s only the denominator that matters)

If durationActual or durationNormal are None, then they will be assumed to be eighth notes (for the basic 3:2 eighth-note triplet)

>>> a = duration.Tuplet()
>>> a.totalTupletLength()
1.0
>>> a.numberNotesActual = 3
>>> a.numberNotesNormal = 2
>>> a.setDurationType('half')
>>> a.totalTupletLength()
4.0

Let’s make it five halfs in the place of four:

>>> a.setRatio(5, 4)
>>> a.setDurationType('half')
>>> a.totalTupletLength()
8.0

Now five halfs in the place of two whole notes (same thing):

>>> a.setRatio(5, 2)
>>> a.totalTupletLength()
4.0
>>> a.durationNormal = duration.durationTupleFromTypeDots('whole', 0)
>>> a.totalTupletLength()
8.0
Tuplet.tupletMultiplier() float | Fraction

Get a Fraction() by which to scale the duration that this Tuplet is associated with.

>>> myTuplet = duration.Tuplet()
>>> myTuplet.tupletMultiplier()
Fraction(2, 3)
>>> myTuplet.tupletActual = [5, duration.Duration('eighth')]
>>> myTuplet.numberNotesActual
5
>>> myTuplet.durationActual.type
'eighth'
>>> print(myTuplet.tupletMultiplier())
2/5
>>> myTuplet.numberNotesNormal = 4
>>> print(myTuplet.tupletMultiplier())
4/5

Methods inherited from ProtoM21Object:

GraceDuration

class music21.duration.GraceDuration(typeOrDuration: str | int | float | Fraction | DurationTuple | None = None, **keywords)

A Duration that, no matter how it is created, always has a quarter length of zero.

GraceDuration can be created with an implied quarter length and type; these values are used to configure the duration, but then may not be relevant after instantiation.

>>> gd = duration.GraceDuration(type='half')
>>> gd.quarterLength
0.0
>>> gd.type
'half'
>>> gd = duration.GraceDuration(0.25)
>>> gd.type
'16th'
>>> gd.quarterLength
0.0
>>> gd.linked
False
>>> gd = duration.GraceDuration(1.25)
>>> gd.type
'complex'
>>> gd.quarterLength
0.0
>>> [(x.quarterLength, x.type) for x in gd.components]
[(0.0, 'quarter'), (0.0, '16th')]

GraceDuration bases

GraceDuration read-only properties

Read-only properties inherited from Duration:

Read-only properties inherited from ProtoM21Object:

GraceDuration read/write properties

GraceDuration.makeTime

True, False, or None (=unknown) whether the grace note should occupy time in performance. Default False. Currently not used in generated playback.

TODO: allow a duration object or number for duration.

GraceDuration.slash

True, False, or None (=unknown) whether the grace note should have a slash through it. Default True.

Read/write properties inherited from Duration:

GraceDuration methods

Methods inherited from Duration:

Methods inherited from ProtoM21Object:

GraceDuration instance variables

GraceDuration.stealTimeFollowing

Float number from 0.0 to 1.0 or None (default) for the proportion of the following duration to steal from the following note.

GraceDuration.stealTimePrevious

Float number from 0.0 to 1.0, or None (default) for the proportion of the previous duration to steal from the previous note.

Instance variables inherited from Duration:

TupletFixer

class music21.duration.TupletFixer(streamIn: stream.Stream | None = None)

The TupletFixer object takes in a flat stream and tries to fix the brackets and time modification values of the tuplet so that they reflect proper beaming, etc. It does not alter the quarterLength of any notes.

See findTupletGroups() and fixBrokenTupletDuration() for demonstrations.

TupletFixer methods

TupletFixer.findTupletGroups(incorporateGroupings: bool = False) list[list[note.GeneralNote]]

Finds all tuplets in the stream and puts them into groups.

If incorporateGroupings is True, then a tuplet.type=”stop” ends a tuplet group even if the next note is a tuplet.

This demonstration has three groups of tuplets, two sets of 8th note tuplets and one of 16ths:

>>> c = converter.parse(
...    'tinynotation: 4/4 trip{c8 d e} f4 trip{c#8 d# e#} g8 trip{c-16 d- e-}',
...    makeNotation=False)
>>> tf = duration.TupletFixer(c)  # no need to flatten this stream
>>> tupletGroups = tf.findTupletGroups()
>>> tupletGroups
[[<music21.note.Note C>, <music21.note.Note D>, <music21.note.Note E>],
 [<music21.note.Note C#>, <music21.note.Note D#>, <music21.note.Note E#>],
 [<music21.note.Note C->, <music21.note.Note D->, <music21.note.Note E->]]

These groups are stored in TupletFixer.allTupletGroups:

>>> tupletGroups is tf.allTupletGroups
True

Demonstration with incorporateGroupings:

>>> s = stream.Stream()
>>> for i in range(9):
...    n = note.Note()
...    n.pitch.ps = 60 + i
...    n.duration.quarterLength = 1/3
...    if i % 3 == 2:
...        n.duration.tuplets[0].type = 'stop'
...    s.append(n)
>>> tf = duration.TupletFixer(s)
>>> tupletGroups = tf.findTupletGroups(incorporateGroupings=True)
>>> tupletGroups
[[<music21.note.Note C>, <music21.note.Note C#>, <music21.note.Note D>],
 [<music21.note.Note E->, <music21.note.Note E>, <music21.note.Note F>],
 [<music21.note.Note F#>, <music21.note.Note G>, <music21.note.Note G#>]]

Without incorporateGroupings we just get one big set of tuplets

>>> tupletGroups = tf.findTupletGroups()
>>> len(tupletGroups)
1
>>> len(tupletGroups[0])
9
TupletFixer.fixBrokenTupletDuration(tupletGroup: list[note.GeneralNote]) None

Tries to fix cases like triplet quarter followed by triplet eighth to be a coherent tuplet.

Requires a tuplet group from findTupletGroups() or TupletFixer.allTupletGroups. Note: this works on a single tupletGroup while findTupletGroups() returns a list of groups.

>>> s = stream.Stream()
>>> n1 = note.Note('C')
>>> n1.duration.quarterLength = 2/3
>>> n1.duration.quarterLength
Fraction(2, 3)
>>> s.append(n1)
>>> n2 = note.Note('D')
>>> n2.duration.quarterLength = 1/3
>>> n2.duration.quarterLength
Fraction(1, 3)
>>> s.append(n2)

Here are the current tuplets for the two notes:

>>> n1.duration.tuplets[0]
<music21.duration.Tuplet 3/2/quarter>
>>> n2.duration.tuplets[0]
<music21.duration.Tuplet 3/2/eighth>

Notice how the first note is waiting for 3 triplet quarters to complete itself. But it could be 2/3 of a quarter note divided into eighth note triplets. TupletFixer will work on this.

It takes in a flattened stream, like this one:

>>> tf = duration.TupletFixer(s)

Find the tuplet groups. Returning a list of one group, which has two notes in it:

>>> tupletGroups = tf.findTupletGroups()
>>> tupletGroups
[[<music21.note.Note C>, <music21.note.Note D>]]

Now fix that single group:

>>> tg0 = tupletGroups[0]
>>> [n.duration.tuplets[0].type for n in tg0]
[None, None]
>>> tf.fixBrokenTupletDuration(tg0)

Now the first quarter-note triplet knows that its group will be complete after the next note:

>>> n1.duration.tuplets[0]
<music21.duration.Tuplet 3/2/eighth>
>>> n1.duration.quarterLength
Fraction(2, 3)
>>> n2.duration.tuplets[0]
<music21.duration.Tuplet 3/2/eighth>

Note that the tuplet type is not affected by this call:

>>> [n.duration.tuplets[0].type for n in tg0]
[None, None]

To do that, call makeTupletBrackets() on the flattened stream:

>>> stream.makeNotation.makeTupletBrackets(s, inPlace=True)
>>> [n.duration.tuplets[0].type for n in tg0]
['start', 'stop']

More complex example, from a piece by Josquin:

>>> humdrumExcerpt = '**kern *M3/1 3.c 6d 3e 3f 3d 3%2g 3e 3f#'
>>> humdrumLines = '\n'.join(humdrumExcerpt.split())

There is a side format of humdrum that the Josquin Research Project uses for long notes like the 3%2.

>>> humdrum.spineParser.flavors['JRP'] = True

Since Humdrum parsing is going to apply TupletFixer, we will temporarily disable it:

>>> saved_fixed_broken = duration.TupletFixer.fixBrokenTupletDuration
>>> duration.TupletFixer.fixBrokenTupletDuration = lambda x,y: None
>>> s = converter.parse(humdrumLines, format='humdrum')
>>> m1 = s.parts.first().measure(1)
>>> m1.show('text', addEndTimes=True)
{0.0 - 0.0} <music21.meter.TimeSignature 3/1>
{0.0 - 2.0} <music21.note.Note C>
{2.0 - 2.6667} <music21.note.Note D>
{2.6667 - 4.0} <music21.note.Note E>
{4.0 - 5.3333} <music21.note.Note F>
{5.3333 - 6.6667} <music21.note.Note D>
{6.6667 - 9.3333} <music21.note.Note G>
{9.3333 - 10.6667} <music21.note.Note E>
{10.6667 - 12.0} <music21.note.Note F#>
>>> duration.TupletFixer.fixBrokenTupletDuration = saved_fixed_broken
>>> tf = duration.TupletFixer(m1)
>>> tupletGroups = tf.findTupletGroups(incorporateGroupings=True)
>>> tupletGroups
[[<music21.note.Note C>, <music21.note.Note D>, <music21.note.Note E>],
 [<music21.note.Note F>, <music21.note.Note D>, <music21.note.Note G>,
  <music21.note.Note E>, <music21.note.Note F#>]]

There’s a problem with the last group: it contains 5 notes and the third note is twice as long as the others, so none of them form a coherent triplet.

>>> [n.duration.tuplets[0] for n in tupletGroups[1]]
[<music21.duration.Tuplet 3/2/half>, <music21.duration.Tuplet 3/2/half>,
 <music21.duration.Tuplet 3/2/whole>,
 <music21.duration.Tuplet 3/2/half>, <music21.duration.Tuplet 3/2/half>]
>>> [n.duration.tuplets[0].type for n in tupletGroups[1]]
['start', None, None, None, 'stop']

Fix the last broken tuplet group.

>>> tf.fixBrokenTupletDuration(tupletGroups[1])
>>> [n.duration.tuplets[0] for n in tupletGroups[1]]
[<music21.duration.Tuplet 3/2/whole>, <music21.duration.Tuplet 3/2/whole>,
 <music21.duration.Tuplet 3/2/whole>,
 <music21.duration.Tuplet 3/2/whole>, <music21.duration.Tuplet 3/2/whole>]

Note that the changes appear in the notes in the Stream as well.

>>> m1.last().duration.tuplets[0]
<music21.duration.Tuplet 3/2/whole>

Again, TupletFixer is automatically called when parsing from Humdrum. (MusicXML specifies its tuplet groups explicitly.) But you may need it when building up a stream from scratch in your own projects.

TupletFixer.setStream(streamIn: stream.Stream) None

Define a stream to work on and reset all temporary variables.

AppoggiaturaDuration

class music21.duration.AppoggiaturaDuration(typeOrDuration: str | int | float | Fraction | DurationTuple | None = None, **keywords)

Renamed in v6 to correct spelling.

AppoggiaturaDuration bases

AppoggiaturaDuration read-only properties

Read-only properties inherited from Duration:

Read-only properties inherited from ProtoM21Object:

AppoggiaturaDuration read/write properties

Read/write properties inherited from GraceDuration:

Read/write properties inherited from Duration:

AppoggiaturaDuration methods

Methods inherited from Duration:

Methods inherited from ProtoM21Object:

AppoggiaturaDuration instance variables

Instance variables inherited from GraceDuration:

Instance variables inherited from Duration:

DurationTuple

class music21.duration.DurationTuple(type, dots, quarterLength)

DurationTuple read-only properties

DurationTuple.ordinal

Converts type to an ordinal number where maxima = 1 and 1024th = 14; whole = 4 and quarter = 6. Based on duration.ordinalTypeFromNum

>>> a = duration.DurationTuple('whole', 0, 4.0)
>>> a.ordinal
4
>>> b = duration.DurationTuple('maxima', 0, 32.0)
>>> b.ordinal
1
>>> c = duration.DurationTuple('1024th', 0, 1/256)
>>> c.ordinal
14

DurationTuple methods

DurationTuple.augmentOrDiminish(amountToScale)

FrozenDuration

class music21.duration.FrozenDuration(*arguments, **keywords)

A FrozenDuration is one that must have all of its arguments specified at the time of construction. After that, it is immutable, like a Tuple and thus can be shared across different objects (like in MeterTerminals) or used as a hash.

>>> fd = duration.FrozenDuration(type='half', dots=2)
>>> fd.quarterLength
3.5
>>> fd.dots = 1
Traceback (most recent call last):
TypeError: This FrozenDuration instance is immutable.

FrozenDurations can be used as set/dict keys with stability.

>>> {fd}
{<music21.duration.FrozenDuration 3.5>}

Copying a FrozenDuration returns the original, so it is super fast.

>>> import copy
>>> copy.deepcopy(fd) is fd
True

FrozenDuration bases

FrozenDuration read-only properties

Read-only properties inherited from Duration:

Read-only properties inherited from ProtoM21Object:

FrozenDuration read/write properties

Read/write properties inherited from Duration:

FrozenDuration methods

Methods inherited from EqualSlottedObjectMixin:

Methods inherited from Duration:

Methods inherited from ProtoM21Object:

FrozenDuration instance variables

Instance variables inherited from Duration:

QuarterLengthConversion

class music21.duration.QuarterLengthConversion(components, tuplet)

Functions

music21.duration.convertQuarterLengthToType(qLen: int | float | Fraction) str

Return a type if there exists a type that is exactly equal to the duration of the provided quarterLength. Similar to quarterLengthToClosestType() but this function only returns exact matches.

>>> duration.convertQuarterLengthToType(2)
'half'
>>> duration.convertQuarterLengthToType(0.125)
'32nd'
>>> duration.convertQuarterLengthToType(0.33333)
Traceback (most recent call last):
music21.duration.DurationException: cannot convert quarterLength 0.33333 exactly to type
music21.duration.convertTypeToNumber(dType: str) float

Convert a duration type string (dType) to a numerical scalar representation that shows how many of that duration type fits within a whole note.

>>> duration.convertTypeToNumber('quarter')
4.0
>>> duration.convertTypeToNumber('half')
2.0
>>> duration.convertTypeToNumber('1024th')
1024.0
>>> duration.convertTypeToNumber('maxima')
0.125

These other types give these results:

>>> duration.convertTypeToNumber('zero')
0.0
>>> duration.convertTypeToNumber('complex')
Traceback (most recent call last):
music21.duration.DurationException: Could not determine durationNumber from complex
music21.duration.convertTypeToQuarterLength(dType: str, dots: int = 0, tuplets: list[music21.duration.Tuplet] | None = None, dotGroups=None) float | Fraction

Given a rhythm type (dType), number of dots (dots), an optional list of Tuplet objects (tuplets), and a (very) optional list of Medieval dot groups (dotGroups), return the equivalent quarter length.

>>> duration.convertTypeToQuarterLength('whole')
4.0
>>> duration.convertTypeToQuarterLength('16th')
0.25
>>> duration.convertTypeToQuarterLength('quarter', 2)
1.75
>>> tup = duration.Tuplet(numberNotesActual=5, numberNotesNormal=4)
>>> duration.convertTypeToQuarterLength('quarter', 0, [tup])
Fraction(4, 5)
>>> duration.convertTypeToQuarterLength('quarter', 1, [tup])
Fraction(6, 5)
>>> tup = duration.Tuplet(numberNotesActual=3, numberNotesNormal=4)
>>> duration.convertTypeToQuarterLength('quarter', 0, [tup])
Fraction(4, 3)

Also can handle those rare medieval dot groups (such as dotted-dotted half notes that take a full measure of 9/8. Conceptually, these are dotted-(dotted-half) notes. See trecento.trecentoCadence for more information ). >>> duration.convertTypeToQuarterLength(‘half’, dots=1, dotGroups=[1, 1]) 4.5

Unknown values raise DurationException:

>>> duration.convertTypeToQuarterLength('minim')
Traceback (most recent call last):
music21.duration.DurationException: no such type (minim) available for conversion
music21.duration.dottedMatch(qLen: int | float | Fraction, maxDots=4) tuple[int, str] | tuple[Literal[False], Literal[False]]

Given a quarterLength, determine if there is a dotted (or non-dotted) type that exactly matches. Returns a pair of (numDots, type) or (False, False) if no exact matches are found.

Returns a maximum of four dots by default.

>>> duration.dottedMatch(3.0)
(1, 'half')
>>> duration.dottedMatch(1.75)
(2, 'quarter')

This value is not equal to any dotted note length

>>> duration.dottedMatch(1.6)
(False, False)

maxDots can be lowered for certain searches

>>> duration.dottedMatch(1.875)
(3, 'quarter')
>>> duration.dottedMatch(1.875, 2)
(False, False)
>>> duration.dottedMatch(0.00001, 2)
(False, False)
music21.duration.durationTupleFromQuarterLength(ql=1.0) DurationTuple

Returns a DurationTuple for a given quarter length if the ql can be expressed as a type and number of dots (no tuplets, no complex duration, etc.). If it can’t be expressed, returns an “inexpressible” DurationTuple.

>>> dt = duration.durationTupleFromQuarterLength(3.0)
>>> dt
DurationTuple(type='half', dots=1, quarterLength=3.0)

If it’s not possible, we return an “inexpressible” type:

>>> dt = duration.durationTupleFromQuarterLength(2.5)
>>> dt
DurationTuple(type='inexpressible', dots=0, quarterLength=2.5)
music21.duration.durationTupleFromTypeDots(durType='quarter', dots=0)

Returns a DurationTuple (which knows its quarterLength) for a given type and dots (no tuplets)

>>> dt = duration.durationTupleFromTypeDots('quarter', 0)
>>> dt
DurationTuple(type='quarter', dots=0, quarterLength=1.0)
>>> dt2 = duration.durationTupleFromTypeDots('quarter', 0)
>>> dt is dt2
True

Also with keyword arguments.

>>> dt = duration.durationTupleFromTypeDots(durType='zero', dots=0)
>>> dt
DurationTuple(type='zero', dots=0, quarterLength=0.0)

Unknown values raise DurationException:

>>> dt = duration.durationTupleFromTypeDots(durType='minim', dots=0)
Traceback (most recent call last):
music21.duration.DurationException: Unknown type: minim
music21.duration.nextLargerType(durType: str) str

Given a type (such as 16th or quarter), return the next larger type.

>>> duration.nextLargerType('16th')
'eighth'
>>> duration.nextLargerType('whole')
'breve'
>>> duration.nextLargerType('duplex-maxima')
Traceback (most recent call last):
music21.duration.DurationException: cannot get the next larger of duplex-maxima
music21.duration.nextSmallerType(durType: str) str

Given a type (such as 16th or quarter), return the next smaller type.

>>> duration.nextSmallerType('16th')
'32nd'
>>> duration.nextSmallerType('whole')
'half'
>>> duration.nextSmallerType('1024th')
'2048th'
>>> duration.nextSmallerType('2048th')
Traceback (most recent call last):
music21.duration.DurationException: cannot get the next smaller of 2048th
music21.duration.quarterLengthToClosestType(qLen: int | float | Fraction) tuple[str, bool]

Returns a two-unit tuple consisting of

1. The type string (“quarter”) that is smaller than or equal to the quarterLength of provided.

  1. Boolean, True or False, whether the conversion was exact.

>>> duration.quarterLengthToClosestType(0.5)
('eighth', True)
>>> duration.quarterLengthToClosestType(0.75)
('eighth', False)
>>> duration.quarterLengthToClosestType(1.8)
('quarter', False)

Some extremely close types will return True for exact conversion…

>>> duration.quarterLengthToClosestType(2.0000000000000001)
('half', True)

Very big durations… are fine:

>>> duration.quarterLengthToClosestType(129.99)
('duplex-maxima', False)

Durations smaller than 2048th note raise a DurationException

>>> qL = duration.typeToDuration['2048th']
>>> qL
0.001953125
>>> qL = qL * 0.75
>>> duration.quarterLengthToClosestType(qL)
Traceback (most recent call last):
music21.duration.DurationException: Cannot return types smaller than 2048th;
    qLen was: 0.00146484375
music21.duration.quarterLengthToNonPowerOf2Tuplet(qLen: int | float | Fraction) tuple[music21.duration.Tuplet, music21.duration.DurationTuple]

Slow, last chance function that returns a tuple of a single tuplet, probably with a non power of 2 denominator (such as 7:6) that represents the quarterLength and the DurationTuple that should be used to express the note.

This could be a double-dotted note, but also a tuplet…

>>> duration.quarterLengthToNonPowerOf2Tuplet(7)
(<music21.duration.Tuplet 8/7/quarter>, DurationTuple(type='breve', dots=0, quarterLength=8.0))
>>> duration.quarterLengthToNonPowerOf2Tuplet(7/16)
(<music21.duration.Tuplet 8/7/64th>, DurationTuple(type='eighth', dots=0, quarterLength=0.5))
>>> duration.quarterLengthToNonPowerOf2Tuplet(7/3)
(<music21.duration.Tuplet 12/7/16th>, DurationTuple(type='whole', dots=0, quarterLength=4.0))

And of course…

>>> duration.quarterLengthToNonPowerOf2Tuplet(1)
(<music21.duration.Tuplet 1/1/quarter>,
 DurationTuple(type='quarter', dots=0, quarterLength=1.0))
music21.duration.quarterLengthToTuplet(qLen: int | float | Fraction, maxToReturn: int = 4, tupletNumerators=(3, 5, 7, 11, 13)) list[music21.duration.Tuplet]

Returns a list of possible Tuplet objects for a given qLen (quarterLength). As there may be more than one possible solution, the maxToReturn integer specifies the maximum number of values returned.

Searches for numerators specified in duration.defaultTupletNumerators (3, 5, 7, 11, 13). Does not return dotted tuplets, nor nested tuplets.

Note that 4:3 tuplets won’t be found, but will be found as dotted notes by dottedMatch.

>>> duration.quarterLengthToTuplet(0.33333333)
[<music21.duration.Tuplet 3/2/eighth>, <music21.duration.Tuplet 3/1/quarter>]
>>> duration.quarterLengthToTuplet(0.20)
[<music21.duration.Tuplet 5/4/16th>,
 <music21.duration.Tuplet 5/2/eighth>,
 <music21.duration.Tuplet 5/1/quarter>]

By specifying only 1 maxToReturn, a single-length list containing the Tuplet with the smallest type will be returned.

>>> duration.quarterLengthToTuplet(0.3333333, 1)
[<music21.duration.Tuplet 3/2/eighth>]
>>> tup = duration.quarterLengthToTuplet(0.3333333, 1)[0]
>>> tup.tupletMultiplier()
Fraction(2, 3)
music21.duration.unitSpec(durationObjectOrObjects)

DEPRECATED and to be removed in v10.

A simple data representation of most Duration objects. Processes a single Duration or a List of Durations, returning a single or list of unitSpecs.

A unitSpec is a tuple of qLen, durType, dots, tupleNumerator, tupletDenominator, and tupletType (assuming top and bottom tuplets are the same).

This function does not deal with nested tuplets, etc.

>>> aDur = duration.Duration()
>>> aDur.quarterLength = 3
>>> duration.unitSpec(aDur)
(3.0, 'half', 1, None, None, None)
>>> bDur = duration.Duration()
>>> bDur.quarterLength = 1.125
>>> duration.unitSpec(bDur)
(1.125, 'complex', 0, None, None, None)
>>> cDur = duration.Duration()
>>> cDur.quarterLength = 0.3333333
>>> duration.unitSpec(cDur)
(Fraction(1, 3), 'eighth', 0, 3, 2, 'eighth')
>>> duration.unitSpec([aDur, bDur, cDur])
[(3.0, 'half', 1, None, None, None),
 (1.125, 'complex', 0, None, None, None),
 (Fraction(1, 3), 'eighth', 0, 3, 2, 'eighth')]