VocalEasel/mma/MMA/docs.py

461 lines
14 KiB
Python
Raw Permalink Normal View History

2006-11-10 08:07:56 +00:00
# docs.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
2007-04-29 06:47:40 +00:00
Bob van der Poel <bob@mellowood.ca>
2006-11-10 08:07:56 +00:00
"""
import os
import time
2009-05-17 22:34:44 +00:00
import MMA.midiC
import MMA.grooves
2006-11-10 08:07:56 +00:00
import gbl
from MMA.common import *
2009-05-17 22:34:44 +00:00
2006-11-10 08:07:56 +00:00
def docDrumNames(order):
2007-04-29 06:47:40 +00:00
""" Print LaTex table of drum names. """
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
notenames = ['E\\flat', 'E', 'F', 'G\\flat', 'G', 'A\\flat',
'A', 'B\\flat', 'B', 'C', 'D\\flat', 'D'] * 5
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
n=zip( MMA.midiC.drumNames, range(27,len(MMA.midiC.drumNames)+27), notenames )
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
if order == "a":
for a,v,m in sorted(n):
print "\\insline{%s} {%s$^{%s}$}" % (a, v, m )
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
else:
for a,v,m in n:
print "\\insline{%s} {%s$^{%s}$}" % (v, a, m)
2006-11-10 08:07:56 +00:00
def docCtrlNames(order):
2007-04-29 06:47:40 +00:00
""" Print LaTex table of MIDI controller names. """
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
n=zip( MMA.midiC.ctrlNames, range(len(MMA.midiC.ctrlNames)) )
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
if order == "a":
for a,v in sorted(n):
print "\\insline{%s} {%02x}" % (a, v)
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
else:
for a,v in n:
print "\\insline{%02x} {%s}" % (v, a)
2006-11-10 08:07:56 +00:00
def docInstNames(order):
2007-04-29 06:47:40 +00:00
""" Print LaTex table of instrument names. """
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
n=zip( MMA.midiC.voiceNames, range(len(MMA.midiC.voiceNames)) )
if order == "a":
for a,v in sorted(n):
a=a.replace('&', '\&')
print "\\insline{%s} {%s}" % (a, v)
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
else:
for a,v in n:
a=a.replace('&', '\&')
print "\\insline{%s} {%s}" % (v, a)
2006-11-10 08:07:56 +00:00
""" Whenever MMA encounters a DOC command, or if it defines
2009-05-17 22:34:44 +00:00
a groove with DEFGROOVE it calls the docAdd() function.
2006-11-10 08:07:56 +00:00
2009-05-17 22:34:44 +00:00
The saved docs are printed to stdout with the docDump() command.
This is called whenever parse() encounters an EOF.
2006-11-10 08:07:56 +00:00
2009-05-17 22:34:44 +00:00
Both routines are ignored if the -Dx command line option has
not been set.
2006-11-10 08:07:56 +00:00
2009-05-17 22:34:44 +00:00
Storage is done is in the following arrays.
2006-11-10 08:07:56 +00:00
"""
2009-05-17 22:34:44 +00:00
fname = ''
author = ""
notes = ""
defs = []
variables = []
2006-11-10 08:07:56 +00:00
def docAuthor(ln):
2007-04-29 06:47:40 +00:00
global author
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
author = ' '.join(ln)
2006-11-10 08:07:56 +00:00
def docNote(ln):
2007-04-29 06:47:40 +00:00
""" Add a doc line. """
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
global fname, notes
2006-11-10 08:07:56 +00:00
2009-05-17 22:34:44 +00:00
if not gbl.createDocs or not ln:
2007-04-29 06:47:40 +00:00
return
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
# Grab the arg and data, save it
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
fname = os.path.basename(gbl.inpath.fname)
if notes:
notes += ' '
notes += ' '.join(ln)
def docVars(ln):
""" Add a VARIABLE line (docs vars used in lib file)."""
global fname, variables
2009-05-17 22:34:44 +00:00
if not gbl.createDocs or not ln:
2007-04-29 06:47:40 +00:00
return
fname = os.path.basename(gbl.inpath.fname)
variables.append([ln[0], ' '.join(ln[1:]) ] )
2006-11-10 08:07:56 +00:00
def docDefine(ln):
2007-04-29 06:47:40 +00:00
""" Save a DEFGROOVE comment string.
2006-11-10 08:07:56 +00:00
2009-05-17 22:34:44 +00:00
Entries are stored as a list. Each item in the list is
complete groove def looking like:
2011-07-26 22:49:39 +00:00
defs[ [ Name, Seqsize, Description, [ [TRACK,INST, [Sequences...] ]...]] ...]
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
"""
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
global defs
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
l = [ ln[0], gbl.seqSize, ' '.join(ln[1:]) ]
for a in sorted(gbl.tnames.keys()):
c=gbl.tnames[a]
if c.sequence and len(c.sequence) != c.sequence.count(None):
if c.vtype=='DRUM':
2011-07-26 22:49:39 +00:00
v= [ MMA.midiC.valueToDrum(x) for x in c.toneList]
2007-04-29 06:47:40 +00:00
else:
2011-07-26 22:49:39 +00:00
v= [ MMA.midiC.valueToInst(x) for x in c.voice]
seq = [ c.formatPattern(c.sequence[x]) for x in range(gbl.seqSize) ]
l.append( [c.name, v, seq ] )
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
defs.append(l)
2006-11-10 08:07:56 +00:00
2009-05-17 22:34:44 +00:00
2006-11-10 08:07:56 +00:00
def docDump():
2007-04-29 06:47:40 +00:00
""" Print the LaTex docs. """
global fname, author, notes, defs, variables
2009-05-17 22:34:44 +00:00
if gbl.createDocs == 1: # latex docs
2007-04-29 06:47:40 +00:00
if notes:
2011-07-26 22:49:39 +00:00
notes = notes.replace("<P>", "\\\\[.5ex]")
notes = notes.replace("<p>", "\\\\[.5ex]")
2007-04-29 06:47:40 +00:00
if fname.endswith(gbl.ext):
fname='.'.join(fname.split('.')[:-1])
print "\\filehead{%s}{%s}" % (totex(fname), totex(notes))
print
if variables:
print " \\variables{"
for l in variables:
print " \\insvar{%s}{%s}" % ( totex(l[0]), totex(l[1]) )
print " }"
print
if defs:
for l in defs:
2009-05-17 22:34:44 +00:00
alias = MMA.grooves.getAlias(l[0])
if alias:
if len(alias)>1:
alias="Aliases: %s" % alias
else:
alias="Alias: %s" % alias
else:
alias=''
print " \\instable{%s}{%s}{%s}{%s}{" % \
(totex(l[0]), totex(l[2]), l[1], alias)
2011-07-26 22:49:39 +00:00
for c,v,s in l[3:]: # we ignore the seqence data here
print " \\insline{%s}{%s}" % (c.title(), totex(v[0]))
2007-04-29 06:47:40 +00:00
print " }"
2009-05-17 22:34:44 +00:00
2011-07-26 22:49:39 +00:00
elif gbl.createDocs == 2: # html docs
2007-04-29 06:47:40 +00:00
if notes:
print '<!-- Auto-Generated by MMA on: %s -->' % time.ctime()
print '<HTML>'
print '<BODY BGCOLOR="#B7DFFF" Text=Black>'
if fname.endswith(gbl.ext):
fname='.'.join(fname.split('.')[:-1])
print "<H1>%s</H1>" % fname.title()
print "<P>%s" % notes
if variables:
print "<P>"
print '<Table Border=3 CELLSPACING=0 CELLPADDING=5 BGColor="#eeeeee" Width="60%">'
print ' <TR><TD>'
print ' <H2> Variables </H2> '
print ' </TD></TR>'
print ' <TR><TD>'
print ' <Table CELLSPACING=0 CELLPADDING=5 BGColor="#eeeeee" Width="100%">'
for l in variables:
print " <TR>"
print " <TD Valign=Top> <B> %s </B> </TD> " % l[0]
print " <TD Valign=Top> %s </TD>" % l[1]
print " </TR>"
print ' </Table>'
print ' </TD></TR>'
print '</Table>'
if defs:
print "<ul>"
for l in defs:
print "<LI><A Href=#%s>%s</a>" % (l[0], l[0])
print "</ul>"
for l in defs:
2011-07-26 22:49:39 +00:00
gg = l[0]
iname = os.path.basename(gbl.infile)
iname, ext = os.path.splitext(iname)
gfile = "%s_%s.html" % (iname, gg.lower() )
print '<!-- GROOVE=%s FILE=%s SRC=%s -->' % (gg.lower(), gfile, gbl.infile)
print '<A Name=%s></a>' % gg
2007-04-29 06:47:40 +00:00
print '<Table Border=3 CELLSPACING=0 CELLPADDING=5 BGColor="#eeeeee" Width="60%">'
print ' <TR><TD>'
2011-07-26 22:49:39 +00:00
print ' <H2> <A Href=%s> %s </a> </H2> ' % (gfile, l[0])
2009-05-17 22:34:44 +00:00
alias=MMA.grooves.getAlias(l[0])
if alias:
if len(alias)>1:
ll="Aliases"
else:
ll="Alias"
print ' <H4> %s: %s </H4>' % (ll, alias)
2011-07-26 22:49:39 +00:00
2007-04-29 06:47:40 +00:00
print ' %s <B>(%s)</B> ' % ( l[2], l[1] )
print ' </TD></TR>'
print ' <TR><TD>'
print ' <Table CELLSPACING=0 CELLPADDING=5 BGColor="#eeeeee" Width="10%">'
2011-07-26 22:49:39 +00:00
for c,v,s in l[3:]:
print " <TR><TD> %s </TD> <TD> %s </TD></TR>" % (c.title(), v[0])
2007-04-29 06:47:40 +00:00
print ' </Table>'
print ' </TD></TR>'
print '</Table>'
print
print '</Body></HTML>'
2009-05-17 22:34:44 +00:00
2011-07-26 22:49:39 +00:00
elif gbl.createDocs == 3: # sequence table
for l in defs:
print "GROOVE", l[0]
print "DESCRIPTION", l[2]
print "SIZE", l[1]
for c,v,s in l[3:]:
print "TRACK", c
print "VOICE", v
print "SEQ", s
elif gbl.createDocs == 99: # creating entry for groove browser
for a,b in (("``", '"'), ("''", '"'), (' ', ' ')):
notes = notes.replace(a,b)
if not notes:
notes = "No header available ... please add DOC to file"
print notes
for l in defs:
print "%s\n %s" % (l[0], l[2])
else:
return
2007-04-29 06:47:40 +00:00
defs = []
variables=[]
notes = ""
author = ""
2006-11-10 08:07:56 +00:00
def totex(s):
2007-04-29 06:47:40 +00:00
""" Parse a string and quote tex stuff.
2006-11-10 08:07:56 +00:00
2009-05-17 22:34:44 +00:00
Also handles proper quotation style.
2007-04-29 06:47:40 +00:00
"""
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
s = s.replace("$", "\$")
s = s.replace("*", "$*$")
s = s.replace("_", "\\_")
#s = s.replace("\\", "\\\\")
s = s.replace("#", "\\#")
s = s.replace("&", "\\&")
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
q="``"
while s.count('"'):
s=s.replace('"', q, 1)
if q=="``":
q="''"
else:
q="``"
2006-11-10 08:07:56 +00:00
2007-04-29 06:47:40 +00:00
return s
2009-05-17 22:34:44 +00:00
2011-07-26 22:49:39 +00:00
def htmlGraph(f):
""" Print (stdout) an html file representing a graph and details of the current groove."""
2009-05-17 22:34:44 +00:00
2011-07-26 22:49:39 +00:00
global fname, author, notes, variables, defs
def getAbsPair(x1, x2):
if abs(x1) == abs(x2):
return "%s" % abs(x1)
else:
return "%s,%s" % (x1,x2)
def docol(lab, data):
if lab == '':
return
if type(data) != type([]):
data = [data]
print " <td width=50%>",
print "%s: " % lab,
if len(set(data)) == 1:
print str(data[0]),
else:
print '&nbsp;&nbsp;'.join([str(x) for x in data]),
print " </td>"
def dorow(a, a1, b, b1):
print "<tr>"
docol(a, a1)
docol(b, b1)
print "</tr>"
2009-05-17 22:34:44 +00:00
defs = []
variables=[]
notes = ""
author = ""
2011-07-26 22:49:39 +00:00
desc = ""
if '/' in f:
u, f = f.rsplit('/', 1)
MMA.parse.usefile([u])
MMA.grooves.groove([f])
groove = MMA.grooves.currentGroove.title()
for a in defs:
if a[0].upper() == groove.upper():
desc = a[2]
print '<!-- Auto-Generated by MMA on: %s -->' % time.ctime()
print '<HTML>'
print '<BODY BGCOLOR="#B7DFFF" Text=Black>'
#if fname.endswith(gbl.ext):
# fname='.'.join(fname.split('.')[:-1])
print "<h2>File: %s</h2>" % fname
print "<h2>Groove: %s</h2>" % groove
print "<p><b>Notes:</b> %s" % notes
print "<p><b>Author:</b> %s" % author
print "<p><b>Description:</b> %s" % desc
print "<p><Table Border=0 Width=75%>"
dorow("SeqSize", gbl.seqSize, "Time (beats per bar)", gbl.QperBar)
print "</Table>"
if variables:
print "<p> <b>Variables</b>"
print '<Table Border=.2em Cellspacing=5 Cellpadding=5 Width="90%">'
for l in variables:
print " <TD Valign=Top> <B> %s </B> </TD> " % l[0]
print " <TD Valign=Top> %s </TD>" % l[1]
print " </TR>"
print '</Table>'
for t in sorted(gbl.tnames.keys()):
trk = gbl.tnames[t]
if not trk.sequence or len(trk.sequence)==trk.sequence.count(None):
continue
print "<p> <b>Track Name: %s</b>" % t.title()
print "<p><Table Border=0 Width=75%>"
if trk.vtype == "DRUM":
v = [ MMA.midiC.valueToDrum(x) for x in trk.toneList]
else:
v = [ MMA.midiC.valueToInst(x) for x in trk.voice]
dorow("Voice/Tones", v, "Articulate", trk.artic)
v = [str(x *100) for x in trk.volume]
if trk.vtype != "DRUM":
oct = [x/12 for x in trk.octave]
dorow('Unify', trk.unify, "Octave", oct )
dorow("Volume", v, "Harmony", trk.harmony)
v1=[str(x * 100) for x in trk.rSkip]
v2=[getAbsPair(x1 * 100, x2 * 100) for x1,x2 in trk.rVolume]
dorow("Rskip", v1, "Rvolume", v2)
v1=[getAbsPair(x1, x2) for x1,x2 in trk.rTime]
if trk.seqRnd:
v2="On"
else:
v2="Off"
dorow("Rtime", v1, "SeqRND", v2)
if trk.vtype == 'CHORD':
vv=trk.voicing.mode
v='Voicing'
else:
vv=v=''
dorow("Strum", trk.strum, v, vv)
print "</Table>"
pointx = 2.5
pointPerS = pointx * gbl.QperBar
pointy = 5
boxx = gbl.seqSize * pointPerS
boxy = pointy
print r'<div style="position:relative;background-color:#99bdf4;'
print r'padding:0; border:.1em solid black; left:5em;'
print r'height:%sem; width:%sem">' % (boxy, boxx)
if gbl.seqSize > 1:
for a in range(1, gbl.seqSize):
print r'<img style="position:absolute; bottom:0; left:%sem;' % (a * pointPerS),
print r'width:.05em; height:%sem; border:.05em solid white"' % pointy,
print r'src="../black.gif">'
for a in range(gbl.seqSize):
s = trk.sequence[a]
if not s: continue
for p in s:
bwidth = p.duration * pointx * (trk.artic[a]/100.)/gbl.BperQ
if bwidth < .1:
bwidth=.1
offset = (p.offset * pointx)/gbl.BperQ + (a * pointPerS)
#offset = (p.offset/gbl.BperQ) * pointx + (a * pointPerS)
if trk.vtype == 'CHORD' or trk.vtype == 'PLECTRUM':
ll = len(p.vol) - p.vol.count(0)
vol = sum(p.vol) / ll
else:
vol = p.vol
height = (vol * pointy) / 127
print r'<img style="position:absolute; border:.02em solid red;bottom:0; ',
print r'left:%sem; width:%sem; height:%sem"' % ( offset, bwidth, height)
print r' src="../blue.gif">'
print r'</div>'
print '</Body></HTML>'
2009-05-17 22:34:44 +00:00
2011-07-26 22:49:39 +00:00
sys.exit(0)