+
+class Note(object) :
+ def __init__(self, node, divisions) :
+ self.step = _getNodeValue(node, 'pitch/step')
+ self.octave = int(_getNodeValue(node, 'pitch/octave'))
+ self.alter = int(_getNodeValue(node, 'pitch/alter', 0))
+ self._duration = float(_getNodeValue(node, 'duration'))
+ self.lyric = _getNodeValue(node, 'lyric/text', '')
+
+ self.divisions = divisions
+
+ @property
+ def midi(self) :
+ mid = DIATO_SCALE[self.step]
+ mid = mid + (self.octave - OCTAVE_REF) * 12
+ mid = mid + self.alter
+ return mid
+
+ @property
+ def duration(self) :
+ return self._duration / self.divisions
+
+ @property
+ def name(self) :
+ name = '%s%d' % (self.step, self.octave)
+ if self.alter < 0 :
+ alterext = 'b'
+ else :
+ alterext = '#'
+ name = '%s%s' % (name, abs(self.alter) * alterext)
+ return name
+
+
+
+
+def _getNodeValue(node, path, default=_marker) :
+ try :
+ for name in path.split('/') :
+ node = node.getElementsByTagName(name)[0]
+ return node.firstChild.nodeValue
+ except :
+ if default is _marker :
+ raise
+ else :
+ return default
+
+def musicXml2Song(input, output, partIndex=0, printNotes=False) :
+ if isinstance(input, StringTypes) :
+ input = open(input, 'r')
+
+ d = parse(input)
+ doc = d.documentElement
+
+ # TODO conversion préalable score-timewise -> score-partwise
+ assert doc.nodeName == u'score-partwise'
+
+ parts = doc.getElementsByTagName('part')
+ leadPart = parts[partIndex]
+
+ # divisions de la noire
+ divisions = 0
+ midiNotes, durations, lyrics = [], [], []
+
+ for measureNode in leadPart.getElementsByTagName('measure') :
+ divisions = int(_getNodeValue(measureNode, 'attributes/divisions', divisions))
+ for noteNode in measureNode.getElementsByTagName('note') :
+ note = Note(noteNode, divisions)
+ if printNotes :
+ print note.name, note.midi, note.duration, note.lyric
+ midiNotes.append(note.midi)
+ durations.append(note.duration)
+ lyrics.append(note.lyric)
+
+ song = Song(None,
+ midiNoteNumbers = midiNotes,
+ noteLengths = durations,
+ lyrics = lyrics,
+ notesInExtendedScale=None)
+ song.save(output)
+
+