package vafusion.inst;
import jm.audio.io.*;
import jm.audio.Instrument;
import jm.audio.synth.*;
import jm.music.data.Note;
import jm.audio.AudioObject;
public final class SubtractiveSampleInst extends jm.audio.Instrument{
//----------------------------------------------
// Attributes
//----------------------------------------------
/** the name of the sample file */
private String fileName;
/** How many channels is the sound file we are using */
private int numOfChannels;
/** the base frequency of the sample file to be read in */
private double baseFreq;
/** should we play the wholeFile or just what we need for note duration */
private boolean wholeFile;
/** The envelope sustain value */
private double sustain;
/** low pass filter cutoff */
private int cutoff;
/** Envelope attributes */
private int attack, decay, release;
/** Depth of filter modulation (0.0 - 1.0)*/
private double modAmount = 0.9;
/** Speed of filter modulation in Hertz*/
private double modRate = 0.05;
/** The volume of the squarewave sub oscillator */
private double subAmp = 0.0;
//----------------------------------------------
// Constructor
//----------------------------------------------
/**
* Constructor
*/
public SubtractiveSampleInst(String fileName){
this(fileName, 440.00, 500);
}
public SubtractiveSampleInst(String fileName, double baseFreq){
this(fileName, baseFreq, 500);
}
public SubtractiveSampleInst(String fileName, double baseFreq, int cutoff){
this(fileName, baseFreq, cutoff, 2, 50, 0.4, 200);
}
public SubtractiveSampleInst(String fileName, double baseFreq, int cutoff, int attack, int decay, double sustain, int release){
this.fileName = fileName;
this.baseFreq = baseFreq;
this.cutoff = cutoff;
this.attack = attack;
this.decay = decay;
this.release = release;
this.sustain = sustain;
}
//----------------------------------------------
// Methods
//----------------------------------------------
public void setModAmount(double val) {
this.modAmount = val;
}
public void setModDepth(double val) {
this.modAmount = val;
}
public void setModRate(double val) {
this.modRate = val;
}
public void setWholeFile(boolean val) {
this.wholeFile = val;
}
public void setSubAmp(double val) {
this.subAmp = val;
}
/**
* Create the Audio Chain for this Instrument
* and assign the primary Audio Object(s). The
* primary audio object(s) are the one or more
* objects which head up the chain(s)
*/
@Override
public void createChain(){
//define the chain
SampleIn sin = new SampleIn(this, fileName);
sin.setWholeFile(wholeFile);
ReSample reSample = new ReSample(sin, this.baseFreq);
// modulate filter cutoff
Oscillator sineMod = new Oscillator(this, Oscillator.SINE_WAVE, sin.getSampleRate(),
sin.getChannels(), Oscillator.FREQUENCY, (float)modRate);
sineMod.setAmp((float) this.modAmount * this.cutoff);
// sub
Oscillator subOsc = new Oscillator(this, Oscillator.SQUARE_WAVE, sin.getSampleRate(),
sin.getChannels());
subOsc.setFrqRatio(0.5f);
subOsc.setAmp((float)subAmp);
Add adder = new Add(new AudioObject[] {reSample, subOsc});
// filter
Filter filt = new Filter(new AudioObject[] {adder, sineMod}, this.cutoff, Filter.LOW_PASS);
ADSR env = new ADSR(filt, attack, decay, sustain, release);
Volume vol = new Volume(env);
StereoPan span = new StereoPan(vol);
SampleOut sout = new SampleOut(span);
}
}