brief moments held

sound waves of varying pitch and amplitude

Log in to post a comment.

synth.def(
  class {
    process() {
    
    // instrument
    function vox(mx, rt, t) {
    let y = 0.,
      p = mx * [fmod(t, 4.) > 3. ? 10.97 : 8.26, fmod(2. * t, 4.) > 2. ? 9.28 : 12.25, 10.85, 12.25].ring(.5 * t),
      an = 0.,
      en = fract(t),
      em = [550, 739, 550, 550, 739, 550, 550, 418].ring(.5 * t),
    v = pow(3.07406, [.377, .41, .377, .355].ring(.25 * t) * log(12.5 + fract(t)));
    an = exp(3. * en * exp(log2(1. - en) / (sin(.3 * rt) * .65 + .65) * .1));
    y = v * em * p + an;
    y = exp(-.5 * fract(t)) * sin(8. * y) * ((cos(.05 * rt) * .25 + .25) + cos((trunc(.5 * t) % 6 > 5 ? 8 : 4) * y));
    y *= env_ad(en, .7, .95)
    y = y + tanh(softkn(25 * y, (2 + fract(.1 * rt))) * .51);
    return y;
    }
    
    this.phase += 1;
    let time = this.phase / 44100;
    // print msg to the gui
    debug.log('',['|', '/ ', '-\t', '  \\'].ring(5. * time) + "Prepare yourself" + "\nfor something good that happens\nany second now.");
    
    // initialization 
    let dtm = fmod(time,16), tm = time; // + .001*sin(time);
    let o = [0.0];
    let sn = 0, sl = 0;
    let l = 0., m = 0, n = 0;
    
    for (let it = 0; it < 4; ++it) {
      //chords           
      tm += (exp2(.05 / (it + 1) % 4) - .85);
      tm /= 2;
      m = .7 * vox(1, .5 * time, 3 + fmod(4 * tm, [2, 15][trunc(tm) % 2])) * exp(-15. * fmod(tm, 1));
      m += 2. * vox(1, .25 * time, 3. + fmod(tm, 6)) * exp(-8. * fmod(tm, 1));
      n = 4. * vox(1, time, 1. + fmod(tm, [5, 7][trunc(tm) % 2])) * exp(-10. * fmod(tm, 1));
      n += 1. * vox(1, 2 * time, 3 + fmod(2 * tm, 17)) * exp(-5. * fmod(tm, 1));
      sn += m + n;
      // background voice
      dtm += (exp2(.125 / (it + 1) % 4) - .85);
      let dd =  [2, 4][trunc(.5 * dtm) % 2] *  .25 * dtm;
      l = cos(2. * vox(.5, .25 * time,  [2, 4][trunc(.5 * dtm) % 2] *  .25 * dtm )-1.57)
      l = abs(2. * l  * max(0, 1-fract(dd )) * min(1, 20.*fract( dd )));
      l *= max(0., 1.-time/40) *  max(0, 1.-fmod(time/4, 1)) * 1.5 * fmod(time/4, 1);
      sl += l;
    }
    // premix
    let tmp = [(sl + sn)*4., (sl + sn)*4.];
    //sl += .3 * sn;
    
    // reverb
    let rv = [0, 0];
    let y = Array(8).fill(0);

    const samp = this.phase;
    // prime power delaytimes
    //const dt = [1331, 2197, 4913, 6859, 2401, 3125, 2187, 2048];
    const dt = [2401, 3125, 2187,2048, 6859, 4913,2197, 1331];

    // hadamard matrix
    let hmt = [1, 1, 1, 1, -1, 1, -1, -1, 1, -1, -1, 1, 1, -1, 1, -1];
    hmt = hmt.map((n, idx) => SQRT1_2 * hmt[idx]);
    
// input 
const minput = {
  rev_dly_scR: 1,
  rev_dly_scL: .95,
  lop_cf: 4200,
  lop_q: 1.1,
  rev_gn1 :.45,
  rev_gn2 :.95
};
    // rescale delay
    let scl = minput.rev_dly_scL, scr = minput.rev_dly_scR;

    for (let i = 0; i < 8; ++i) {

        const it4 = i % 4, ip4 = i % 4 + 4;
        // gain exp higher more reverb
        const g1 = exp2(-minput.rev_gn1 / (1 + i) % 8);
        const g2 = minput.rev_gn2;
        const sclR = round(scr * dt[it4]), sclL = round(scl * dt[ip4]);
        
        // summer and fix lowh-end accumulation
        y[it4] += this.dcblock(this.zn[it4][samp % sclR], 0.994, it4);
        y[ip4] += this.dcblock(this.zn[ip4][samp % sclL], 0.994, ip4);
        // correlate 
        y[it4] *= hmt[it4];
        y[ip4] *= hmt[it4];
        // lowpass
        tmp[0] = g2 * this.e2poleLp(tmp[0], minput.lop_cf, minput.lop_q, it4);
        tmp[1] = g2 * this.e2poleLp(tmp[1], minput.lop_cf, minput.lop_q, ip4);
        
        //shuffle
        //tmp[0] *=hmt[ip4];
        //tmp[1] *=hmt[it4];
        
        // buffer last frame
        this.zn[it4][samp % sclR] = g1 * y[it4] + tmp[0];
        this.zn[ip4][samp % sclL] = g1 * y[ip4] + tmp[1];
        
        // shuffle 
        rv[0] = (y[0] + y[5] + y[6] + y[2]) / 4;
        rv[1] = (y[4] + y[1] + y[3] + y[7]) / 4;

    }

    o[0] =  (rv[0] + l + tmp[0]);
    o[1] =  (rv[1] + l + tmp[0]);
    return o;
    }
   // buffers, eq states 
   constructor() {
      this.phase = 0;
      this.zn = Array(8).fill().map(() => Array(6859).fill(0));
      this.e2pSt = Array(8).fill().map(() => Array(4).fill(0));
      this.dcbSt = Array(8).fill().map(() => Array(2).fill(0));
    }
    // highpass
    dcblock(xin, at, instnc) {
      let state = this.dcbSt[instnc];
      let y = xin - state[0] + at * state[1];
      state[0] = xin;
      state[1] = y;
      return y;
    }
    // lowpass direct form II
    e2poleLp(val, cf, q, instnc) {
      let state = this.e2pSt[instnc];
      let y = 0,
        w0 = tau * cf / 44100,
        a0 = sin(w0) / (2.0 * q),
        ac = 1.0 + a0,
        bc = ((1.0 - cos(w0)) / 2) / ac,
        a1 = (-2.0 * cos(w0)) / ac,
        a2 = (1.0 - a0) / ac;
      y = bc * val + bc * state[0] 
        + bc * state[1] 
        - a1 * state[2] - a2 * state[3];
      state[1] = state[0];
      state[0] = val;
      state[3] = state[2];
      state[2] = y;
      return y;
    }
 }).play(0, {
  duration: 1e8
});
// utilities 
const{PI,exp,log,sin,cos,floor,trunc,pow,log2,min,max,abs,tanh,round,SQRT1_2}=Math;
const tau=PI*2.0;
const fract=x=>x-floor(x);
const fmod=(x,y)=>x-floor(x/y)*y;
const exp2=x=>exp(x*log(2.0));
const env_ad=(t,a,d)=>min(t/max(1e-5,a),exp(-(t-a)*5.436563/d));
const msat=(x,a)=>0.5*(abs(x+a)-abs(x-a));
const softkn=(x,kn)=>x/(kn*abs(x)+1);
const smoothstep=(e0,e1,value)=>{const x=max(0,min(1,(value-e0)/(e1-e0)));return x*x*(3-2*x);};