@@ -1,28 +1,30 @@ #!/usr/bin/env python + +import tree from ply import lex, yacc class Note(): - def __init__(self, value, duration=.25, octave=8): - self.value = value + def __init__(self, value, duration=1, octave=8): self.duration = duration + if value > 12: + self.duration += 1 + value = value % 12 + self.value = value self.octave = octave - self.accidental = None def __repr__(self): return "Note %s %s %s" % (self.value, self.duration, self.octave) class Chord(): - def __init__(self, value, duration=.5, chord_type="major", octave=5): - self.value = value + def __init__(self, notes, duration=1): + self.notes = notes self.duration = duration - self.chord_type = chord_type - self.octave = octave def __repr__(self): - return "Chord %s %s %s" % (self.value, self.duration, self.chord_type, self.octave) + return "Chord %s" % (self.notes) class Rest(): - def __init__(self, duration=.25): + def __init__(self, duration=1): self.duration = duration def __repr__(self): return "Rest node %s" % self.duration @@ -33,24 +35,26 @@ "BASENOTE", "ACCIDENTAL", "REST", "OCTAVE", "CHORD_TYPE", - "PAREN", + "BRACKET", "SYNCOPATE", + "NODE", ) t_ignore = " |" - #t_BASENOTE = r"[A-Ga-g]" - t_BASENOTE = r"I+V?|VI*|i+v?|vi*" - t_ACCIDENTAL = r"\^{1,2}|_{1,2}|=" + t_BASENOTE = r"[A-Ga-g]" +# t_BASENOTE = r"I+V?|VI*|i+v?|vi*" + t_ACCIDENTAL = r"\^{1}|_{1}|=" t_REST = r"z" t_OCTAVE = r"'+|,+" t_CHORD_TYPE = r"m|7|m7|0|o|\+|mb5|sus|sus4|maj7|mmaj7|7sus4|dim|dim7|7b5|m7b5|6|b6|m6|mb6|46|maj9|9|add9|7b9|m9" - t_PAREN = "\(|\)" + t_BRACKET = r"\[|\]" t_SYNCOPATE = "\+|-" + t_NODE = r"\([a-zA-Z0-9_-]+\)" def t_NOTE_LENGTH(t): r"/?\d+" multiplier = float(t.value.strip("/")) if t.value.startswith("/"): @@ -69,17 +73,19 @@ def p_note_list(p): '''score : score note | score chord | score rest + | score node ''' p[0] = p[1] + [p[2]] def p_score(p): '''score : note | chord | rest + | node ''' p[0] = [p[1]] def p_chord_length(p): @@ -97,18 +103,20 @@ new_note.duration = p[2] p[0] = new_note def p_chord(p): - '''chord : PAREN note PAREN - | PAREN note CHORD_TYPE PAREN + '''chord : BRACKET note BRACKET + | BRACKET note CHORD_TYPE BRACKET ''' - pitch = p[2].value - pitch = pitch.upper() - p[0] = Chord(value=pitch, octave=default_octave) - if len(p) > 3: - p[0].chord_type = p[3] + root_note = p[2].value + chorded_notes = [] + + for offset in [0, 4, 7]: + chorded_notes.append(Note(root_note+offset, octave=p[2].octave)) + + p[0] = Chord(notes=chorded_notes) def p_note_syncopate(p): ''' note : note SYNCOPATE ''' @@ -116,11 +124,14 @@ def p_accidental(p): '''note : ACCIDENTAL note ''' - p[2].accidental = p[1] + if p[1] == "^": + p[2].value += 1 + else: + p[2].value -= 1 p[0] = p[2] def p_octave(p): '''note : note OCTAVE ''' @@ -130,21 +141,40 @@ p[0] = p[1] def p_note(p): '''note : BASENOTE ''' - p[0] = Note(p[1], octave=default_octave) + notes = { + "C": 0, + "D": 2, + "E": 4, + "F": 5, + "G": 7, + "A": 9, + "B": 11 + } + n = notes[p[1]] + p[0] = Note(n, octave=default_octave) def p_rest(p): ''' rest : REST | REST NOTE_LENGTH ''' p[0] = Rest() if len(p) > 2: p[0].duration = p[2] + + def p_node(p): + '''node : NODE + ''' + p[0] = tree.Tree(p[1].strip("(").strip(")")) + def p_error(p): +# import ipdb +# ipdb.set_trace() + print p raise Exception("Syntax error at '%s' of element type %s" % (p.value, p.type)) yacc.yacc() return yacc.parse(score)