music21.tree.node

Internal data structures for timespan collections.

This is an implementation detail of the TimespanTree class. Most music21 users can happily ignore this module.

ElementNode

class music21.tree.node.ElementNode(position, payload=None)

A node containing a single element, which is aware of the element’s endTime and index within a stream, as well as the endTimes and indices of the elements to the left and right of it.

Here’s an element node that is at first is no different from an AVL node, except in representation:

>>> n = note.Note('C4')
>>> n.duration.quarterLength = 2.0
>>> n.offset = 4.0
>>> elNode = tree.node.ElementNode(n.sortTuple(), n)
>>> elNode
<ElementNode: Start:4.0 <0.20...> Indices:(l:-1 *-1* r:-1) Payload:<music21.note.Note C>>
>>> avlNode = tree.core.AVLNode(4.0, n)
>>> avlNode
<AVLNode: Start:4.0 Height:0 L:None R:None>

But now let’s give n a stream context where it is element 1 in the measure.

>>> m = stream.Measure()
>>> r = note.Rest(type='whole')
>>> m.insert(0.0, r)
>>> m.insert(4.0, n)
>>> n2 = note.Note('F#4')
>>> n2.duration.quarterLength = 3.0
>>> m.insert(6.0, n2)

Let’s create an ElementNode object for each object and attach them to elNode:

>>> rElNode = tree.node.ElementNode(r.sortTuple(), r)
>>> n2ElNode = tree.node.ElementNode(n2.sortTuple(), n2)
>>> elNode.leftChild = rElNode
>>> elNode.update()
>>> elNode.rightChild = n2ElNode
>>> elNode.update()

Everything so far could be done w/ AVLNodes, which are not music21 specific. But now we’ll add some specific features, based on the information from the Stream.

>>> m[1] is n
True
>>> m.highestTime
9.0
>>> m.offset
0.0
>>> rElNode.payloadElementIndex = 0
>>> elNode.payloadElementIndex = 1
>>> n2ElNode.payloadElementIndex = 2
>>> elNode.updateIndices()
>>> elNode.updateEndTimes()

Now let’s look at the ElementNode:

>>> elNode
<ElementNode: Start:4.0 <0.20...> Indices:(l:0 *1* r:3) Payload:<music21.note.Note C>>

elNode is at 4.0. With it and underneath it are nodes from 0 <= nodes < 3. This seemingly screwy way of counting is exactly slice notation: [0:3].

So if we wanted to get a certain index number we would know that index 0 is in the leftChild’s payload, index 2 is in the rightChild’s payload, and index 1 is this node’s payload. This information is easy to compute for ElementNodes, since each node holds exactly one element (since the position here is a sortTuple which is unique); for its subclass, OffsetNode, where multiple elements can be in a node, this will become much more important and harder to work with.

Here is how you can actually get this information.

>>> elNode.subtreeElementsStartIndex
0
>>> elNode.subtreeElementsStopIndex
3

If they’ve never been set then they return -1, but the .updateIndices() call above set these number for the children also. So they return the node’s own index and one above:

>>> n2ElNode.subtreeElementsStartIndex
2
>>> n2ElNode.subtreeElementsStopIndex
3

Now here is the thing that actually makes trees based on ElementNodes useful. Each node knows the lowest and highest endTime as well as offset among the nodes below it:

>>> elNode.lowestPosition.offset
0.0
>>> elNode.endTimeLow
4.0
>>> elNode.highestPosition.offset
6.0
>>> elNode.endTimeHigh
9.0

What’s so great about this? If you’re doing a “getElementsByOffset” search, you can know from this information whether there’s going to be an element whose start or end time will possibly match the offset span without needing to descend into the tree.

The last element in a Stream often doesn’t has the highest endTime (think of different voices, flat streams, etc.) so searches for offsets are often O(n) when they could be O(log n) if the information were cached into a tree as this does.

ElementNode bases

ElementNode read-only properties

ElementNode.highestPosition

Returns the highest position in the tree rooted on this node.

ElementNode.lowestPosition

Returns the lowest position in the tree rooted on this node.

ElementNode methods

ElementNode.updateEndTimes()

Traverses the tree structure and updates cached maximum and minimum endTime values for the subtrees rooted at each node.

Used internally by ElementTree.

Returns None.

ElementNode.updateIndices(parentStopIndex=None)

