From 85dcabe4930537df227c04d2dc4336a32e5e4f30 Mon Sep 17 00:00:00 2001 From: Greg Chapman <75333244+gregchapman-dev@users.noreply.github.com> Date: Mon, 6 May 2024 13:21:13 -0700 Subject: [PATCH 1/2] MusicXML import: forward tags with voice should trigger the same behavior as note tags with voice. I see this quite a bit in leadsheets where these are used to position chord changes within a melody note. --- music21/musicxml/xmlToM21.py | 27 ++++++++++++++++++++------- 1 file changed, 20 insertions(+), 7 deletions(-) diff --git a/music21/musicxml/xmlToM21.py b/music21/musicxml/xmlToM21.py index 9c1ea7017..711f541da 100644 --- a/music21/musicxml/xmlToM21.py +++ b/music21/musicxml/xmlToM21.py @@ -6202,7 +6202,7 @@ def parseMeasureNumbers(self, mNumRaw=None): def updateVoiceInformation(self): # noinspection PyShadowingNames ''' - Finds all the "voice" information in tags and updates the set of + Finds all the "voice" information in and tags and updates the set of `.voiceIndices` to be a set of all the voice texts, and if there is more than one voice in the measure, sets `.useVoices` to True and creates a voice for each. @@ -6231,14 +6231,27 @@ def updateVoiceInformation(self): 2 >>> list(MP.stream.getElementsByClass(stream.Voice)) [, ] + >>> MP = musicxml.xmlToM21.MeasureParser() + >>> MP.mxMeasure = ET.fromstring('1' + ... + '2') + >>> MP.updateVoiceInformation() + >>> sorted(list(MP.voiceIndices)) + ['1', '2'] + >>> MP.useVoices + True + >>> len(MP.stream) + 2 + >>> list(MP.stream.getElementsByClass(stream.Voice)) + [, ] ''' mxm = self.mxMeasure - for mxn in mxm.findall('note'): - voice = mxn.find('voice') - if vIndex := strippedText(voice): - self.voiceIndices.add(vIndex) - # it is a set, so no need to check if already there - # additional time < 1 sec per ten million ops. + for tagSearch in ('note', 'forward'): + for mxn in mxm.findall(tagSearch): + voice = mxn.find('voice') + if vIndex := strippedText(voice): + self.voiceIndices.add(vIndex) + # it is a set, so no need to check if already there + # additional time < 1 sec per ten million ops. if len(self.voiceIndices) > 1: for vIndex in sorted(self.voiceIndices): From 57ed01f4a7fd7378ae92960d8d55b7a3191a216e Mon Sep 17 00:00:00 2001 From: Greg Chapman <75333244+gregchapman-dev@users.noreply.github.com> Date: Wed, 8 May 2024 15:59:17 -0700 Subject: [PATCH 2/2] OffsetHierarchyFilter was a little inaccurate when tuplets were involved. Added missing opFrac(). --- music21/stream/filters.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/music21/stream/filters.py b/music21/stream/filters.py index 105855818..ba9d19eff 100644 --- a/music21/stream/filters.py +++ b/music21/stream/filters.py @@ -490,7 +490,7 @@ def __call__(self, e, iterator=None): if not hasattr(iterator, 'iteratorStartOffsetInHierarchy'): raise FilterException('Can only run OffsetHierarchyFilter on a RecursiveIterator') - offset = s.elementOffset(e) + iterator.iteratorStartOffsetInHierarchy + offset = opFrac(s.elementOffset(e) + iterator.iteratorStartOffsetInHierarchy) return self.isElementOffsetInRange(e, offset, stopAfterEnd=False)