diff options
| author | Graham Northup <grissess@nexusg.org> | 2019-02-05 00:20:44 -0500 | 
|---|---|---|
| committer | Graham Northup <grissess@nexusg.org> | 2019-02-05 00:20:44 -0500 | 
| commit | b82bc24308b3c34caf6d9ef3786c9da4ef5fd3af (patch) | |
| tree | 9708370b837dbfc003ba346108e6787b602287fe | |
| parent | fac3853e5f77fb0eb84c68f58a1adbf804debf73 (diff) | |
lotsa stuff
| -rw-r--r-- | .gitignore | 1 | ||||
| -rw-r--r-- | broadcast.py | 23 | ||||
| -rw-r--r-- | client.py | 25 | ||||
| -rw-r--r-- | deep_note.py | 199 | ||||
| -rw-r--r-- | ivtomid.py | 13 | ||||
| -rw-r--r-- | shiv.py | 10 | 
6 files changed, 267 insertions, 4 deletions
| @@ -7,4 +7,5 @@ client  *.swo  *.pyc  *.mid +*.wav  *~ diff --git a/broadcast.py b/broadcast.py index 0c57935..c4753da 100644 --- a/broadcast.py +++ b/broadcast.py @@ -108,10 +108,11 @@ The specifier consists of a comma-separated list of attribute-colon-value pairs,  GUIS = {}  BASETIME = play_time()  # XXX fixes a race with the GUI +factor = options.factor  def gui_pygame():      # XXX Racy, do this fast -    global tap_func +    global tap_func, BASETIME, factor      key_cond = threading.Condition()      if options.tapper is not None: @@ -152,6 +153,8 @@ def gui_pygame():      clock = pygame.time.Clock()      font = pygame.font.SysFont(pygame.font.get_default_font(), 24) +    status = ('', 0.0) +    DISP_TIME = 4.0      print 'Pygame GUI initialized, running...' @@ -169,6 +172,9 @@ def gui_pygame():          tsurf = font.render('%0.3f' % ((play_time() - BASETIME) / factor,), True, (255, 255, 255), (0, 0, 0))          disp.fill((0, 0, 0), tsurf.get_rect())          disp.blit(tsurf, (0, 0)) +        if time.time() - DISP_TIME < status[1]: +            ssurf = font.render(status[0], True, (0, 255, 0), (0, 0, 0)) +            disp.blit(ssurf, (0, tsurf.get_height()))          pygame.display.flip()          for ev in pygame.event.get(): @@ -179,13 +185,24 @@ def gui_pygame():                      thread.interrupt_main()                      pygame.quit()                      exit() +                elif ev.key == pygame.K_LEFT: +                    BASETIME += 5 +                elif ev.key == pygame.K_RIGHT: +                    BASETIME -= 5 +                elif ev.key in (pygame.K_LEFTBRACKET, pygame.K_RIGHTBRACKET): +                    pt = play_time() +                    rtime = (pt - BASETIME) / factor +                    if ev.key == pygame.K_LEFTBRACKET: +                        factor /= 1.1 +                    elif ev.key == pygame.K_RIGHTBRACKET: +                        factor *= 1.1 +                    BASETIME = pt - rtime * factor +                    status = ('factor: ' + str(factor), time.time())          clock.tick(60)  GUIS['pygame'] = gui_pygame -factor = options.factor -  print 'Factor:', factor  try: @@ -254,6 +254,31 @@ def square_wave(theta):  def noise(theta):      return random.random() * 2 - 1 +@generator('Square generator with polynomial falloff') +class sq_cub(object): +    def __init__(self, mina, degree=1.0/3): +        self.mina = mina +        self.degree = degree +    def __call__(self, theta): +        if theta < math.pi: +            return 1 - (1 - self.mina) * ((theta / math.pi) ** self.degree) +        else: +            return -1 + (1 - self.mina) * (((theta - math.pi) / math.pi) ** self.degree) + +@generator('Impulse-like square') +class impulse(object): +    def __init__(self, dc=0.01): +        self.dc = dc +    def __call__(self, theta): +        if theta < self.dc * math.pi: +            return 1 +        elif theta < math.pi: +            return 0 +        elif theta < (1+self.dc) * math.pi: +            return -1 +        else: +            return 0 +  @generator('File generator', '(<file>[, <bits=8>[, <signed=True>[, <0=linear interp (default), 1=nearest>[, <swapbytes=False>[, <loop=(fraction to loop, 0.0 is all, 1.0 is end, or False to not loop)>[, <loopend=1.0>[, periods=1 (periods in wave file)/freq=None (base frequency)/pitch=None (base MIDI pitch)]]]]]]])')  class file_samp(object):      LINEAR = 0 diff --git a/deep_note.py b/deep_note.py new file mode 100644 index 0000000..5b2183e --- /dev/null +++ b/deep_note.py @@ -0,0 +1,199 @@ +from xml.etree import ElementTree as ET +import optparse, random, math, sys + +def store_tup(opt, optstr, val, parser, nm, idx, filt=float): +    tup = getattr(parser.values, nm) +    setattr(parser.values, nm, tup[:idx] + (filt(val),) + tup[(idx+1):]) + +fund = 440.0 +def to_freq(p): +    return fund * 2.0**((p - 69) / 12.0) + +def to_pitch(f): +    try: +        return 12 * math.log(f / fund, 2) + 69 +    except ValueError: +        print 'bad frequency', f +        raise + +def store_target(opt, optstr, val, parser): +    print 'opt', opt, 'optstr', optstr, 'val', val, 'parser', parser +    if val.startswith('@'): +        val = int(val[1:]) +    else: +        val = int(to_freq(float(val))) +    parser.values.targets.append(( +        parser.values.voices, +        val, +        parser.values.fadeint, +        parser.values.inita, +        parser.values.randt, +        parser.values.randf, +        parser.values.fstep, +        parser.values.fina, +        parser.values.sweept, +        parser.values.fadeoutt +    )) + +parser = optparse.OptionParser() +parser.add_option('-v', '--verbose', dest='verbose', action='store_true', help='Be verbose') +parser.add_option('-V', '--voices', dest='voices', type='int', help='Subsequent -t will have this many voices directed toward it') +parser.add_option('--fadeinlow', action='callback', nargs=1, type='float', callback=store_tup, callback_args=('fadeint', 0), help='Low fade-in time for subsequent -t') +parser.add_option('--fadeinhigh', action='callback', nargs=1, type='float', callback=store_tup, callback_args=('fadeint', 1), help='High fade-in time for subsequent -t') +parser.add_option('--initalow', action='callback', nargs=1, type='float', callback=store_tup, callback_args=('inita', 0), help='Low init amplitude time for subsequent -t') +parser.add_option('--initahigh', action='callback', nargs=1, type='float', callback=store_tup, callback_args=('inita', 1), help='High init amplitude time for subsequent -t') +parser.add_option('--randtlow', action='callback', nargs=1, type='float', callback=store_tup, callback_args=('randt', 0), help='Low random time for subsequent -t') +parser.add_option('--randthigh', action='callback', nargs=1, type='float', callback=store_tup, callback_args=('randf', 1), help='High random time for subsequent -t') +parser.add_option('--randflow', action='callback', nargs=1, type='float', callback=store_tup, callback_args=('randf', 0), help='Low random freq for subsequent -t') +parser.add_option('--randfhigh', action='callback', nargs=1, type='float', callback=store_tup, callback_args=('randt', 1), help='High random freq for subsequent -t') +parser.add_option('--fstep', dest='fstep', type='float', help='Frequency to wander by in random phase') +parser.add_option('--finalow', action='callback', nargs=1, type='float', callback=store_tup, callback_args=('fina', 0), help='Low init amplitude time for subsequent -t') +parser.add_option('--finahigh', action='callback', nargs=1, type='float', callback=store_tup, callback_args=('fina', 1), help='High init amplitude time for subsequent -t') +parser.add_option('--sweeplow', action='callback', nargs=1, type='float', callback=store_tup, callback_args=('sweept', 0), help='Low sweep time for subsequent -t') +parser.add_option('--sweephigh', action='callback', nargs=1, type='float', callback=store_tup, callback_args=('sweept', 1), help='High sweep time for subsequent -t') +parser.add_option('--fadeoutlow', action='callback', nargs=1, type='float', callback=store_tup, callback_args=('fadeoutt', 0), help='Low fade-out time for subsequent -t') +parser.add_option('--fadeouthigh', action='callback', nargs=1, type='float', callback=store_tup, callback_args=('fadeoutt', 1), help='High fade-out time for subsequent -t') +parser.add_option('-t', '--target', action='callback', type='str', callback=store_target, nargs=1, help='Have some voices target this MIDI pitch or @frequency') +parser.add_option('-C', '--clear-targets', action='store_const', const=[], dest='targets', help='Clear all targets (including built in ones) prior') +parser.add_option('-r', '--resolution', dest='resolution', type='float', help='Period of generated samples') +parser.add_option('-c', '--chorus', dest='chorus', type='float', help='Random variation of frequency (factor)') +parser.add_option('--smooth', dest='smooth', type='int', help='Number of random uniform samples to perform to smooth') +parser.add_option('-D', '--duration', dest='duration', type='float', help='Length of the note') +parser.add_option('--slack', dest='slack', type='float', help='Slack added to duration to overcommit clients') +parser.add_option('--fundamental', dest='fundamental', type='float', help='Frequency of the A above middle C (traditionally 440 Hz)') + +fadeint = (4.0, 6.0) +inita = (0.1, 0.2) +randt = (8.5, 9.0) +fina = (0.95, 1.0) +sweept = (17.5, 19.0) +fadeoutt = (1.5, 3.5) +randf = (200, 400) +fstep = 5 +parser.set_defaults( +    resolution = 0.05, +    chorus = 0.02, +    smooth = 5, +    duration = 30.0, +    slack = 0.02, +    fundamental = 430.0, +    voices = 1, +    fadeint = fadeint, +    inita = inita, +    randt = randt, +    fina = fina, +    sweept = sweept, +    fadeoutt = fadeoutt, +    randf = randf, +    fstep = fstep, +    targets = [ +        (2, to_freq(26), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        (2, to_freq(38), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        (3, to_freq(45), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        (3, to_freq(50), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        (3, to_freq(57), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        (3, to_freq(62), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        (3, to_freq(69), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        (3, to_freq(74), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        (3, to_freq(81), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        (3, to_freq(86), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        (3, to_freq(90), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), # Moorer claims this is here, but I'm not sure I believe him +        # Below this is the "shit note" +        #(2, to_freq(34), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        #(2, to_freq(35), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        #(2, to_freq(36), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        #(2, to_freq(42), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        #(3, to_freq(47), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        #(3, to_freq(48), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        #(3, to_freq(54), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        #(3, to_freq(59), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        #(3, to_freq(60), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        #(3, to_freq(66), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        #(3, to_freq(72), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        #(3, to_freq(77), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        #(3, to_freq(83), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        #(3, to_freq(84), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        #(3, to_freq(85), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +        #(3, to_freq(90), fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt), +    ], +) +options, args = parser.parse_args() +fund = options.fundamental + +smooth_rand_buf = [random.random() for i in range(options.smooth)] +def smooth_rand(): +    global smooth_rand_buf +    rv = sum(smooth_rand_buf) / options.smooth +    smooth_rand_buf = [random.random()] + smooth_rand_buf[:-1] +    return rv + +def smooth_uniform(a, b): +    return a + (b-a) * smooth_rand() + +def clamp(u, l, h): +    return min((h, max((l, u)))) + +iv = ET.Element('iv') +ivmeta = ET.SubElement(iv, 'meta') +ivstreams = ET.SubElement(iv, 'streams') + +for nvoice, freq, fadeint, inita, randt, randf, fstep, fina, sweept, fadeoutt in options.targets: +    for vn in range(nvoice): +        ivstream = ET.SubElement(ivstreams, 'stream', type='ns') +        vfadeint = random.uniform(*fadeint) +        vinita = random.uniform(*inita) +        vrandt = random.uniform(*randt) +        vfina = random.uniform(*fina) +        vsweept = random.uniform(*sweept) +        vfadeoutt = random.uniform(*fadeoutt) +        lastf = smooth_uniform(*randf) +        if options.verbose: +            print '-- setup:', freq, vn, 'fadeint', vfadeint, 'inita', vinita, 'randt', vrandt, 'fina', vfina, 'sweept', vsweept, 'fadeoutt', vfadeoutt, 'lastf', lastf +        cntr = 0 +        for ts in range(int(options.duration / options.resolution)): +            tm = ts * options.resolution + +            if tm < vfadeint: +                a = vinita * (tm / vfadeint) +                ap = 'fadein' +            elif tm < vrandt: +                a = vinita +                ap = 'inita' +            elif tm < vsweept: +                u = (clamp((tm - vfadeint - vrandt) / (vsweept - vfadeint - vrandt), 0.0, 1.0)) ** 2 +                a = u * vfina + (1 - u) * vinita +                ap = 'sweep' +            else: +                a = vfina +                ap = 'fina' +            if tm > options.duration - vfadeoutt: +                a = vfina * max((((options.duration - tm) / vfadeoutt), 0)) +                ap = 'fadeout' + +            if tm < vfadeint + vrandt: +                lastf = clamp(lastf + smooth_uniform(-fstep, fstep), randf[0], randf[1]) +                f = lastf +                fp = 'rand' +            elif tm < vsweept: +                u = clamp((tm - vfadeint - vrandt) / (vsweept - vfadeint - vrandt), 0.0, 1.0) +                f = u * freq + (1 - u) * lastf +                fp = 'sweep' +            else: +                f = freq * (1.0 + smooth_uniform(-options.chorus, options.chorus)) +                fp = 'finf' + +            if options.verbose: +                print freq, vn, tm, fp, f, ap, a + +            ivnote = ET.SubElement(ivstream, 'note', +                id = str(cntr), +                pitch = str(to_pitch(f)), +                ampl = str(a), +                time = str(tm), +                dur = str(options.resolution + options.slack), +            ) +            cntr += 1 +            if cntr > 0: +                ivnote.set('par', '0') + +sys.stdout.write(ET.tostring(iv, 'UTF-8')) diff --git a/ivtomid.py b/ivtomid.py new file mode 100644 index 0000000..c439502 --- /dev/null +++ b/ivtomid.py @@ -0,0 +1,13 @@ +''' +itl_chorus -- ITL Chorus Suite +ivtomid -- Convert IV to MIDI + +Revert the conversion of mkiv. +''' + +import xml.etree.ElementTree as ET +import optparse, sys +import midi + +parser = optparse.OptionParser() +parser.add_option(' @@ -4,6 +4,8 @@ import xml.etree.ElementTree as ET  import optparse  import sys  import math +import gzip +import bz2  parser = optparse.OptionParser()  parser.add_option('-n', '--number', dest='number', action='store_true', help='Show number of tracks') @@ -122,7 +124,13 @@ for fname in args:      print      print 'File :', fname      try: -        iv = ET.parse(fname).getroot() +        if fname.endswith('.ivz'): +            ivf = gzip.open(fname, 'rb') +        elif fname.endswith('.ivb'): +            ivf = bz2.BZ2File(fname, 'r') +        else: +            ivf = open(fname, 'rb') +        iv = ET.parse(ivf).getroot()      except Exception:          import traceback          traceback.print_exc() | 