Updates the payloadElementIndex, and the subtreeElementsStartIndex and subtreeElementsStopIndex (and does so for all child nodes) by traversing the tree structure.

Updates cached indices which keep track of the index of the element stored at each node, and of the minimum and maximum indices of the subtrees rooted at each node.

Called on rootNode of a tree that uses ElementNodes, such as ElementTree

parentStopIndex specifies the stop index of the parent node, if known. It is used internally by the function.

Returns None.

Methods inherited from AVLNode:

ElementNode instance variables

ElementNode.endTimeHigh

The highest endTime of any node in the subtree rooted on this node.

ElementNode.endTimeLow

The lowest endTime of any node in the subtree rooted on this node.

ElementNode.payloadElementIndex

The index in a stream of the element stored in the payload of this node.

ElementNode.subtreeElementsStartIndex

The lowest element index of an element in the payload of any node of the subtree rooted on this node.

ElementNode.subtreeElementsStopIndex

The highest element index of an element in the payload of any node of the subtree rooted on this node.

Instance variables inherited from AVLNode:

OffsetNode

class music21.tree.node.OffsetNode(offset, payload=None)

A node representing zero, one, or many elements at an offset. It has all the power of an ElementNode but substantially more.

Here’s an example of what it means and does:

>>> score = tree.makeExampleScore()
>>> sf = score.flatten()
>>> sf.show('text', addEndTimes=True)
{0.0 - 0.0} <music21.instrument.Instrument 'PartA: : '>
{0.0 - 0.0} <music21.instrument.Instrument 'PartB: : '>
{0.0 - 0.0} <music21.clef.BassClef>
{0.0 - 0.0} <music21.clef.BassClef>
{0.0 - 0.0} <music21.meter.TimeSignature 2/4>
{0.0 - 0.0} <music21.meter.TimeSignature 2/4>
{0.0 - 1.0} <music21.note.Note C>
{0.0 - 2.0} <music21.note.Note C#>
{1.0 - 2.0} <music21.note.Note D>
{2.0 - 3.0} <music21.note.Note E>
{2.0 - 4.0} <music21.note.Note G#>
{3.0 - 4.0} <music21.note.Note F>
{4.0 - 5.0} <music21.note.Note G>
{4.0 - 6.0} <music21.note.Note E#>
{5.0 - 6.0} <music21.note.Note A>
{6.0 - 7.0} <music21.note.Note B>
{6.0 - 8.0} <music21.note.Note D#>
{7.0 - 8.0} <music21.note.Note C>
{8.0 - 8.0} <music21.bar.Barline type=final>
{8.0 - 8.0} <music21.bar.Barline type=final>
>>> scoreTree = tree.fromStream.asTimespans(sf, flatten=False, classList=None)
>>> rn = scoreTree.rootNode

The RootNode here represents the starting position of the Note F at 3.0; It is the center of the elements in the flat Stream. Its index is 5 (that is, it’s the sixth note in the element list) and its offset is 3.0

>>> rn
<OffsetNode 3.0 Indices:0,11,12,20 Length:1>
>>> sf[11]
<music21.note.Note F>
>>> sf[11].offset
3.0

Thus, the indices of 0:5:6:12 indicate that the left-side of the node handles indices from >= 0 to < 5; and the right-side of the node handles indices >= 6 and < 12, and this node handles indices >= 5 and < 6.

The Length: {1} indicates that there is exactly one element at this location, that is, the F.

The “payload” of the node, is just that element wrapped in a list wrapped in an ElementTimespan or PitchedTimespan:

>>> rn.payload
[<PitchedTimespan (3.0 to 4.0) <music21.note.Note F>>]
>>> rn.payload[0].element
<music21.note.Note F>
>>> rn.payload[0].element is sf[11]
True

We can look at the leftChild of the root node to get some more interesting cases:

>>> left = rn.leftChild
>>> left
<OffsetNode 1.0 Indices:0,8,9,11 Length:1>

In the leftNode of the leftNode of the rootNode there are eight elements: metadata and both notes that begin on offset 0.0:

