|
|
@@ -4,30 +4,39 @@ |
|
|
|
struct VCMixer : Module { |
|
|
|
enum ParamIds { |
|
|
|
MIX_LVL_PARAM, |
|
|
|
ENUMS(LVL_PARAM, 4), |
|
|
|
ENUMS(LVL_PARAMS, 4), |
|
|
|
NUM_PARAMS |
|
|
|
}; |
|
|
|
enum InputIds { |
|
|
|
MIX_CV_INPUT, |
|
|
|
ENUMS(CH_INPUT, 4), |
|
|
|
ENUMS(CV_INPUT, 4), |
|
|
|
ENUMS(CH_INPUTS, 4), |
|
|
|
ENUMS(CV_INPUTS, 4), |
|
|
|
NUM_INPUTS |
|
|
|
}; |
|
|
|
enum OutputIds { |
|
|
|
MIX_OUTPUT, |
|
|
|
ENUMS(CH_OUTPUT, 4), |
|
|
|
ENUMS(CH_OUTPUTS, 4), |
|
|
|
NUM_OUTPUTS |
|
|
|
}; |
|
|
|
enum LightIds { |
|
|
|
ENUMS(LVL_LIGHTS, 4), |
|
|
|
NUM_LIGHTS |
|
|
|
}; |
|
|
|
|
|
|
|
dsp::VuMeter2 chMeters[4]; |
|
|
|
dsp::ClockDivider lightDivider; |
|
|
|
|
|
|
|
VCMixer() { |
|
|
|
config(NUM_PARAMS, NUM_INPUTS, NUM_OUTPUTS); |
|
|
|
config(NUM_PARAMS, NUM_INPUTS, NUM_OUTPUTS, NUM_LIGHTS); |
|
|
|
// x^1 scaling up to 6 dB |
|
|
|
configParam(MIX_LVL_PARAM, 0.0, 2.0, 1.0, "Master level", " dB", -10, 20); |
|
|
|
// x^2 scaling up to 6 dB |
|
|
|
configParam(LVL_PARAM + 0, 0.0, M_SQRT2, 1.0, "Ch 1 level", " dB", -10, 40); |
|
|
|
configParam(LVL_PARAM + 1, 0.0, M_SQRT2, 1.0, "Ch 2 level", " dB", -10, 40); |
|
|
|
configParam(LVL_PARAM + 2, 0.0, M_SQRT2, 1.0, "Ch 3 level", " dB", -10, 40); |
|
|
|
configParam(LVL_PARAM + 3, 0.0, M_SQRT2, 1.0, "Ch 4 level", " dB", -10, 40); |
|
|
|
configParam(LVL_PARAMS + 0, 0.0, M_SQRT2, 1.0, "Ch 1 level", " dB", -10, 40); |
|
|
|
configParam(LVL_PARAMS + 1, 0.0, M_SQRT2, 1.0, "Ch 2 level", " dB", -10, 40); |
|
|
|
configParam(LVL_PARAMS + 2, 0.0, M_SQRT2, 1.0, "Ch 3 level", " dB", -10, 40); |
|
|
|
configParam(LVL_PARAMS + 3, 0.0, M_SQRT2, 1.0, "Ch 4 level", " dB", -10, 40); |
|
|
|
|
|
|
|
lightDivider.setDivision(512); |
|
|
|
} |
|
|
|
|
|
|
|
void process(const ProcessArgs &args) override { |
|
|
@@ -38,24 +47,25 @@ struct VCMixer : Module { |
|
|
|
for (int i = 0; i < 4; i++) { |
|
|
|
int channels = 1; |
|
|
|
float in[16] = {}; |
|
|
|
float sum = 0.f; |
|
|
|
|
|
|
|
if (inputs[CH_INPUT + i].isConnected()) { |
|
|
|
channels = inputs[CH_INPUT + i].getChannels(); |
|
|
|
if (inputs[CH_INPUTS + i].isConnected()) { |
|
|
|
channels = inputs[CH_INPUTS + i].getChannels(); |
|
|
|
maxChannels = std::max(maxChannels, channels); |
|
|
|
|
|
|
|
// Get input |
|
|
|
inputs[CH_INPUT + i].readVoltages(in); |
|
|
|
inputs[CH_INPUTS + i].readVoltages(in); |
|
|
|
|
|
|
|
// Apply fader gain |
|
|
|
float gain = std::pow(params[LVL_PARAM + i].getValue(), 2.f); |
|
|
|
float gain = std::pow(params[LVL_PARAMS + i].getValue(), 2.f); |
|
|
|
for (int c = 0; c < channels; c++) { |
|
|
|
in[c] *= gain; |
|
|
|
} |
|
|
|
|
|
|
|
// Apply CV gain |
|
|
|
if (inputs[CV_INPUT + i].isConnected()) { |
|
|
|
if (inputs[CV_INPUTS + i].isConnected()) { |
|
|
|
for (int c = 0; c < channels; c++) { |
|
|
|
float cv = clamp(inputs[CV_INPUT + i].getPolyVoltage(c) / 10.f, 0.f, 1.f); |
|
|
|
float cv = clamp(inputs[CV_INPUTS + i].getPolyVoltage(c) / 10.f, 0.f, 1.f); |
|
|
|
in[c] *= cv; |
|
|
|
} |
|
|
|
} |
|
|
@@ -64,12 +74,19 @@ struct VCMixer : Module { |
|
|
|
for (int c = 0; c < channels; c++) { |
|
|
|
mix[c] += in[c]; |
|
|
|
} |
|
|
|
|
|
|
|
// Sum channel for VU meter |
|
|
|
for (int c = 0; c < channels; c++) { |
|
|
|
sum += in[c]; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
chMeters[i].process(args.sampleTime, sum / 5.f); |
|
|
|
|
|
|
|
// Set channel output |
|
|
|
if (outputs[CH_OUTPUT + i].isConnected()) { |
|
|
|
outputs[CH_OUTPUT + i].setChannels(channels); |
|
|
|
outputs[CH_OUTPUT + i].writeVoltages(in); |
|
|
|
if (outputs[CH_OUTPUTS + i].isConnected()) { |
|
|
|
outputs[CH_OUTPUTS + i].setChannels(channels); |
|
|
|
outputs[CH_OUTPUTS + i].writeVoltages(in); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
@@ -93,6 +110,14 @@ struct VCMixer : Module { |
|
|
|
outputs[MIX_OUTPUT].setChannels(maxChannels); |
|
|
|
outputs[MIX_OUTPUT].writeVoltages(mix); |
|
|
|
} |
|
|
|
|
|
|
|
// VU lights |
|
|
|
if (lightDivider.process()) { |
|
|
|
for (int i = 0; i < 4; i++) { |
|
|
|
float b = chMeters[i].getBrightness(-24.f, 0.f); |
|
|
|
lights[LVL_LIGHTS + i].setBrightness(b); |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
}; |
|
|
|
|
|
|
@@ -108,27 +133,27 @@ struct VCMixerWidget : ModuleWidget { |
|
|
|
addChild(createWidget<ScrewSilver>(Vec(box.size.x - 2 * RACK_GRID_WIDTH, RACK_GRID_HEIGHT - RACK_GRID_WIDTH))); |
|
|
|
|
|
|
|
addParam(createParam<RoundLargeBlackKnob>(mm2px(Vec(19.049999, 21.161154)), module, VCMixer::MIX_LVL_PARAM)); |
|
|
|
addParam(createParam<LEDSliderGreen>(mm2px(Vec(5.8993969, 44.33149).plus(Vec(-2, 0))), module, VCMixer::LVL_PARAM + 0)); |
|
|
|
addParam(createParam<LEDSliderGreen>(mm2px(Vec(17.899343, 44.331486).plus(Vec(-2, 0))), module, VCMixer::LVL_PARAM + 1)); |
|
|
|
addParam(createParam<LEDSliderGreen>(mm2px(Vec(29.899292, 44.331486).plus(Vec(-2, 0))), module, VCMixer::LVL_PARAM + 2)); |
|
|
|
addParam(createParam<LEDSliderGreen>(mm2px(Vec(41.90065, 44.331486).plus(Vec(-2, 0))), module, VCMixer::LVL_PARAM + 3)); |
|
|
|
addParam(createLightParam<LEDLightSlider<GreenLight>>(mm2px(Vec(5.8993969, 44.33149).plus(Vec(-2, 0))), module, VCMixer::LVL_PARAMS + 0, VCMixer::LVL_LIGHTS + 0)); |
|
|
|
addParam(createLightParam<LEDLightSlider<GreenLight>>(mm2px(Vec(17.899343, 44.331486).plus(Vec(-2, 0))), module, VCMixer::LVL_PARAMS + 1, VCMixer::LVL_LIGHTS + 1)); |
|
|
|
addParam(createLightParam<LEDLightSlider<GreenLight>>(mm2px(Vec(29.899292, 44.331486).plus(Vec(-2, 0))), module, VCMixer::LVL_PARAMS + 2, VCMixer::LVL_LIGHTS + 2)); |
|
|
|
addParam(createLightParam<LEDLightSlider<GreenLight>>(mm2px(Vec(41.90065, 44.331486).plus(Vec(-2, 0))), module, VCMixer::LVL_PARAMS + 3, VCMixer::LVL_LIGHTS + 3)); |
|
|
|
|
|
|
|
// Use old interleaved order for backward compatibility with <0.6 |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(3.2935331, 23.404598)), module, VCMixer::MIX_CV_INPUT)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(3.2935331, 78.531639)), module, VCMixer::CH_INPUT + 0)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(3.2935331, 93.531586)), module, VCMixer::CV_INPUT + 0)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(15.29348, 78.531639)), module, VCMixer::CH_INPUT + 1)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(15.29348, 93.531586)), module, VCMixer::CV_INPUT + 1)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(27.293465, 78.531639)), module, VCMixer::CH_INPUT + 2)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(27.293465, 93.531586)), module, VCMixer::CV_INPUT + 2)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(39.293411, 78.531639)), module, VCMixer::CH_INPUT + 3)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(39.293411, 93.531586)), module, VCMixer::CV_INPUT + 3)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(3.2935331, 78.531639)), module, VCMixer::CH_INPUTS + 0)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(3.2935331, 93.531586)), module, VCMixer::CV_INPUTS + 0)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(15.29348, 78.531639)), module, VCMixer::CH_INPUTS + 1)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(15.29348, 93.531586)), module, VCMixer::CV_INPUTS + 1)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(27.293465, 78.531639)), module, VCMixer::CH_INPUTS + 2)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(27.293465, 93.531586)), module, VCMixer::CV_INPUTS + 2)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(39.293411, 78.531639)), module, VCMixer::CH_INPUTS + 3)); |
|
|
|
addInput(createInput<PJ301MPort>(mm2px(Vec(39.293411, 93.531586)), module, VCMixer::CV_INPUTS + 3)); |
|
|
|
|
|
|
|
addOutput(createOutput<PJ301MPort>(mm2px(Vec(39.293411, 23.4046)), module, VCMixer::MIX_OUTPUT)); |
|
|
|
addOutput(createOutput<PJ301MPort>(mm2px(Vec(3.2935331, 108.53153)), module, VCMixer::CH_OUTPUT + 0)); |
|
|
|
addOutput(createOutput<PJ301MPort>(mm2px(Vec(15.29348, 108.53153)), module, VCMixer::CH_OUTPUT + 1)); |
|
|
|
addOutput(createOutput<PJ301MPort>(mm2px(Vec(27.293465, 108.53153)), module, VCMixer::CH_OUTPUT + 2)); |
|
|
|
addOutput(createOutput<PJ301MPort>(mm2px(Vec(39.293411, 108.53153)), module, VCMixer::CH_OUTPUT + 3)); |
|
|
|
addOutput(createOutput<PJ301MPort>(mm2px(Vec(3.2935331, 108.53153)), module, VCMixer::CH_OUTPUTS + 0)); |
|
|
|
addOutput(createOutput<PJ301MPort>(mm2px(Vec(15.29348, 108.53153)), module, VCMixer::CH_OUTPUTS + 1)); |
|
|
|
addOutput(createOutput<PJ301MPort>(mm2px(Vec(27.293465, 108.53153)), module, VCMixer::CH_OUTPUTS + 2)); |
|
|
|
addOutput(createOutput<PJ301MPort>(mm2px(Vec(39.293411, 108.53153)), module, VCMixer::CH_OUTPUTS + 3)); |
|
|
|
} |
|
|
|
}; |
|
|
|
|
|
|
|