mirror of
https://github.com/microtherion/VocalEasel.git
synced 2025-01-03 17:04:00 +00:00
2151 lines
52 KiB
Python
2151 lines
52 KiB
Python
|
|
# parse.py
|
|
|
|
"""
|
|
This module is an integeral part of the program
|
|
MMA - Musical Midi Accompaniment.
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
Bob van der Poel <bob@mellowood.ca>
|
|
|
|
|
|
This module does all file parsing. Most commands
|
|
are passed to the track classes; however, things
|
|
like TIME, SEQRND, etc. which just set global flags
|
|
are completely handled here.
|
|
|
|
"""
|
|
|
|
import os
|
|
import random
|
|
import copy
|
|
|
|
import gbl
|
|
from MMA.common import *
|
|
|
|
import MMA.notelen
|
|
import MMA.chords
|
|
import MMA.file
|
|
import MMA.midi
|
|
import MMA.midiIn
|
|
import MMA.grooves
|
|
import MMA.docs
|
|
import MMA.auto
|
|
import MMA.translate
|
|
import MMA.patSolo
|
|
import MMA.mdefine
|
|
import MMA.volume
|
|
import MMA.seqrnd
|
|
import MMA.patch
|
|
|
|
import gbl
|
|
|
|
from MMA.common import *
|
|
from MMA.lyric import lyric
|
|
from MMA.macro import macros
|
|
from MMA.alloc import trackAlloc
|
|
|
|
lastChord = None # tracks last chord for "/ /" data lines.
|
|
|
|
beginData = [] # Current data set by a BEGIN statement
|
|
beginPoints = [] # since BEGINs can be nested, we need ptrs for backing out of BEGINs
|
|
|
|
gmagic = 9988 # magic name for groove saved with USE
|
|
|
|
|
|
|
|
""" This table is passed to the track classes. It has
|
|
an instance for each chord in the current bar.
|
|
"""
|
|
|
|
class CTable:
|
|
chord = None # A pointer to the chordNotes structures
|
|
chordZ = None # set if chord is tacet
|
|
arpeggioZ = None # set if arpeggio is tacet
|
|
walkZ = None # set if walking bass is tacet
|
|
drumZ = None # set if drums are tacet
|
|
bassZ = None # set if bass is tacet
|
|
scaleZ = None # set if scale track is tacet
|
|
ariaZ = None # set if aria track is tacet
|
|
|
|
def __init__(self, offset):
|
|
self.offset=offset
|
|
|
|
|
|
|
|
########################################
|
|
# File processing. Mostly jumps to pats
|
|
########################################
|
|
|
|
|
|
def parseFile(n):
|
|
""" Open and process a file. Errors exit. """
|
|
|
|
fp=gbl.inpath
|
|
|
|
f=MMA.file.ReadFile(n)
|
|
|
|
parse(f)
|
|
gbl.inpath=fp
|
|
|
|
if gbl.debug:
|
|
print "File '%s' closed." % n
|
|
|
|
|
|
def parse(inpath):
|
|
""" Process a mma input file. """
|
|
|
|
global beginData, lastChord
|
|
|
|
gbl.inpath = inpath
|
|
|
|
curline = None
|
|
|
|
while 1:
|
|
curline = inpath.read()
|
|
|
|
if curline == None:
|
|
MMA.docs.docDump()
|
|
break
|
|
|
|
l = macros.expand(curline)
|
|
|
|
""" Handle BEGIN and END here. This is outside of the Repeat/End
|
|
and variable expand loops so SHOULD be pretty bullet proof.
|
|
Note that the beginData stuff is global to this module ... the
|
|
Include/Use directives check to make sure we're not doing that
|
|
inside a Begin/End.
|
|
|
|
beginData[] is a list which we append to as more Begins are
|
|
encountered.
|
|
|
|
The placement here is pretty deliberate. Variable expand comes
|
|
later so you can't macroize BEGIN ... I think this makes sense.
|
|
|
|
The tests for 'begin', 'end' and the appending of the current
|
|
begin[] stuff have to be here, in this order.
|
|
"""
|
|
|
|
action=l[0].upper() # 1st arg in line
|
|
|
|
if action == 'BEGIN':
|
|
if not l:
|
|
error("Use: Begin STUFF")
|
|
beginPoints.append(len(beginData))
|
|
beginData.extend(l[1:])
|
|
continue
|
|
|
|
if action == 'END':
|
|
if len(l) > 1:
|
|
error("No arguments permitted for END")
|
|
if not beginData:
|
|
error("No 'BEGIN' for 'END'")
|
|
beginData=beginData[:beginPoints.pop(-1)]
|
|
continue
|
|
|
|
if beginData:
|
|
l = beginData + l
|
|
action = l[0].upper()
|
|
|
|
|
|
if gbl.showExpand and action !='REPEAT':
|
|
print l
|
|
|
|
# If the command is in the simple function table, jump & loop.
|
|
|
|
if action in simpleFuncs:
|
|
simpleFuncs[action](l[1:])
|
|
continue
|
|
|
|
|
|
""" We have several possibilities ...
|
|
1. The command is a valid assigned track name,
|
|
2. The command is a valid track name, but needs to be
|
|
dynamically allocated,
|
|
3. It's really a chord action
|
|
"""
|
|
|
|
if not action in gbl.tnames:
|
|
trackAlloc(action, 0) # ensure that track is allocated
|
|
|
|
if action in gbl.tnames: # BASS/DRUM/APEGGIO/CHORD
|
|
|
|
name = action
|
|
if len(l) < 2:
|
|
error("Expecting argument after '%s'" % name)
|
|
action = l[1].upper()
|
|
|
|
if action in trackFuncs:
|
|
trackFuncs[action](name, l[2:])
|
|
else:
|
|
error ("Don't know '%s'" % curline)
|
|
|
|
continue
|
|
|
|
### Gotta be a chord data line!
|
|
|
|
|
|
""" A data line can have an optional bar number at the start
|
|
of the line. Makes debugging input easier. The next
|
|
block strips leading integers off the line. Note that
|
|
a line number on a line by itself it okay.
|
|
"""
|
|
|
|
if action.isdigit(): # isdigit() matches '1', '1234' but not '1a'!
|
|
l = l[1:]
|
|
if not l: # ignore empty lines
|
|
continue
|
|
|
|
""" A bar can have an optional repeat count. This must
|
|
be at the end of bar in the form '* xx'.
|
|
"""
|
|
|
|
if len(l)>1 and l[-2]=='*':
|
|
rptcount = stoi(l[-1], "Expecting integer after '*'")
|
|
l=l[:-2]
|
|
else:
|
|
rptcount = 1
|
|
|
|
|
|
""" Extract solo(s) from line ... this is anything in {}s.
|
|
The solo data is pushed into RIFFs and discarded from
|
|
the current line.
|
|
"""
|
|
|
|
l = ' '.join(l)
|
|
l = MMA.patSolo.extractSolo(l, rptcount)
|
|
|
|
""" Set lyrics from [stuff] in the current line.
|
|
NOTE: lyric.extract() inserts previously created
|
|
data from LYRICS SET and inserts the chord names
|
|
if that flag is active.
|
|
"""
|
|
|
|
l, lyrics = lyric.extract(l, rptcount)
|
|
|
|
l = l.split()
|
|
|
|
""" At this point we have only chord info. A number
|
|
of sanity checks are made:
|
|
1. Make sure there is some chord data,
|
|
2. Ensure the correct number of chords.
|
|
"""
|
|
|
|
if not l:
|
|
error("Expecting music (chord) data. Even lines with\n"
|
|
" lyrics or solos still need a chord")
|
|
|
|
i = gbl.QperBar - len(l)
|
|
if i < 0:
|
|
error("Too many chords in line. Max is %s, not %s" %
|
|
(gbl.QperBar, len(l) ) )
|
|
if i:
|
|
l.extend( ['/'] * i )
|
|
|
|
|
|
""" We now have a valid line. It'll look something like:
|
|
|
|
['Cm', '/', 'z', 'F#'] or ['C', '/', '/', '/' ]
|
|
|
|
For each bar we create a ctable structure. This is just
|
|
a list of CTables, one for each beat division.
|
|
Each entry has the offset (in midi ticks), chordname, etc.
|
|
|
|
Special processing in needed for 'z' options in chords. A 'z' can
|
|
be of the form 'CHORDzX', 'z!' or just 'z'.
|
|
"""
|
|
|
|
beat = 0
|
|
ctable = []
|
|
|
|
for c in l:
|
|
if c == '/':
|
|
if not lastChord:
|
|
error("A chord has to be set before you can use a '/'")
|
|
c = lastChord
|
|
else:
|
|
lastChord = c
|
|
|
|
ctable.append(parseZs(c, beat))
|
|
beat += 1
|
|
|
|
# Create MIDI data for the bar
|
|
|
|
for rpt in range(rptcount): # for each bar in the repeat count ( Cm * 3)
|
|
|
|
""" Handle global (de)cresc by popping a new volume off stack. """
|
|
|
|
if MMA.volume.futureVol:
|
|
MMA.volume.volume = MMA.volume.futureVol.pop(0)
|
|
if MMA.volume.futureVol:
|
|
MMA.volume.nextVolume = MMA.volume.futureVol[0]
|
|
else:
|
|
MMA.volume.nextVolume = None
|
|
|
|
|
|
""" Set up for rnd seq. This may set the current seq point. If return
|
|
is >=0 then we're doing track rnd.
|
|
"""
|
|
|
|
rsq, seqlist = MMA.seqrnd.setseq()
|
|
|
|
|
|
""" Process each track. It is important that the track classes
|
|
are written so that the ctable passed to them IS NOT MODIFIED.
|
|
This applies especially to chords. If the track class changes
|
|
the chord, then restore it before returning!!!
|
|
"""
|
|
|
|
for a in gbl.tnames.values():
|
|
if rsq >= 0:
|
|
seqSave = gbl.seqCount
|
|
if a.name in seqlist: # for seqrnd with tracklist
|
|
gbl.seqCount = rsq
|
|
|
|
a.bar(ctable) ## process entire bar!
|
|
|
|
if rsq >= 0:
|
|
gbl.seqCount = seqSave
|
|
|
|
# Adjust counters
|
|
|
|
gbl.totTime += float(gbl.QperBar) / gbl.tempo
|
|
|
|
gbl.barNum += 1
|
|
|
|
if gbl.barNum > gbl.maxBars:
|
|
error("Capacity exceeded. Maxbar setting is %s. Use -m option"
|
|
% gbl.maxBars)
|
|
|
|
gbl.tickOffset += (gbl.QperBar * gbl.BperQ)
|
|
|
|
gbl.seqCount = (gbl.seqCount+1) % gbl.seqSize
|
|
|
|
MMA.grooves.nextGroove() # using groove list? Advance.
|
|
|
|
# Enabled with the -r command line option
|
|
|
|
if gbl.showrun:
|
|
print "%3d:" % gbl.barNum,
|
|
for c in l:
|
|
print c,
|
|
if lyrics:
|
|
print lyrics,
|
|
print
|
|
|
|
|
|
def parseZs(c, beat):
|
|
""" Parse a chord in a barline, create Ctable and strips 'z's.
|
|
|
|
This is called only from the main parser, but it's
|
|
complicated (ugly) enough to have its own function.
|
|
"""
|
|
|
|
ctab = CTable(beat * gbl.BperQ)
|
|
|
|
if 'z' in c:
|
|
c, r = c.split('z', 1) # chord name/track mute
|
|
|
|
if not c:
|
|
if r=='!': # mute all for 'z!'
|
|
r='DCAWBSR'
|
|
c='z' # dummy chord name
|
|
elif not r: # mute all tracks except Drum 'z'
|
|
r='CBAWSR'
|
|
c='z'
|
|
|
|
else:
|
|
error("To mute individual tracks you must "
|
|
"use a chord/z combination not '%s'" % r)
|
|
|
|
else: # illegal construct -- 'Cz!'
|
|
if r=='!':
|
|
error("'%sz!' is illegal. 'z!' mutes all tracks "
|
|
"so you can't include the chord" % c)
|
|
|
|
elif not r:
|
|
error("'%sz' is illegal. You must specify tracks "
|
|
"if you use a chord" % c )
|
|
|
|
for v in r:
|
|
if v == 'C':
|
|
ctab.chordZ = 1
|
|
elif v == 'B':
|
|
ctab.bassZ = 1
|
|
elif v == 'A':
|
|
ctab.arpeggioZ = 1
|
|
elif v == 'W':
|
|
ctab.walkZ = 1
|
|
elif v == 'D':
|
|
ctab.drumZ = 1
|
|
elif v == 'S':
|
|
ctab.scaleZ = 1
|
|
elif v == 'R':
|
|
ctab.ariaZ = 1
|
|
|
|
else:
|
|
error("Unknown voice '%s' for rest in '%s'" % (v,r))
|
|
|
|
ctab.chord = MMA.chords.ChordNotes(c)
|
|
|
|
return ctab
|
|
|
|
##################################################################
|
|
|
|
def allTracks(ln):
|
|
""" Apply track to all tracks. """
|
|
|
|
allTypes = ('BASS', 'CHORD', 'ARPEGGIO', 'SCALE', 'DRUM', 'WALK', 'MELODY', 'SOLO')
|
|
ttypes = []
|
|
|
|
if len(ln) < 1:
|
|
error("AllTracks: argument (track?) required")
|
|
|
|
i = 0
|
|
while i < len(ln) and ln[i].upper() in allTypes:
|
|
ttypes.append(ln[i].upper())
|
|
i += 1
|
|
|
|
if ttypes == []:
|
|
ttypes = allTypes
|
|
|
|
if i>=len(ln):
|
|
error("AllTracks: Additional argument (command?) required")
|
|
|
|
cmd = ln[i].upper()
|
|
args = i+1
|
|
|
|
if not cmd in trackFuncs:
|
|
error("AllTracks: command '%s' doen't exist" % cmd)
|
|
|
|
for n in gbl.tnames:
|
|
if not gbl.tnames[n].vtype in ttypes:
|
|
continue
|
|
|
|
trackFuncs[cmd](n, ln[args:])
|
|
|
|
|
|
#######################################
|
|
# Do-nothing functions
|
|
|
|
def comment(ln):
|
|
pass
|
|
|
|
def repeatend(ln):
|
|
error("Repeatend/EndRepeat without Repeat")
|
|
|
|
def repeatending(ln):
|
|
error("Repeatending without Repeat")
|
|
|
|
def endmset(ln):
|
|
error("EndMset/MSetEnd without If")
|
|
|
|
def ifend(ln):
|
|
error("ENDIF without IF")
|
|
|
|
def ifelse(ln):
|
|
error("ELSE without IF")
|
|
|
|
|
|
|
|
#######################################
|
|
# Repeat/jumps
|
|
|
|
|
|
def repeat(ln):
|
|
""" Repeat/RepeatEnd/RepeatEnding.
|
|
|
|
Read input until a RepeatEnd is found. The entire
|
|
chunk is pushed back into the input stream the
|
|
correct number of times. This accounts for endings and
|
|
nested repeats.
|
|
"""
|
|
|
|
|
|
def repeatChunk():
|
|
q=[]
|
|
qnum=[]
|
|
nesting = 0
|
|
|
|
while 1:
|
|
l=gbl.inpath.read()
|
|
|
|
if not l:
|
|
error("EOF encountered processing Repeat")
|
|
|
|
act=l[0].upper()
|
|
|
|
if act=='REPEAT':
|
|
nesting += 1
|
|
|
|
elif act in ('REPEATEND', 'ENDREPEAT') and nesting:
|
|
nesting -= 1
|
|
|
|
elif act == 'REPEATENDING' and nesting:
|
|
pass
|
|
|
|
elif act in ('REPEATEND', 'ENDREPEAT', 'REPEATENDING'):
|
|
return (q, qnum, act, l[1:])
|
|
|
|
q.append(l)
|
|
qnum.append(gbl.lineno)
|
|
|
|
stack=[]
|
|
stacknum=[]
|
|
main=[]
|
|
mainnum=[]
|
|
ending = 0
|
|
|
|
if ln:
|
|
error("REPEAT takes no arguments")
|
|
|
|
main, mainnum, act, l = repeatChunk()
|
|
|
|
while 1:
|
|
if act in ('REPEATEND', 'ENDREPEAT'):
|
|
if l:
|
|
l = macros.expand(l)
|
|
if len(l) == 2 and l[0].upper() == 'NOWARN':
|
|
l=l[1:]
|
|
warn=0
|
|
else:
|
|
warn=1
|
|
|
|
if len(l) != 1:
|
|
error("%s: Use [NoWarn] Count" % act)
|
|
|
|
count=stoi(l[0], "%s takes an integer arg" % act)
|
|
|
|
if count == 2 and warn:
|
|
warning("%s count of 2 duplicates default. Did you mean 3 or more?" % act)
|
|
|
|
elif count == 1 and warn:
|
|
warning("%s count of 1 means NO REPEAT" % act)
|
|
|
|
elif count == 0 and warn:
|
|
warning("%s count of 0, Skipping entire repeated section" % act)
|
|
|
|
elif count < 0:
|
|
error("%s count must be 0 or greater" % act)
|
|
|
|
elif count > 10 and warn:
|
|
warning("%s is a large value for %s" % (count, act) )
|
|
|
|
else:
|
|
count=2
|
|
|
|
if not ending:
|
|
count += 1
|
|
for c in range(count-1):
|
|
stack.extend(main)
|
|
stacknum.extend(mainnum)
|
|
gbl.inpath.push(stack, stacknum)
|
|
break
|
|
|
|
elif act == 'REPEATENDING':
|
|
ending = 1
|
|
|
|
if l:
|
|
l = macros.expand(l)
|
|
if len(l) == 2 and l[0].upper() == 'NOWARN':
|
|
warn=0
|
|
l=l[1:]
|
|
else:
|
|
warn=1
|
|
|
|
if len(l) != 1:
|
|
error("REPEATENDING: Use [NoWarn] Count")
|
|
|
|
count=stoi(l[0], "RepeatEnding takes an integer arg")
|
|
|
|
if count < 0:
|
|
error("RepeatEnding count must be postive, not '%s'" % count)
|
|
|
|
elif count == 0 and warn:
|
|
warning("RepeatEnding count of 0, skipping section")
|
|
|
|
elif count == 1 and warn:
|
|
warning("RepeatEnding count of 1 duplicates default")
|
|
|
|
elif count > 10 and warn:
|
|
warning("%s is a large value for RepeatEnding" % count)
|
|
else:
|
|
count = 1
|
|
|
|
rpt, rptnum, act, l = repeatChunk()
|
|
|
|
for c in range(count):
|
|
stack.extend(main)
|
|
stacknum.extend(mainnum)
|
|
stack.extend(rpt)
|
|
stacknum.extend(rptnum)
|
|
|
|
|
|
else:
|
|
error("Unexpected line in REPEAT")
|
|
|
|
def goto(ln):
|
|
if len(ln) != 1:
|
|
error("Usage: GOTO Label")
|
|
gbl.inpath.goto(ln[0].upper())
|
|
|
|
def eof(ln):
|
|
gbl.inpath.toEof()
|
|
|
|
|
|
#######################################
|
|
# Tempo/timing
|
|
|
|
|
|
def setTime(ln):
|
|
""" Set the 'time sig'.
|
|
|
|
We do restrict the time setting to the range of 1..12.
|
|
No particular reason, but we do need some limit? Certainly
|
|
it has to be greater than 0.
|
|
"""
|
|
|
|
if len(ln) != 1:
|
|
error("Use: Time N")
|
|
|
|
n = stoi(ln[0], "Argument for time must be integer")
|
|
|
|
if n < 1 or n > 12:
|
|
error("Time (beats/bar) must be 1..12")
|
|
|
|
# If no change, just ignore this.
|
|
|
|
if gbl.QperBar != n:
|
|
gbl.QperBar = int(n)
|
|
|
|
# Time changes zap all predfined sequences
|
|
|
|
for a in gbl.tnames.values():
|
|
a.clearSequence()
|
|
|
|
|
|
def tempo(ln):
|
|
""" Set tempo. """
|
|
|
|
if not ln or len(ln) >2:
|
|
error("Use: Tempo [*,+,-]BperM [BARS]")
|
|
|
|
# Get new value.
|
|
|
|
a = ln[0][0]
|
|
if a in "+-*":
|
|
v = stof(ln[0][1:], "Tempo expecting value for rate adjustment, not '%s'" % ln[0])
|
|
if a == '-':
|
|
v = gbl.tempo - v
|
|
elif a == '+':
|
|
v += gbl.tempo
|
|
elif a == '*':
|
|
v *= gbl.tempo
|
|
|
|
else:
|
|
v = stof(ln[0], "Tempo expecting rate, not '%s'" % ln[0])
|
|
|
|
|
|
# is this immediate or over time?
|
|
|
|
if len(ln) == 1:
|
|
gbl.tempo = int(v)
|
|
gbl.mtrks[0].addTempo(gbl.tickOffset, gbl.tempo)
|
|
if gbl.debug:
|
|
print "Set Tempo to %s" % gbl.tempo
|
|
|
|
|
|
else: # Do a tempo change over bar count
|
|
bars = ln[1]
|
|
|
|
bars = stof(bars, "Expecting value, not %s" % bars )
|
|
numbeats = int(bars * gbl.QperBar)
|
|
|
|
if numbeats < 1:
|
|
error("Beat count must be greater than 1")
|
|
|
|
# Vary the rate in the meta track
|
|
|
|
tincr = (v - gbl.tempo) / float(numbeats) # incr per beat
|
|
bstart = gbl.tickOffset # start
|
|
boff = 0
|
|
tempo = gbl.tempo
|
|
|
|
for n in range(numbeats):
|
|
tempo += tincr
|
|
if tempo:
|
|
gbl.mtrks[0].addTempo(bstart + boff, int(tempo))
|
|
boff += gbl.BperQ
|
|
|
|
if tempo != v:
|
|
gbl.mtrks[0].addTempo(bstart + boff, int(v) )
|
|
|
|
gbl.tempo = int(v)
|
|
|
|
if gbl.debug:
|
|
print "Set future Tempo to %s over %s beats" % \
|
|
( int(tempo), numbeats)
|
|
|
|
if gbl.tempo <=0:
|
|
error("Tempo setting must be greater than 0.")
|
|
|
|
|
|
def beatAdjust(ln):
|
|
""" Delete or insert some beats into the sequence.
|
|
|
|
This just adjusts the current song position. Nothing is
|
|
lost or added to the actual file.
|
|
"""
|
|
|
|
|
|
if len(ln) != 1:
|
|
error("Use: BeatAdjust NN")
|
|
|
|
adj = stof(ln[0], "Expecting a value (not %s) for BeatAdjust" % ln[0])
|
|
|
|
gbl.tickOffset += int(adj * gbl.BperQ)
|
|
|
|
gbl.totTime += adj / gbl.tempo # adjust total time
|
|
|
|
if gbl.debug:
|
|
print "BeatAdjust: inserted %s at bar %s." % (adj, gbl.barNum + 1)
|
|
|
|
|
|
def cut(ln):
|
|
""" Insert a all-note-off into all tracks. """
|
|
|
|
if not len(ln):
|
|
ln=['0']
|
|
|
|
if len(ln) != 1:
|
|
error("Use: Cut Offset")
|
|
|
|
""" Loop though all the tracks. Note that trackCut() checks
|
|
to make sure that there is a need to insert in specified track.
|
|
In this loop we create a list of channels as we loop though
|
|
all the tracks, skipping over any duplicate channels or
|
|
tracks with no channel assigned.
|
|
"""
|
|
|
|
l=[]
|
|
for t in sorted(gbl.tnames.keys()):
|
|
c = gbl.tnames[t].channel
|
|
if not c or c in l:
|
|
continue
|
|
l.append(c)
|
|
trackCut(t, ln)
|
|
|
|
|
|
def fermata(ln):
|
|
""" Apply a fermata timing to the specified beat. """
|
|
|
|
if len(ln) != 3:
|
|
error("Use: Fermata 'offset' 'duration' 'adjustment'")
|
|
|
|
offset = stof(ln[0], "Expecting a value (not '%s') "
|
|
"for Fermata Offset" % ln[0] )
|
|
|
|
if offset < -gbl.QperBar or offset > gbl.QperBar:
|
|
warning("Fermata: %s is a large beat offset" % offset)
|
|
|
|
dur = stof(ln[1], "Expecting a value (not '%s') for Fermata Duration" % ln[1])
|
|
|
|
if dur <= 0:
|
|
error("Fermata duration must be greater than 0")
|
|
|
|
if dur > gbl.QperBar:
|
|
warning("Fermata: %s is a large duration" % dur)
|
|
|
|
adj = stof(ln[2], "Expecting a value (not '%s') for Fermata Adjustment" % ln[2])
|
|
|
|
if adj< 100:
|
|
warning("Fermata: Adjustment less than 100 is shortening beat value")
|
|
|
|
if adj == 100:
|
|
error("Fermata: using value of 100 makes no difference, must be an error")
|
|
|
|
moff=int(gbl.tickOffset + (gbl.BperQ * offset))
|
|
|
|
if moff < 0:
|
|
error("Fermata offset comes before track start")
|
|
|
|
gbl.mtrks[0].addTempo(moff, int(gbl.tempo / (adj/100)) )
|
|
|
|
tickDur = int(gbl.BperQ * dur)
|
|
|
|
gbl.mtrks[0].addTempo(moff + tickDur, gbl.tempo)
|
|
|
|
# Clear out NoteOn events in all tracks
|
|
|
|
if offset < 0:
|
|
start = moff + int(.05 * gbl.BperQ)
|
|
end = moff + tickDur - int(.05 * gbl.BperQ)
|
|
|
|
for n, tr in gbl.mtrks.items():
|
|
if n <= 0: continue # skip meta track
|
|
tr.zapRangeTrack(start, end )
|
|
|
|
if gbl.debug:
|
|
print "Fermata: Beat %s, Duration %s, Change %s, Bar %s" % \
|
|
(offset, dur, adj, gbl.barNum + 1)
|
|
if offset < 0:
|
|
print "\tNoteOn Events removed in tick range %s to %s" \
|
|
% (start, end)
|
|
|
|
|
|
|
|
#######################################
|
|
# File and I/O
|
|
|
|
def include(ln):
|
|
""" Include a file. """
|
|
|
|
global beginData
|
|
|
|
if beginData:
|
|
error("INCLUDE not permitted in Begin/End block")
|
|
|
|
if len(ln) != 1:
|
|
error("Use: Include FILE" )
|
|
|
|
fn = MMA.file.locFile(ln[0], gbl.incPath)
|
|
if not fn:
|
|
error("Could not find include file '%s'" % ln)
|
|
|
|
else:
|
|
parseFile(fn)
|
|
|
|
|
|
def usefile(ln):
|
|
""" Include a library file. """
|
|
|
|
global beginData
|
|
|
|
if beginData:
|
|
error("USE not permitted in Begin/End block")
|
|
|
|
if len(ln) != 1:
|
|
error("Use: Use FILE")
|
|
|
|
ln = ln[0]
|
|
fn = MMA.file.locFile(ln, gbl.libPath)
|
|
|
|
if not fn:
|
|
error("Unable to locate library file '%s'" % ln)
|
|
|
|
""" USE saves current state, just like defining a groove.
|
|
Here we use a magic number which can't be created with
|
|
a defgroove ('cause it's an integer). Save, read, restore.
|
|
"""
|
|
|
|
slot = gmagic
|
|
MMA.grooves.grooveDefineDo(slot)
|
|
parseFile(fn)
|
|
MMA.grooves.grooveDo(slot)
|
|
|
|
def mmastart(ln):
|
|
if not ln:
|
|
error ("Use: MMAstart FILE [file...]")
|
|
|
|
gbl.mmaStart.extend(ln)
|
|
|
|
if gbl.debug:
|
|
print "MMAstart set to:",
|
|
printList(ln)
|
|
|
|
def mmaend(ln):
|
|
if not ln:
|
|
error ("Use: MMAend FILE [file...]")
|
|
|
|
gbl.mmaEnd.extend(ln)
|
|
|
|
if gbl.debug:
|
|
print "MMAend set to:",
|
|
printList(ln)
|
|
|
|
|
|
def setLibPath(ln):
|
|
""" Set the LibPath variable. """
|
|
|
|
if len(ln) > 1:
|
|
error("Only one path can be entered for LibPath")
|
|
|
|
f = os.path.expanduser(ln[0])
|
|
|
|
if gbl.debug:
|
|
print "LibPath set to", f
|
|
|
|
gbl.libPath = f
|
|
|
|
|
|
def setAutoPath(ln):
|
|
""" Set the autoPath variable. """
|
|
|
|
if len(ln) > 1:
|
|
error("Only one path can be entered for AutoLibPath")
|
|
|
|
f = os.path.expanduser(ln[0])
|
|
|
|
MMA.auto.grooveDir = {}
|
|
|
|
# To avoid conflicts, delete all existing grooves (current seq not effected)
|
|
|
|
MMA.grooves.glist = {}
|
|
MMA.grooves.lastGroove = ''
|
|
MMA.grooves.currentGroove = ''
|
|
|
|
if gbl.debug:
|
|
print "AutoLibPath set to", f
|
|
|
|
gbl.autoLib = f
|
|
|
|
|
|
def setIncPath(ln):
|
|
""" Set the IncPath variable. """
|
|
|
|
if len(ln)>1:
|
|
error("Only one path is permitted in SetIncPath")
|
|
|
|
f = os.path.expanduser(ln[0])
|
|
|
|
if gbl.debug:
|
|
print "IncPath set to", f
|
|
|
|
gbl.incPath=f
|
|
|
|
|
|
def setOutPath(ln):
|
|
""" Set the Outpath variable. """
|
|
|
|
if not ln:
|
|
gbl.outPath = ""
|
|
|
|
elif len(ln) > 1:
|
|
error ("Use: SetOutPath PATH")
|
|
|
|
else:
|
|
gbl.outPath = os.path.expanduser(ln[0])
|
|
|
|
if gbl.debug:
|
|
print "OutPath set to", gbl.outPath
|
|
|
|
|
|
def setMidiPlayer(ln):
|
|
""" Set the MIDI file player (used with -P). """
|
|
|
|
if len(ln) != 1:
|
|
error("Use: MidiPlayer <program name>")
|
|
|
|
gbl.midiPlayer = ln[0]
|
|
|
|
if gbl.debug:
|
|
print "MidiPlayer set to", gbl.MidiPlayer
|
|
|
|
|
|
#######################################
|
|
# Sequence
|
|
|
|
def seqsize(ln):
|
|
""" Set the length of sequences. """
|
|
|
|
if len(ln) !=1:
|
|
error("Usage 'SeqSize N'")
|
|
|
|
n = stoi(ln[0], "Argument for SeqSize must be integer")
|
|
|
|
if n < 1:
|
|
error("SeqSize: sequence size must be 1 or greater, not '%s'." % n)
|
|
|
|
# Setting the sequence size always resets the seq point
|
|
|
|
gbl.seqCount = 0
|
|
|
|
""" Now set the sequence size for each track. The class call
|
|
will expand/contract existing patterns to match the new
|
|
size.
|
|
"""
|
|
|
|
if n != gbl.seqSize:
|
|
gbl.seqSize = n
|
|
for a in gbl.tnames.values():
|
|
a.setSeqSize()
|
|
|
|
MMA.seqrnd.seqRndWeight = seqBump(MMA.seqrnd.seqRndWeight)
|
|
|
|
if gbl.debug:
|
|
print "Set SeqSize to ", n
|
|
|
|
|
|
def seq(ln):
|
|
""" Set the sequence point. """
|
|
|
|
if len(ln) == 0:
|
|
s = 0
|
|
elif len(ln)==1:
|
|
s = stoi(ln[0], "Expecting integer value after SEQ")
|
|
else:
|
|
error("Use: SEQ or SEQ NN to reset seq point")
|
|
|
|
|
|
if s > gbl.seqSize:
|
|
error("Sequence size is '%d', you can't set to '%d'" %
|
|
(gbl.seqSize, s))
|
|
|
|
if s==0:
|
|
s=1
|
|
|
|
if s<0:
|
|
error("Seq parm must be greater than 0, not %s", s)
|
|
|
|
gbl.seqCount = s-1
|
|
|
|
if MMA.seqrnd.seqRnd[0] == 1:
|
|
warning("SeqRnd has been disabled by a Seq command")
|
|
MMA.seqrnd.seqRnd = [0]
|
|
|
|
|
|
def seqClear(ln):
|
|
""" Clear all sequences (except SOLO tracks). """
|
|
|
|
if ln:
|
|
error ("Use: 'SeqClear' with no args")
|
|
|
|
for n in gbl.tnames.values():
|
|
if n.vtype != "SOLO":
|
|
n.clearSequence()
|
|
MMA.volume.futureVol = []
|
|
|
|
MMA.seqrnd.setSeqRndWeight(['1'])
|
|
|
|
|
|
|
|
def restart(ln):
|
|
""" Restart all tracks to almost-default condidions. """
|
|
|
|
if ln:
|
|
error ("Use: 'Restart' with no args")
|
|
|
|
for n in gbl.tnames.values():
|
|
n.restart()
|
|
|
|
|
|
#######################################
|
|
# Midi
|
|
|
|
def midiMarker(ln):
|
|
""" Parse off midi marker. """
|
|
|
|
if len(ln) == 2:
|
|
offset = stof(ln[0])
|
|
msg = ln[1]
|
|
elif len(ln) == 1:
|
|
offset = 0
|
|
msg = ln[0]
|
|
else:
|
|
error("Usage: MidiMark [offset] Label")
|
|
|
|
offset = int(gbl.tickOffset + (gbl.BperQ * offset))
|
|
if offset < 0:
|
|
error("MidiMark offset points before start of file")
|
|
|
|
gbl.mtrks[0].addMarker(offset, msg)
|
|
|
|
|
|
def rawMidi(ln):
|
|
""" Send hex bytes as raw midi stream. """
|
|
|
|
mb=''
|
|
for a in ln:
|
|
a=stoi(a)
|
|
|
|
if a<0 or a >0xff:
|
|
error("All values must be in the range "
|
|
"0 to 0xff, not '%s'" % a)
|
|
|
|
mb += chr(a)
|
|
|
|
gbl.mtrks[0].addToTrack(gbl.tickOffset, mb)
|
|
|
|
if gbl.debug:
|
|
print "Inserted raw midi in metatrack: ",
|
|
for b in mb:
|
|
print '%02x' % ord(b),
|
|
print
|
|
|
|
|
|
def mdefine(ln):
|
|
""" Set a midi seq pattern. """
|
|
|
|
if not ln:
|
|
error("MDefine needs arguments")
|
|
|
|
name = ln[0]
|
|
if name.startswith('_'):
|
|
error("Names with a leading underscore are reserved")
|
|
|
|
if name.upper() == 'Z':
|
|
error("The name 'Z' is reserved")
|
|
|
|
MMA.mdefine.mdef.set(name, ' '.join(ln[1:]))
|
|
|
|
|
|
def setMidiFileType(ln):
|
|
""" Set some MIDI file generation flags. """
|
|
|
|
if not ln:
|
|
error("USE: MidiFile [SMF=0/1] [RUNNING=0/1]")
|
|
|
|
for l in ln:
|
|
try:
|
|
mode, val = l.upper().split('=')
|
|
except:
|
|
error("Each arg must contain an '=', not '%s'" % l)
|
|
|
|
if mode == 'SMF':
|
|
if val == '0':
|
|
gbl.midiFileType = 0
|
|
elif val == '1':
|
|
gbl.midiFileType = 1
|
|
else:
|
|
error("Use: MIDIFile SMF=0/1")
|
|
|
|
if gbl.debug:
|
|
print "Midi Filetype set to", gbl.midiFileType
|
|
|
|
|
|
elif mode == 'RUNNING':
|
|
if val == '0':
|
|
gbl.runningStatus = 0
|
|
elif val == '1':
|
|
gbl.runningStatus = 1
|
|
else:
|
|
error("Use: MIDIFile RUNNING=0/1")
|
|
|
|
if gbl.debug:
|
|
print "Midi Running Status Generation set to",
|
|
if gbl.runningStatus:
|
|
print 'ON (Default)'
|
|
else:
|
|
print 'OFF'
|
|
|
|
|
|
else:
|
|
error("Use: MIDIFile [SMF=0/1] [RUNNING=0/1]")
|
|
|
|
|
|
def setChPref(ln):
|
|
""" Set MIDI Channel Preference. """
|
|
|
|
if not ln:
|
|
error("Use: ChannelPref TRACKNAME=CHANNEL [...]")
|
|
|
|
for i in ln:
|
|
if '=' not in i:
|
|
error("Each item in ChannelPref must have an '='")
|
|
|
|
n,c = i.split('=')
|
|
|
|
c = stoi(c, "Expecting an integer for ChannelPref, not '%s'" % c)
|
|
|
|
if c<1 or c>16:
|
|
error("Channel for ChannelPref must be 1..16, not %s" % c)
|
|
|
|
gbl.midiChPrefs[n.upper()]=c
|
|
|
|
if gbl.debug:
|
|
print "ChannelPref:",
|
|
for n,c in gbl.midiChPrefs.items():
|
|
print "%s=%s" % (n,c),
|
|
print
|
|
|
|
|
|
def setTimeSig(ln):
|
|
""" Set the midi time signature. """
|
|
|
|
if len(ln) == 1:
|
|
a=ln[0].upper()
|
|
if a == 'COMMON':
|
|
ln=('4','4')
|
|
elif a == 'CUT':
|
|
ln=('2','2')
|
|
|
|
if len(ln) != 2:
|
|
error("TimeSig: Usage (num dem) or ('cut' or 'common')")
|
|
|
|
nn = stoi(ln[0])
|
|
|
|
if nn<1 or nn>126:
|
|
error("Timesig NN must be 1..126")
|
|
|
|
dd = stoi(ln[1])
|
|
if dd == 1: dd = 0
|
|
elif dd == 2: dd = 1
|
|
elif dd == 4: dd = 2
|
|
elif dd == 8: dd = 3
|
|
elif dd == 16: dd = 4
|
|
elif dd == 32: dd = 5
|
|
elif dd == 64: dd = 6
|
|
else:
|
|
error("Unknown value for timesig denominator")
|
|
|
|
MMA.midi.timeSig.set(nn,dd)
|
|
|
|
|
|
|
|
|
|
#######################################
|
|
# Misc
|
|
|
|
def synchronize(ln):
|
|
""" Set synchronization in the MIDI. A file mode for -0 and -1. """
|
|
|
|
if not ln:
|
|
error("SYNCHRONIZE: requires args END and/or START.")
|
|
|
|
for a in ln:
|
|
if a.upper() == 'END':
|
|
gbl.endsync = 1
|
|
elif a.upper() == 'START':
|
|
gbl.synctick = 1
|
|
else:
|
|
error("SYNCHRONIZE: expecting END or START")
|
|
|
|
|
|
def rndseed(ln):
|
|
""" Reseed the random number generator. """
|
|
|
|
if not ln:
|
|
random.seed()
|
|
|
|
elif len(ln)>1:
|
|
error("RNDSEED: requires 0 or 1 arguments")
|
|
else:
|
|
random.seed(stof(ln[0]))
|
|
|
|
def transpose(ln):
|
|
""" Set transpose value. """
|
|
|
|
|
|
if len(ln) != 1:
|
|
error("Use: Transpose N")
|
|
|
|
t = stoi(ln[0], "Argument for Tranpose must be an integer, not '%s'" % ln[0])
|
|
if t < -12 or t > 12:
|
|
error("Tranpose %s out-of-range; must be -12..12" % t)
|
|
|
|
gbl.transpose = t
|
|
|
|
if gbl.debug:
|
|
print "Set Transpose to %s" % t
|
|
|
|
|
|
def lnPrint(ln):
|
|
""" Print stuff in a "print" command. """
|
|
|
|
print " ".join(ln)
|
|
|
|
|
|
def printActive(ln):
|
|
""" Print a list of the active tracks. """
|
|
|
|
print "Active tracks, groove:", MMA.grooves.currentGroove, ' '.join(ln)
|
|
|
|
for a in sorted(gbl.tnames.keys()):
|
|
f=gbl.tnames[a]
|
|
if f.sequence:
|
|
print " ",a
|
|
print
|
|
|
|
|
|
def setDebug(ln):
|
|
""" Set debugging options dynamically. """
|
|
|
|
msg=( "Use: Debug MODE=On/Off where MODE is one or more of "
|
|
"DEBUG, FILENAMES, PATTERNS, SEQUENCE, "
|
|
"RUNTIME, WARNINGS or EXPAND" )
|
|
|
|
|
|
if not len(ln):
|
|
error(msg)
|
|
|
|
# save current flags
|
|
|
|
gbl.Ldebug = gbl.debug
|
|
gbl.LshowFilenames = gbl.showFilenames
|
|
gbl.Lpshow = gbl.pshow
|
|
gbl.Lseqshow = gbl.seqshow
|
|
gbl.Lshowrun = gbl.showrun
|
|
gbl.LnoWarn = gbl.noWarn
|
|
gbl.LnoOutput = gbl.noOutput
|
|
gbl.LshowExpand = gbl.showExpand
|
|
gbl.Lchshow = gbl.chshow
|
|
|
|
|
|
for l in ln:
|
|
try:
|
|
mode, val = l.upper().split('=')
|
|
except:
|
|
error("Each debug option must contain a '=', not '%s'" % l)
|
|
|
|
if val == 'ON' or val == '1':
|
|
setting = 1
|
|
elif val == 'OFF' or val == '0':
|
|
setting = 0
|
|
else:
|
|
error(msg)
|
|
|
|
if mode == 'DEBUG':
|
|
gbl.debug = setting
|
|
if gbl.debug:
|
|
print "Debug=%s." % val
|
|
|
|
elif mode == 'FILENAMES':
|
|
gbl.showFilenames = setting
|
|
if gbl.debug:
|
|
print "ShowFilenames=%s." % val
|
|
|
|
elif mode == 'PATTERNS':
|
|
gbl.pshow = setting
|
|
if gbl.debug:
|
|
print "Pattern display=%s." % val
|
|
|
|
elif mode == 'SEQUENCE':
|
|
gbl.seqshow = setting
|
|
if gbl.debug:
|
|
print "Sequence display=%s." % val
|
|
|
|
elif mode == 'RUNTIME':
|
|
gbl.showrun = setting
|
|
if gbl.debug:
|
|
print "Runtime display=%s." % val
|
|
|
|
elif mode == 'WARNINGS':
|
|
gbl.noWarn = not(setting)
|
|
if gbl.debug:
|
|
print "Warning display=%s" % val
|
|
|
|
elif mode == 'EXPAND':
|
|
gbl.showExpand = setting
|
|
if gbl.debug:
|
|
print "Expand display=%s." % val
|
|
|
|
else:
|
|
error(msg)
|
|
|
|
|
|
|
|
###########################################################
|
|
###########################################################
|
|
## Track specific commands
|
|
|
|
|
|
#######################################
|
|
# Pattern/Groove
|
|
|
|
def trackDefPattern(name, ln):
|
|
""" Define a pattern for a track.
|
|
|
|
Use the type-name for all defines.... check the track
|
|
names and if it has a '-' in it, we use only the
|
|
part BEFORE the '-'. So DRUM-Snare becomes DRUM.
|
|
"""
|
|
|
|
ln=ln[:]
|
|
|
|
name=name.split('-')[0]
|
|
|
|
trackAlloc(name, 1)
|
|
|
|
if ln:
|
|
pattern = ln.pop(0).upper()
|
|
else:
|
|
error("Define is expecting a pattern name")
|
|
|
|
if pattern in ('z', 'Z', '-'):
|
|
error("Pattern name '%s' is reserved" % pattern)
|
|
|
|
if pattern.startswith('_'):
|
|
error("Names with a leading underscore are reserved")
|
|
|
|
if not ln:
|
|
error("No pattern list given for '%s %s'" % (name, pattern) )
|
|
|
|
ln=' '.join(ln)
|
|
gbl.tnames[name].definePattern(pattern, ln)
|
|
|
|
|
|
def trackSequence(name, ln):
|
|
""" Define a sequence for a track.
|
|
|
|
The format for a sequence:
|
|
TrackName Seq1 [Seq2 ... ]
|
|
|
|
Note, that SeqX can be a predefined seq or { seqdef }
|
|
The {} is dynamically interpreted into a def.
|
|
"""
|
|
|
|
if not ln:
|
|
error ("Use: %s Sequence NAME [...]" % name)
|
|
|
|
ln = ' '.join(ln)
|
|
|
|
""" Extract out any {} definitions and assign them to new
|
|
define variables (__1, __99, etc) and melt them
|
|
back into the string.
|
|
"""
|
|
|
|
ids=1
|
|
while 1:
|
|
sp = ln.find("{")
|
|
|
|
if sp<0:
|
|
break
|
|
|
|
ln, s = pextract(ln, "{", "}", 1)
|
|
if not s:
|
|
error("Did not find matching '}' for '{'")
|
|
|
|
pn = "_%s" % ids
|
|
ids+=1
|
|
|
|
trk=name.split('-')[0]
|
|
trackAlloc(trk, 1)
|
|
|
|
gbl.tnames[trk].definePattern(pn, s[0])
|
|
ln = ln[:sp] + ' ' + pn + ' ' + ln[sp:]
|
|
|
|
ln=ln.split()
|
|
|
|
gbl.tnames[name].setSequence(ln)
|
|
|
|
|
|
def trackSeqClear(name, ln):
|
|
""" Clear sequence for specified tracks.
|
|
|
|
Note: "Drum SeqClear" clears all Drum tracks,
|
|
"Drum-3 SeqClear" clears track Drum-3.
|
|
"""
|
|
|
|
if ln:
|
|
error("No args permitted. Use %s SEQCLEAR" % name)
|
|
|
|
for n in gbl.tnames:
|
|
if n.find(name) == 0:
|
|
if gbl.debug:
|
|
print "SeqClear: Track %s cleared." % n
|
|
gbl.tnames[n].clearSequence()
|
|
|
|
|
|
def trackSeqRnd(name, ln):
|
|
""" Set random order for specified track. """
|
|
|
|
if len(ln) != 1:
|
|
error("Use: %s SeqRnd [On, Off]" % name)
|
|
|
|
gbl.tnames[name].setRnd(ln[0].upper())
|
|
|
|
def trackSeqRndWeight(name, ln):
|
|
""" Set rnd weight for track. """
|
|
|
|
if not ln:
|
|
error("Use: %s RndWeight <weight factors>" % name)
|
|
|
|
gbl.tnames[name].setRndWeight(ln)
|
|
|
|
|
|
def trackRestart(name, ln):
|
|
""" Restart track to almost-default condidions. """
|
|
|
|
if ln:
|
|
error ("Use: '%s Resart' with no args", name)
|
|
|
|
gbl.tnames[name].restart()
|
|
|
|
|
|
def trackRiff(name, ln):
|
|
""" Set a riff for a track. """
|
|
|
|
gbl.tnames[name].setRiff(' '.join(ln))
|
|
|
|
|
|
|
|
def deleteTrks(ln):
|
|
""" Delete a track and free the MIDI track. """
|
|
|
|
if not len(ln):
|
|
error("Use Delete Track [...]")
|
|
|
|
for name in ln:
|
|
name=name.upper()
|
|
if name in gbl.tnames:
|
|
tr = gbl.tnames[name]
|
|
else:
|
|
error("Track '%s' does not exist" % name)
|
|
|
|
if tr.channel:
|
|
tr.doMidiClear()
|
|
tr.clearPending()
|
|
|
|
if tr.riff:
|
|
warning("%s has pending RIFF(s)" % name)
|
|
gbl.midiAvail[tr.channel] -= 1
|
|
|
|
# NOTE: Don't try deleting 'tr' since it's just a copy!!
|
|
|
|
del gbl.tnames[name]
|
|
|
|
if not name in gbl.deletedTracks:
|
|
gbl.deletedTracks.append(name)
|
|
|
|
if gbl.debug:
|
|
print "Track '%s' deleted" % name
|
|
|
|
|
|
|
|
#######################################
|
|
# Volume
|
|
|
|
def trackRvolume(name, ln):
|
|
""" Set random volume for specific track. """
|
|
|
|
if not ln:
|
|
error ("Use: %s RVolume N [...]" % name)
|
|
|
|
gbl.tnames[name].setRVolume(ln)
|
|
|
|
def trackSwell(name, ln):
|
|
gbl.tnames[name].setSwell(ln)
|
|
|
|
def trackCresc(name, ln):
|
|
gbl.tnames[name].setCresc(1, ln)
|
|
|
|
def trackDeCresc(name, ln):
|
|
gbl.tnames[name].setCresc(-1, ln)
|
|
|
|
def trackVolume(name, ln):
|
|
""" Set volume for specific track. """
|
|
|
|
if not ln:
|
|
error ("Use: %s Volume DYN [...]" % name)
|
|
|
|
gbl.tnames[name].setVolume(ln)
|
|
|
|
|
|
def trackChannelVol(name, ln):
|
|
""" Set the channel volume for a track."""
|
|
|
|
if len(ln) != 1:
|
|
error("Use: %s ChannelVolume" % name)
|
|
|
|
v=stoi(ln[0], "Expecting integer arg, not %s" % ln[0])
|
|
|
|
if v<0 or v>127:
|
|
error("ChannelVolume must be 0..127")
|
|
|
|
gbl.tnames[name].setChannelVolume(v)
|
|
|
|
|
|
def trackAccent(name, ln):
|
|
""" Set emphasis beats for track."""
|
|
|
|
gbl.tnames[name].setAccent(ln)
|
|
|
|
|
|
#######################################
|
|
# Timing
|
|
|
|
def trackCut(name, ln):
|
|
""" Insert a ALL NOTES OFF at the given offset. """
|
|
|
|
|
|
if not len(ln):
|
|
ln=['0']
|
|
|
|
if len(ln) != 1:
|
|
error("Use: %s Cut Offset" % name)
|
|
|
|
|
|
offset = stof(ln[0], "Cut offset expecting value, (not '%s')" % ln[0])
|
|
|
|
if offset < -gbl.QperBar or offset > gbl.QperBar:
|
|
warning("Cut: %s is a large beat offset" % offset)
|
|
|
|
|
|
|
|
moff = int(gbl.tickOffset + (gbl.BperQ * offset))
|
|
|
|
if moff < 0:
|
|
error("Calculated offset for Cut comes before start of track")
|
|
|
|
""" Insert allnoteoff directly in track. This skips the normal
|
|
queueing in pats because it would never take if at the end
|
|
of a track.
|
|
"""
|
|
|
|
m = gbl.tnames[name].channel
|
|
if m and len(gbl.mtrks[m].miditrk) > 1:
|
|
gbl.mtrks[m].addNoteOff(moff)
|
|
|
|
|
|
if gbl.debug:
|
|
print "%s Cut: Beat %s, Bar %s" % (name, offset, gbl.barNum + 1)
|
|
|
|
|
|
def trackMallet(name, ln):
|
|
""" Set repeating-mallet options for solo/melody track. """
|
|
|
|
if not ln:
|
|
error("Use: %s Mallet <Option=Value> [...]" % name)
|
|
|
|
gbl.tnames[name].setMallet(ln)
|
|
|
|
|
|
def trackRtime(name, ln):
|
|
""" Set random timing for specific track. """
|
|
|
|
if not ln:
|
|
error ("Use: %s RTime N [...]" % name)
|
|
|
|
|
|
gbl.tnames[name].setRTime(ln)
|
|
|
|
|
|
def trackRskip(name, ln):
|
|
""" Set random skip for specific track. """
|
|
|
|
if not ln:
|
|
error ("Use: %s RSkip N [...]" % name)
|
|
|
|
|
|
gbl.tnames[name].setRSkip(ln)
|
|
|
|
|
|
def trackArtic(name, ln):
|
|
""" Set articulation. """
|
|
|
|
if not ln:
|
|
error("Use: %s Articulation N [...]" % name)
|
|
|
|
|
|
gbl.tnames[name].setArtic(ln)
|
|
|
|
|
|
#######################################
|
|
# Chord stuff
|
|
|
|
|
|
def trackCompress(name, ln):
|
|
""" Set (unset) compress for track. """
|
|
|
|
if not ln:
|
|
error("Use: %s Compress <value[s]>" % name)
|
|
|
|
gbl.tnames[name].setCompress(ln)
|
|
|
|
|
|
def trackVoicing(name, ln):
|
|
""" Set Voicing options. Only valid for chord tracks at this time."""
|
|
|
|
if not ln:
|
|
error("Use: %s Voicing <MODE=VALUE> [...]" % name)
|
|
|
|
|
|
gbl.tnames[name].setVoicing(ln)
|
|
|
|
|
|
|
|
def trackDupRoot(name, ln):
|
|
""" Set (unset) the root note duplication. Only applies to chord tracks. """
|
|
|
|
if not ln:
|
|
error("Use: %s DupRoot <value> ..." % name)
|
|
|
|
gbl.tnames[name].setDupRoot(ln)
|
|
|
|
|
|
def trackChordLimit(name, ln):
|
|
""" Set (unset) ChordLimit for track. """
|
|
|
|
if len(ln) != 1:
|
|
error("Use: %s ChordLimit <value>" % name)
|
|
|
|
gbl.tnames[name].setChordLimit(ln[0])
|
|
|
|
def trackRange(name, ln):
|
|
""" Set (unset) Range for track. Only effects arp and scale. """
|
|
|
|
if not ln:
|
|
error("Use: %s Range <value> ... " % name)
|
|
|
|
|
|
gbl.tnames[name].setRange(ln)
|
|
|
|
|
|
def trackInvert(name, ln):
|
|
""" Set invert for track."""
|
|
|
|
if not ln:
|
|
error("Use: %s Invert N [...]" % name)
|
|
|
|
gbl.tnames[name].setInvert(ln)
|
|
|
|
|
|
def trackSpan(name, ln):
|
|
""" Set midi note span for track. """
|
|
|
|
if len(ln) != 2:
|
|
error("Use: %s Start End" % name)
|
|
|
|
start = stoi(ln[0], "Expecting integer for SPAN 1st arg")
|
|
if start <0 or start >127:
|
|
error("Start arg for Span must be 0..127, not %s" % start)
|
|
|
|
end = stoi(ln[1], "Expecting integer for SPAN 2nd arg")
|
|
if end <0 or end >127:
|
|
error("End arg for Span must be 0..127, not %s" % end)
|
|
|
|
if end <= start:
|
|
error("End arg for Span must be greater than start")
|
|
|
|
if end-start < 11:
|
|
error("Span range must be at least 12")
|
|
|
|
gbl.tnames[name].setSpan(start, end)
|
|
|
|
|
|
|
|
def trackOctave(name, ln):
|
|
""" Set octave for specific track. """
|
|
|
|
if not ln:
|
|
error ("Use: %s Octave N [...], (n=0..10)" % name)
|
|
|
|
|
|
gbl.tnames[name].setOctave( ln )
|
|
|
|
|
|
def trackStrum(name, ln):
|
|
""" Set all specified track strum. """
|
|
|
|
if not ln:
|
|
error ("Use: %s Strum N [...]" % name)
|
|
|
|
|
|
gbl.tnames[name].setStrum( ln )
|
|
|
|
|
|
def trackHarmony(name, ln):
|
|
""" Set harmony value. """
|
|
|
|
if not ln:
|
|
error("Use: %s Harmony N [...]" % name)
|
|
|
|
gbl.tnames[name].setHarmony(ln)
|
|
|
|
|
|
def trackHarmonyOnly(name, ln):
|
|
""" Set harmony only for track. """
|
|
|
|
if not ln:
|
|
error("Use: %s HarmonyOnly N [...]" % name)
|
|
|
|
gbl.tnames[name].setHarmonyOnly(ln)
|
|
|
|
def trackHarmonyVolume(name, ln):
|
|
""" Set harmony volume for track."""
|
|
|
|
if not ln:
|
|
error("Use: %s HarmonyVolume N [...]" % name)
|
|
|
|
gbl.tnames[name].setHarmonyVolume(ln)
|
|
|
|
|
|
#######################################
|
|
# MIDI setting
|
|
|
|
|
|
def trackChannel(name, ln):
|
|
""" Set the midi channel for a track."""
|
|
|
|
if not ln:
|
|
error("Use: %s Channel" % name)
|
|
|
|
gbl.tnames[name].setChannel(ln[0])
|
|
|
|
|
|
def trackMdefine(name, ln):
|
|
""" Set a midi seq pattern. Ignore track name."""
|
|
|
|
mdefine(ln)
|
|
|
|
|
|
def trackMidiExt(ln):
|
|
""" Helper for trackMidiSeq() and trackMidiVoice()."""
|
|
|
|
ids=1
|
|
while 1:
|
|
sp = ln.find("{")
|
|
|
|
if sp<0:
|
|
break
|
|
|
|
ln, s = pextract(ln, "{", "}", 1)
|
|
if not s:
|
|
error("Did not find matching '}' for '{'")
|
|
|
|
pn = "_%s" % ids
|
|
ids+=1
|
|
|
|
MMA.mdefine.mdef.set(pn, s[0])
|
|
ln = ln[:sp] + ' ' + pn + ' ' + ln[sp:]
|
|
|
|
return ln.split()
|
|
|
|
|
|
def trackMidiClear(name, ln):
|
|
""" Set MIDI command to send at end of groove. """
|
|
|
|
if not ln:
|
|
error("Use %s MIDIClear Controller Data" % name)
|
|
|
|
|
|
if len(ln) == 1 and ln[0] == '-':
|
|
gbl.tnames[name].setMidiClear( '-' )
|
|
else:
|
|
ln=' '.join(ln)
|
|
if '{' in ln or '}' in ln:
|
|
error("{}s are not permitted in %s MIDIClear command" % name)
|
|
gbl.tnames[name].setMidiClear( trackMidiExt( '{' + ln + '}' ))
|
|
|
|
|
|
def trackMidiSeq(name, ln):
|
|
""" Set reoccurring MIDI command for track. """
|
|
|
|
if not ln:
|
|
error("Use %s MidiSeq Controller Data" % name)
|
|
|
|
if len(ln) == 1 and ln[0]== '-':
|
|
gbl.tnames[name].setMidiSeq('-')
|
|
else:
|
|
gbl.tnames[name].setMidiSeq( trackMidiExt(' '.join(ln) ))
|
|
|
|
|
|
def trackMidiVoice(name, ln):
|
|
""" Set single shot MIDI command for track. """
|
|
|
|
if not ln:
|
|
error("Use %s MidiVoice Controller Data" % name)
|
|
|
|
if len(ln) == 1 and ln[0] == '-':
|
|
gbl.tnames[name].setMidiVoice( '-' )
|
|
else:
|
|
gbl.tnames[name].setMidiVoice( trackMidiExt(' '.join(ln) ))
|
|
|
|
|
|
def trackChShare(name, ln):
|
|
""" Set MIDI channel sharing."""
|
|
|
|
if len(ln) !=1:
|
|
error("Use: %s ChShare TrackName" % name)
|
|
|
|
gbl.tnames[name].setChShare(ln[0])
|
|
|
|
|
|
def trackVoice(name, ln):
|
|
""" Set voice for specific track. """
|
|
|
|
if not ln:
|
|
error ("Use: %s Voice NN [...]" % name)
|
|
|
|
|
|
gbl.tnames[name].setVoice(ln)
|
|
|
|
|
|
def trackPan(name, ln):
|
|
""" Set the Midi Pan value for a track."""
|
|
|
|
if len(ln)==1 or len(ln)==3:
|
|
gbl.tnames[name].setPan(ln)
|
|
else:
|
|
error("Use %s MidiPAN [Value] OR [Initvalue DestValue Beats]." % name)
|
|
|
|
def trackOff(name, ln):
|
|
""" Turn a track off """
|
|
|
|
if ln:
|
|
error("Use: %s OFF with no paramater" % name)
|
|
|
|
gbl.tnames[name].setOff()
|
|
|
|
|
|
def trackOn(name, ln):
|
|
""" Turn a track on """
|
|
|
|
if ln:
|
|
error("Use: %s ON with no paramater" % name)
|
|
|
|
gbl.tnames[name].setOn()
|
|
|
|
|
|
def trackMidiName(name,ln):
|
|
""" Set channel track name."""
|
|
|
|
if not ln:
|
|
error("Use: %s TrackName" % name)
|
|
|
|
gbl.tnames[name].setTname(ln[0])
|
|
|
|
|
|
def trackTone(name, ln):
|
|
""" Set the tone (note). Only valid in drum tracks."""
|
|
|
|
if not ln:
|
|
error("Use: %s Tone N [...]" % name)
|
|
|
|
gbl.tnames[name].setTone(ln)
|
|
|
|
|
|
def trackGlis(name, ln):
|
|
""" Enable/disable portamento. """
|
|
|
|
if len(ln) != 1:
|
|
error("Use: %s Portamento NN, off=0, 1..127==on" % name)
|
|
|
|
gbl.tnames[name].setGlis(ln[0])
|
|
|
|
def trackForceOut(name, ln):
|
|
""" Force output of voice settings. """
|
|
|
|
if len(ln):
|
|
error("Use %s ForceOut (no options)" % name)
|
|
|
|
gbl.tnames[name].setForceOut()
|
|
|
|
|
|
#######################################
|
|
# Misc
|
|
|
|
def trackDrumType(name, ln):
|
|
""" Set a melody or solo track to be a drum solo track."""
|
|
|
|
tr = gbl.tnames[name]
|
|
if tr.vtype not in ('SOLO', 'MELODY'):
|
|
error ("Only Solo and Melody tracks can be to DrumType, not '%s'" % name)
|
|
if ln:
|
|
error("No parmeters permitted for DrumType command")
|
|
|
|
tr.setDrumType()
|
|
|
|
|
|
def trackDirection(name, ln):
|
|
""" Set scale/arp direction. """
|
|
|
|
if not ln:
|
|
error("Use: %s Direction OPT" % name)
|
|
|
|
|
|
gbl.tnames[name].setDirection(ln)
|
|
|
|
|
|
def trackScaletype(name, ln):
|
|
""" Set the scale type. """
|
|
|
|
if not ln:
|
|
error("Use: %s ScaleType OPT" % name)
|
|
|
|
gbl.tnames[name].setScaletype(ln)
|
|
|
|
|
|
def trackCopy(name, ln):
|
|
""" Copy setting in 'ln' to 'name'. """
|
|
|
|
if len(ln) != 1:
|
|
error("Use: %s Copy ExistingTrack" % name)
|
|
|
|
gbl.tnames[name].copySettings(ln[0].upper())
|
|
|
|
|
|
def trackUnify(name, ln):
|
|
""" Set UNIFY for track."""
|
|
|
|
if not len(ln):
|
|
error("Use %s UNIFY 1 [...]" % name)
|
|
|
|
gbl.tnames[name].setUnify(ln)
|
|
|
|
|
|
|
|
""" =================================================================
|
|
|
|
Command jump tables. These need to be at the end of this module
|
|
to avoid undefined name errors. The tables are only used in
|
|
the parse() function.
|
|
|
|
The first table is for the simple commands ... those which DO NOT
|
|
have a leading trackname. The second table is for commands which
|
|
require a leading track name.
|
|
|
|
The alphabetic order is NOT needed, just convenient.
|
|
|
|
"""
|
|
|
|
simpleFuncs={
|
|
'ADJUSTVOLUME': MMA.volume.adjvolume,
|
|
'ALLGROOVES': MMA.grooves.allgrooves,
|
|
'ALLTRACKS': allTracks,
|
|
'AUTHOR': MMA.docs.docAuthor,
|
|
'AUTOSOLOTRACKS': MMA.patSolo.setAutoSolo,
|
|
'BEATADJUST': beatAdjust,
|
|
'CHANNELPREF': setChPref,
|
|
'CHORDADJUST': MMA.chords.chordAdjust,
|
|
'COMMENT': comment,
|
|
'CRESC': MMA.volume.setCresc,
|
|
'CUT': cut,
|
|
'DEBUG': setDebug,
|
|
'DEC': macros.vardec,
|
|
'DECRESC': MMA.volume.setDecresc,
|
|
'DEFALIAS': MMA.grooves.grooveAlias,
|
|
'DEFCHORD': MMA.chords.defChord,
|
|
'DEFGROOVE': MMA.grooves.grooveDefine,
|
|
'DELETE': deleteTrks,
|
|
'DOC': MMA.docs.docNote,
|
|
'DOCVAR': MMA.docs.docVars,
|
|
'DRUMVOLTR': MMA.translate.drumVolTable.set,
|
|
'ELSE': ifelse,
|
|
'ENDIF': ifend,
|
|
'ENDMSET': endmset,
|
|
'ENDREPEAT': repeatend,
|
|
'EOF': eof,
|
|
'FERMATA': fermata,
|
|
'GOTO': goto,
|
|
'GROOVE': MMA.grooves.groove,
|
|
'GROOVECLEAR': MMA.grooves.grooveClear,
|
|
'IF': macros.varIF,
|
|
'IFEND': ifend,
|
|
'INC': macros.varinc,
|
|
'INCLUDE': include,
|
|
'KEYSIG': MMA.patSolo.keySig.set,
|
|
'LABEL': comment,
|
|
'LYRIC': lyric.option,
|
|
'MIDIDEF': mdefine,
|
|
'MIDI': rawMidi,
|
|
'MIDIFILE': setMidiFileType,
|
|
'MIDIINC': MMA.midiIn.midiinc,
|
|
'MIDIMARK': midiMarker,
|
|
'MIDISPLIT': MMA.midi.setSplitChannels,
|
|
'MMAEND': mmaend,
|
|
'MMASTART': mmastart,
|
|
'MSET': macros.msetvar,
|
|
'MSETEND': endmset,
|
|
'NEWSET': macros.newsetvar,
|
|
'PATCH': MMA.patch.patch,
|
|
'PRINT': lnPrint,
|
|
'PRINTACTIVE': printActive,
|
|
'PRINTCHORD': MMA.chords.printChord,
|
|
'REPEAT': repeat,
|
|
'REPEATEND': repeatend,
|
|
'REPEATENDING': repeatending,
|
|
'RESTART': restart,
|
|
'RNDSEED': rndseed,
|
|
'RNDSET': macros.rndvar,
|
|
'SEQ': seq,
|
|
'SEQCLEAR': seqClear,
|
|
'SEQRND': MMA.seqrnd.setSeqRnd,
|
|
'SEQRNDWEIGHT': MMA.seqrnd.setSeqRndWeight,
|
|
'SEQSIZE': seqsize,
|
|
'SET': macros.setvar,
|
|
'SETAUTOLIBPATH': setAutoPath,
|
|
'SETINCPATH': setIncPath,
|
|
'SETLIBPATH': setLibPath,
|
|
'SETMIDIPLAYER': setMidiPlayer,
|
|
'SETOUTPATH': setOutPath,
|
|
'SHOWVARS': macros.showvars,
|
|
'STACKVALUE': macros.stackValue,
|
|
'SWELL': MMA.volume.setSwell,
|
|
'SWINGMODE': MMA.notelen.swingMode,
|
|
'SYNCHRONIZE': synchronize,
|
|
'TEMPO': tempo,
|
|
'TIME': setTime,
|
|
'TIMESIG': setTimeSig,
|
|
'TONETR': MMA.translate.dtable.set,
|
|
'UNSET': macros.unsetvar,
|
|
'USE': usefile,
|
|
'VARCLEAR': macros.clear,
|
|
'VEXPAND': macros.vexpand,
|
|
'VOICEVOLTR': MMA.translate.voiceVolTable.set,
|
|
'VOICETR': MMA.translate.vtable.set,
|
|
'VOLUME': MMA.volume.setVolume,
|
|
'TRANSPOSE': transpose
|
|
}
|
|
|
|
|
|
trackFuncs={
|
|
'ACCENT': trackAccent,
|
|
'ARTICULATE': trackArtic,
|
|
'CHANNEL': trackChannel,
|
|
'MIDIVOLUME': trackChannelVol,
|
|
'CHSHARE': trackChShare,
|
|
'COMPRESS': trackCompress,
|
|
'COPY': trackCopy,
|
|
'CRESC': trackCresc,
|
|
'CUT': trackCut,
|
|
'DECRESC': trackDeCresc,
|
|
'DIRECTION': trackDirection,
|
|
'DRUMTYPE': trackDrumType,
|
|
'DUPROOT': trackDupRoot,
|
|
'FORCEOUT': trackForceOut,
|
|
'GROOVE': MMA.grooves.trackGroove,
|
|
'HARMONY': trackHarmony,
|
|
'HARMONYONLY': trackHarmonyOnly,
|
|
'HARMONYVOLUME': trackHarmonyVolume,
|
|
'INVERT': trackInvert,
|
|
'LIMIT': trackChordLimit,
|
|
'MALLET': trackMallet,
|
|
'MIDIDEF': trackMdefine,
|
|
'MIDIGLIS': trackGlis,
|
|
'MIDICLEAR': trackMidiClear,
|
|
'MIDIPAN': trackPan,
|
|
'MIDIGLIS': trackGlis,
|
|
'MIDISEQ': trackMidiSeq,
|
|
'MIDITNAME': trackMidiName,
|
|
'MIDIVOICE': trackMidiVoice,
|
|
'OCTAVE': trackOctave,
|
|
'OFF': trackOff,
|
|
'ON': trackOn,
|
|
'RANGE': trackRange,
|
|
'RESTART': trackRestart,
|
|
'RIFF': trackRiff,
|
|
'RSKIP': trackRskip,
|
|
'RTIME': trackRtime,
|
|
'RVOLUME': trackRvolume,
|
|
'SCALETYPE': trackScaletype,
|
|
'SEQCLEAR': trackSeqClear,
|
|
'SEQRND': trackSeqRnd,
|
|
'SEQUENCE': trackSequence,
|
|
'SEQRNDWEIGHT': trackSeqRndWeight,
|
|
'SWELL': trackSwell,
|
|
'NOTESPAN': trackSpan,
|
|
'STRUM': trackStrum,
|
|
'TONE': trackTone,
|
|
'UNIFY': trackUnify,
|
|
'VOICE': trackVoice,
|
|
'VOICING': trackVoicing,
|
|
'VOLUME': trackVolume,
|
|
'DEFINE': trackDefPattern
|
|
}
|
|
|
|
|