>>> leftLeft = left.leftChild
>>> leftLeft
<OffsetNode 0.0 Indices:0,0,8,8 Length:8>
>>> leftLeft.payload
[<ElementTimespan (0.0 to 0.0) <music21.instrument.Instrument 'PartA: : '>>,
 <ElementTimespan (0.0 to 0.0) <music21.instrument.Instrument 'PartB: : '>>,
 <ElementTimespan (0.0 to 0.0) <music21.clef.BassClef>>,
 <ElementTimespan (0.0 to 0.0) <music21.clef.BassClef>>,
 <ElementTimespan (0.0 to 0.0) <music21.meter.TimeSignature 2/4>>,
 <ElementTimespan (0.0 to 0.0) <music21.meter.TimeSignature 2/4>>,
 <PitchedTimespan (0.0 to 1.0) <music21.note.Note C>>,
 <PitchedTimespan (0.0 to 2.0) <music21.note.Note C#>>]

The Indices:0,0,8,8 indicates that leftLeft has neither left nor right children

>>> leftLeft.leftChild is None
True
>>> leftLeft.rightChild is None
True

What makes an OffsetNode more interesting than other AWL Nodes is that it is aware of the fact that it might have objects that end at different times, such as the zero-length metadata and the 2.0 length half note

>>> leftLeft.endTimeLow
0.0
>>> leftLeft.endTimeHigh
2.0

OffsetNode bases

OffsetNode read-only properties

Read-only properties inherited from ElementNode:

OffsetNode methods

OffsetNode.payloadEndTimes()

returns a (potentially unsorted) list of all the end times for all TimeSpans or Elements in the payload. Does not trust el.endTime because it might refer to a different offset. Rather, it takes the position and adds it to the duration.quarterLength.

>>> offsetNode = tree.node.OffsetNode(40)
>>> n = note.Note()
>>> offsetNode.payload.append(n)
>>> ts = tree.spans.Timespan(40, 44)
>>> offsetNode.payload.append(ts)
>>> offsetNode.payloadEndTimes()
[41.0, 44.0]
OffsetNode.updateEndTimes()

Traverses the tree structure and updates cached maximum and minimum endTime values for the subtrees rooted at each node.

Used internally by OffsetTree.

Returns None

OffsetNode.updateIndices(parentStopIndex=None)

Updates the payloadElementsStartIndex, the payloadElementsStopIndex and the subtreeElementsStartIndex and subtreeElementsStopIndex (and does so for all child nodes) by traversing the tree structure.

Updates cached indices which keep track of the index of the element stored at each node, and of the minimum and maximum indices of the subtrees rooted at each node.

Called on rootNode of a tree that uses OffsetNodes, such as OffsetTree or TimespanTree

Returns None.

Methods inherited from AVLNode:

OffsetNode instance variables

OffsetNode.payload

The contents of the node at this point. Usually a list of ElementTimespans or PitchedTimespans.

>>> score = tree.makeExampleScore()
>>> scoreTree = tree.fromStream.asTimespans(score, flatten=True,
...                  classList=(note.Note, chord.Chord))
>>> print(scoreTree.rootNode.debug())
<OffsetNode 3.0 Indices:0,5,6,12 Length:1>
L: <OffsetNode 1.0 Indices:0,2,3,5 Length:1>
L: <OffsetNode 0.0 Indices:0,0,2,2 Length:2>
R: <OffsetNode 2.0 Indices:3,3,5,5 Length:2>
R: <OffsetNode 5.0 Indices:6,8,9,12 Length:1>
L: <OffsetNode 4.0 Indices:6,6,8,8 Length:2>
R: <OffsetNode 6.0 Indices:9,9,11,12 Length:2>
R: <OffsetNode 7.0 Indices:11,11,12,12 Length:1>
>>> scoreTree.rootNode.payload
[<PitchedTimespan (3.0 to 4.0) <music21.note.Note F>>]
>>> scoreTree.rootNode.leftChild.payload
[<PitchedTimespan (1.0 to 2.0) <music21.note.Note D>>]
>>> for x in scoreTree.rootNode.leftChild.rightChild.payload:
...     x
...
<PitchedTimespan (2.0 to 3.0) <music21.note.Note E>>
<PitchedTimespan (2.0 to 4.0) <music21.note.Note G#>>
>>> scoreTree.rootNode.rightChild.payload
[<PitchedTimespan (5.0 to 6.0) <music21.note.Note A>>]
OffsetNode.payloadElementsStartIndex

The timespan start index (i.e., the first x where s[x] is found in this Node’s payload) of only those timespans stored in the payload of this node.

OffsetNode.payloadElementsStopIndex

The timespan stop index (i.e., the last x where s[x] is found in this Node’s payload) of only those timespans stored in the payload of this node.

Instance variables inherited from ElementNode:

Instance variables inherited from AVLNode: