BogaudioModules

BogaudioModules for VCV Rack
Log | Files | Refs | README | LICENSE

commit 0e1bbf7abbe12cd0d0695af22fbb6e8292724176
parent dab2980abbc97929762da3e6aad98cbcba2f74e6
Author: Matt Demanett <matt@demanett.net>
Date:   Sun, 29 Apr 2018 14:23:23 -0400

RM: ring mod module.

Diffstat:
Ares-src/RM-src.svg | 134+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Ares/RM.svg | 0
Asrc/RM.cpp | 73+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Asrc/RM.hpp | 44++++++++++++++++++++++++++++++++++++++++++++
Msrc/bogaudio.cpp | 6++++++
Msrc/dsp/signal.cpp | 20++++++++++++++++++++
Msrc/dsp/signal.hpp | 13+++++++++++++
7 files changed, 290 insertions(+), 0 deletions(-)

diff --git a/res-src/RM-src.svg b/res-src/RM-src.svg @@ -0,0 +1,134 @@ +<svg + version="1.1" + xmlns="http://www.w3.org/2000/svg" + xmlns:xlink="http://www.w3.org/1999/xlink" + width="45" + height="380" + viewBox="0 0 45 380" +> + <style> + text { + fill: #333; + font-family: 'Roboto', sans-serif; + font-weight: bold; + } + text.title { + font-family: 'Comfortaa', sans-serif; + font-weight: normal; + } + text.brand { + font-family: 'Audiowide', sans-serif; + font-weight: bold; + } + </style> + + <defs> + <symbol id="knob" viewBox="0 0 45px 45px"> + <g transform="translate(22.5 22.5)"> + <polyline points="-5,0 5,0" stroke-width="1" stroke="#00f" /> + <polyline points="0,-5 0,5" stroke-width="1" stroke="#00f" /> + <circle cx="0" cy="0" r="14.5" stroke-width="1" stroke="#00f" fill="none" /> + </g> + </symbol> + + <symbol id="knobguide-depth" viewBox="0 0 45px 45px"> + <g transform="translate(22.5 22.5)"> + <text font-size="5.0pt" transform="rotate(-240) translate(20 0) rotate(240) translate(-2 2)">0</text> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(-210) translate(17 0)" /> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(-180) translate(17 0)" /> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(-150) translate(17 0)" /> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(-120) translate(17 0)" /> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(-90) translate(17 0)" /> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(-60) translate(17 0)" /> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(-30) translate(17 0)" /> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(0) translate(17 0)" /> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(30) translate(17 0)" /> + <polyline points="0,0 3.5,0" stroke-width="1" stroke="#333" transform="rotate(60) translate(17 0)" /> + </g> + </symbol> + + <symbol id="knobguide-mix" viewBox="0 0 45px 45px"> + <g transform="translate(22.5 22.5)"> + <text font-size="5.0pt" transform="rotate(-240) translate(20 0) rotate(240) translate(-2 2)">D</text> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(-210) translate(17 0)" /> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(-180) translate(17 0)" /> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(-150) translate(17 0)" /> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(-120) translate(17 0)" /> + <polyline points="0,0 3.5,0" stroke-width="1" stroke="#333" transform="rotate(-90) translate(17 0)" /> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(-60) translate(17 0)" /> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(-30) translate(17 0)" /> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(0) translate(17 0)" /> + <polyline points="0,0 2.5,0" stroke-width="0.3" stroke="#333" transform="rotate(30) translate(17 0)" /> + <text font-size="5.0pt" transform="rotate(60) translate(20 0) rotate(-60) translate(-2 2)">W</text> + </g> + </symbol> + + <symbol id="input" viewBox="0 0 24px 24px"> + <g transform="translate(12 12)"> + <circle cx="0" cy="0" r="5" stroke-width="1" stroke="#0f0" fill="#0f0" /> + <circle cx="0" cy="0" r="10.5" stroke-width="3" stroke="#0f0" fill="none" /> + </g> + </symbol> + + <symbol id="output" viewBox="0 0 24px 24px"> + <g transform="translate(12 12)"> + <circle cx="0" cy="0" r="5" stroke-width="1" stroke="#f00" fill="#f00" /> + <circle cx="0" cy="0" r="10.5" stroke-width="3" stroke="#f00" fill="none" /> + </g> + </symbol> + </defs> + + <rect width="100%" height="100%" fill="#ddd" /> + <polyline points="1,1 44,1 44,379 1,379 1,1" stroke="#e4e4e4" stroke-width="0.5" fill="none" /> + <polyline points="0.5,0.5 44.5,0.5 44.5,379.5 0.5,379.5 0.5,0.5" stroke="#ebebeb" stroke-width="0.8" fill="none" /> + <polyline points="0,0 45,0 45,380 0,380 0,0" stroke="#f2f2f2" stroke-width="1" fill="none" /> + + <g transform="rotate(-90) translate(-376 13)"> + <text class="title" font-size="7pt" letter-spacing="2.5px">RM</text> + <g transform="translate(0 12)"> + <text class="brand" font-size="7pt" letter-spacing="2px">BGA</text> + <rect width="3.0" height="3" fill="#ddd" transform="translate(11.5 -5)" /> + </g> + </g> + + <g transform="translate(0 25)"> + <text font-size="6pt" letter-spacing="2px" transform="translate(5.5 0)">DEPTH</text> + <use id="MODULATOR_DEPTH_PARAM" xlink:href="#knob" transform="translate(0 3)" /> + <use xlink:href="#knobguide-depth" transform="scale(1) translate(0 3)" /> + <g transform="translate(5.5 49)"> + <rect width="34" height="39" rx="5" fill="#fafafa" /> + <use id="MODULATOR_DEPTH_INPUT" xlink:href="#input" transform="translate(5 3)" /> + <text font-size="5pt" letter-spacing="2px" transform="translate(11 35)">CV</text> + </g> + </g> + + <g transform="translate(0 131)"> + <text font-size="6pt" letter-spacing="1px" transform="translate(2 0)">DRY/WET</text> + <use id="MIX_PARAM" xlink:href="#knob" transform="translate(0 3)" /> + <use xlink:href="#knobguide-mix" transform="scale(1) translate(0 3)" /> + <g transform="translate(5.5 49)"> + <rect width="34" height="38" rx="5" fill="#fafafa" /> + <use id="MIX_INPUT" xlink:href="#input" transform="translate(5 3)" /> + <text font-size="5pt" letter-spacing="2px" transform="translate(11 35)">CV</text> + </g> + </g> + + <g transform="translate(0 230)"> + <g transform="translate(5.5 0)"> + <rect width="34" height="10" fill="#fafafa" transform="translate(0 63)" /> + <rect width="34" height="70" rx="5" fill="#fafafa" /> + <use id="MODULATOR_INPUT" xlink:href="#input" transform="translate(5 3)" /> + <text font-size="5pt" letter-spacing="2px" transform="translate(7.5 35)">MOD</text> + <use id="CARRIER_INPUT" xlink:href="#input" transform="translate(5 38)" /> + <text font-size="5pt" letter-spacing="2px" transform="translate(8.5 70)">CAR</text> + </g> + <g transform="translate(5.5 76)"> + <rect width="34" height="10" fill="#bbb" transform="translate(0 -3)" /> + <rect width="34" height="35" rx="5" fill="#bbb" /> + <use id="OUT_OUTPUT" xlink:href="#output" transform="translate(5 0)" /> + <text font-size="5pt" letter-spacing="2px" transform="translate(8.3 32)">OUT</text> + </g> + </g> + + <!-- <polyline points="0,0 45,0" stroke-width="0.5" stroke="#0f0" transform="translate(0 73)" /> --> +</svg> diff --git a/res/RM.svg b/res/RM.svg Binary files differ. diff --git a/src/RM.cpp b/src/RM.cpp @@ -0,0 +1,73 @@ + +#include "RM.hpp" + +void RM::step() { + if (!(outputs[OUT_OUTPUT].active && (inputs[MODULATOR_INPUT].active || inputs[CARRIER_INPUT].active))) { + return; + } + if (inputs[MODULATOR_INPUT].active != inputs[CARRIER_INPUT].active) { + if (inputs[MODULATOR_INPUT].active) { + outputs[OUT_OUTPUT].value = inputs[MODULATOR_INPUT].value; + } + else { + outputs[OUT_OUTPUT].value = inputs[CARRIER_INPUT].value; + } + return; + } + + float modulator = inputs[MODULATOR_INPUT].value; + modulator *= params[MODULATOR_DEPTH_PARAM].value; + if (inputs[MODULATOR_DEPTH_INPUT].active) { + modulator *= clamp(inputs[MODULATOR_DEPTH_INPUT].value / 10.0f, 0.0f, 1.0f); + } + + float carrier = inputs[CARRIER_INPUT].value; + + float mix = params[MIX_PARAM].value; + if (inputs[MIX_INPUT].active) { + mix *= clamp(inputs[MIX_INPUT].value / 5.0f, -1.0f, 1.0f); + } + _mix.setMix(mix); + + outputs[OUT_OUTPUT].value = _mix.next(carrier, modulator * carrier * 0.2f); +} + +struct RMWidget : ModuleWidget { + RMWidget(RM* module) : ModuleWidget(module) { + box.size = Vec(RACK_GRID_WIDTH * 3, RACK_GRID_HEIGHT); + + { + SVGPanel *panel = new SVGPanel(); + panel->box.size = box.size; + panel->setBackground(SVG::load(assetPlugin(plugin, "res/RM.svg"))); + addChild(panel); + } + + addChild(Widget::create<ScrewSilver>(Vec(0, 0))); + addChild(Widget::create<ScrewSilver>(Vec(box.size.x - 15, 365))); + + // generated by svg_widgets.rb + auto modulatorDepthParamPosition = Vec(7.5, 35.5); + auto mixParamPosition = Vec(7.5, 141.5); + + auto modulatorDepthInputPosition = Vec(10.5, 77.0); + auto mixInputPosition = Vec(10.5, 183.0); + auto modulatorInputPosition = Vec(10.5, 233.0); + auto carrierInputPosition = Vec(10.5, 268.0); + + auto outOutputPosition = Vec(10.5, 306.0); + // end generated by svg_widgets.rb + + addParam(ParamWidget::create<Knob29>(modulatorDepthParamPosition, module, RM::MODULATOR_DEPTH_PARAM, 0.0, 1.0, 1.0)); + addParam(ParamWidget::create<Knob29>(mixParamPosition, module, RM::MIX_PARAM, -1.0, 1.0, 1.0)); + + addInput(Port::create<Port24>(modulatorInputPosition, Port::INPUT, module, RM::MODULATOR_INPUT)); + addInput(Port::create<Port24>(carrierInputPosition, Port::INPUT, module, RM::CARRIER_INPUT)); + addInput(Port::create<Port24>(modulatorDepthInputPosition, Port::INPUT, module, RM::MODULATOR_DEPTH_INPUT)); + addInput(Port::create<Port24>(mixInputPosition, Port::INPUT, module, RM::MIX_INPUT)); + + addOutput(Port::create<Port24>(outOutputPosition, Port::OUTPUT, module, RM::OUT_OUTPUT)); + } +}; + +Model* modelRM = Model::create<RM, RMWidget>("Bogaudio", "Bogaudio-RM", "RM", RING_MODULATOR_TAG, UTILITY_TAG); diff --git a/src/RM.hpp b/src/RM.hpp @@ -0,0 +1,44 @@ +#pragma once + +#include "bogaudio.hpp" +#include "dsp/signal.hpp" + +using namespace bogaudio::dsp; + +extern Model* modelRM; + +namespace bogaudio { + +struct RM : Module { + enum ParamsIds { + MODULATOR_DEPTH_PARAM, + MIX_PARAM, + NUM_PARAMS + }; + + enum InputsIds { + MODULATOR_INPUT, + CARRIER_INPUT, + MODULATOR_DEPTH_INPUT, + MIX_INPUT, + NUM_INPUTS + }; + + enum OutputsIds { + OUT_OUTPUT, + NUM_OUTPUTS + }; + + enum LightsIds { + NUM_LIGHTS + }; + + CrossFader _mix; + + RM() : Module(NUM_PARAMS, NUM_INPUTS, NUM_OUTPUTS, NUM_LIGHTS) { + } + + virtual void step() override; +}; + +} // namespace bogaudio diff --git a/src/bogaudio.cpp b/src/bogaudio.cpp @@ -15,6 +15,8 @@ #include "Analyzer.hpp" +#include "RM.hpp" + #include "ADSR.hpp" #include "Bool.hpp" #include "Detune.hpp" @@ -60,6 +62,10 @@ void init(rack::Plugin *p) { p->addModel(modelAnalyzer); #ifdef EXPERIMENTAL + p->addModel(modelRM); +#endif + +#ifdef EXPERIMENTAL p->addModel(modelADSR); p->addModel(modelBool); #endif diff --git a/src/dsp/signal.cpp b/src/dsp/signal.cpp @@ -70,3 +70,23 @@ float SlewLimiter::next(float sample) { --_steps; return _current; } + + +void CrossFader::setMix(float mix) { + assert(mix >= -1.0f && mix <= 1.0f); + if (_mix != mix) { + _mix = mix; + if (_mix < 0.0f) { + _aMix = 1.0f; + _bMix = 1.0f + _mix; + } + else { + _aMix = 1.0f - _mix; + _bMix = 1.0f; + } + } +} + +float CrossFader::next(float a, float b) { + return _aMix * a + _bMix * b; +} diff --git a/src/dsp/signal.hpp b/src/dsp/signal.hpp @@ -45,5 +45,18 @@ struct SlewLimiter { float next(float sample); }; +struct CrossFader { + float _mix = 2.0f; + float _aMix; + float _bMix; + + CrossFader() { + setMix(0.0f); + } + + void setMix(float mix); // -1 to 1, 0 for equal output of both inputs. + float next(float a, float b); +}; + } // namespace dsp } // namespace bogaudio