You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

173 lines
6.5KB

  1. //**************************************************************************************
  2. //Quad QuadVCA module for VCV Rack by Alfredo Santamaria - AS - https://github.com/AScustomWorks/AS
  3. //
  4. //Code adapted from the Fundamentals plugins by Andrew Belt http://www.vcvrack.com
  5. //**************************************************************************************
  6. #include "AS.hpp"
  7. struct QuadVCA : Module {
  8. enum ParamIds {
  9. GAIN1_PARAM,
  10. GAIN2_PARAM,
  11. GAIN3_PARAM,
  12. GAIN4_PARAM,
  13. MODE1_PARAM,
  14. MODE2_PARAM,
  15. MODE3_PARAM,
  16. MODE4_PARAM,
  17. NUM_PARAMS
  18. };
  19. enum InputIds {
  20. GAIN1_CV_INPUT,
  21. IN1_INPUT,
  22. GAIN2_CV_INPUT,
  23. IN2_INPUT,
  24. GAIN3_CV_INPUT,
  25. IN3_INPUT,
  26. GAIN4_CV_INPUT,
  27. IN4_INPUT,
  28. NUM_INPUTS
  29. };
  30. enum OutputIds {
  31. OUT1_OUTPUT,
  32. OUT2_OUTPUT,
  33. OUT3_OUTPUT,
  34. OUT4_OUTPUT,
  35. NUM_OUTPUTS
  36. };
  37. enum LightIds {
  38. GAIN1_LIGHT,
  39. GAIN2_LIGHT,
  40. GAIN3_LIGHT,
  41. GAIN4_LIGHT,
  42. NUM_LIGHTS
  43. };
  44. float v1= 0.0f;
  45. float v2= 0.0f;
  46. float v3= 0.0f;
  47. float v4= 0.0f;
  48. const float expBase = 50.0f;
  49. QuadVCA() : Module(NUM_PARAMS, NUM_INPUTS, NUM_OUTPUTS, NUM_LIGHTS) {}
  50. void step() override;
  51. };
  52. void QuadVCA::step() {
  53. //QuadVCA 1
  54. float out = 0.0;
  55. v1 = inputs[IN1_INPUT].value * params[GAIN1_PARAM].value;
  56. if(inputs[GAIN1_CV_INPUT].active){
  57. if(params[MODE1_PARAM].value==1){
  58. v1 *= clamp(inputs[GAIN1_CV_INPUT].value / 10.0f, 0.0f, 1.0f);
  59. }else{
  60. v1 *= rescale(powf(expBase, clamp(inputs[GAIN1_CV_INPUT].value / 10.0f, 0.0f, 1.0f)), 1.0f, expBase, 0.0f, 1.0f);
  61. }
  62. }
  63. out+=v1;
  64. lights[GAIN1_LIGHT].setBrightnessSmooth(fmaxf(0.0f, out / 5.0f));
  65. if (outputs[OUT1_OUTPUT].active) {
  66. outputs[OUT1_OUTPUT].value = out;
  67. out = 0.0f;
  68. }
  69. //QuadVCA 2
  70. v2 = inputs[IN2_INPUT].value * params[GAIN2_PARAM].value;
  71. if(inputs[GAIN2_CV_INPUT].active){
  72. if(params[MODE2_PARAM].value){
  73. v2 *= clamp(inputs[GAIN2_CV_INPUT].value / 10.0f, 0.0f, 1.0f);
  74. }else{
  75. v2 *= rescale(powf(expBase, clamp(inputs[GAIN2_CV_INPUT].value / 10.0f, 0.0f, 1.0f)), 1.0f, expBase, 0.0f, 1.0f);
  76. }
  77. }
  78. out+=v2;
  79. lights[GAIN2_LIGHT].setBrightnessSmooth(fmaxf(0.0f, out / 5.0f));
  80. if (outputs[OUT2_OUTPUT].active) {
  81. outputs[OUT2_OUTPUT].value = out;
  82. out = 0.0f;
  83. }
  84. //QuadVCA 3
  85. v3 = inputs[IN3_INPUT].value * params[GAIN3_PARAM].value;
  86. if(inputs[GAIN3_CV_INPUT].active){
  87. if(params[MODE3_PARAM].value){
  88. v3 *= clamp(inputs[GAIN3_CV_INPUT].value / 10.0f, 0.0f, 1.0f);
  89. }else{
  90. v3 *= rescale(powf(expBase, clamp(inputs[GAIN3_CV_INPUT].value / 10.0f, 0.0f, 1.0f)), 1.0f, expBase, 0.0f, 1.0f);
  91. }
  92. }
  93. out+=v3;
  94. lights[GAIN3_LIGHT].setBrightnessSmooth(fmaxf(0.0f, out / 5.0f));
  95. if (outputs[OUT3_OUTPUT].active) {
  96. outputs[OUT3_OUTPUT].value = out;
  97. out = 0.0f;
  98. }
  99. //QuadVCA 4
  100. v4 = inputs[IN4_INPUT].value * params[GAIN4_PARAM].value;
  101. if(inputs[GAIN4_CV_INPUT].active){
  102. if(params[MODE4_PARAM].value){
  103. v4 *= clamp(inputs[GAIN4_CV_INPUT].value / 10.0f, 0.0f, 1.0f);
  104. }else{
  105. v4 *= rescale(powf(expBase, clamp(inputs[GAIN4_CV_INPUT].value / 10.0f, 0.0f, 1.0f)), 1.0f, expBase, 0.0f, 1.0f);
  106. }
  107. }
  108. out+=v4;
  109. lights[GAIN4_LIGHT].setBrightnessSmooth(fmaxf(0.0f, out / 5.0f));
  110. if (outputs[OUT4_OUTPUT].active) {
  111. outputs[OUT4_OUTPUT].value = out;
  112. out = 0.0f;
  113. }
  114. }
  115. struct QuadVCAWidget : ModuleWidget
  116. {
  117. QuadVCAWidget(QuadVCA *module);
  118. };
  119. QuadVCAWidget::QuadVCAWidget(QuadVCA *module) : ModuleWidget(module) {
  120. setPanel(SVG::load(assetPlugin(plugin, "res/QuadVCA.svg")));
  121. //SCREWS
  122. addChild(Widget::create<as_HexScrew>(Vec(RACK_GRID_WIDTH, 0)));
  123. addChild(Widget::create<as_HexScrew>(Vec(box.size.x - 2 * RACK_GRID_WIDTH, 0)));
  124. addChild(Widget::create<as_HexScrew>(Vec(RACK_GRID_WIDTH, RACK_GRID_HEIGHT - RACK_GRID_WIDTH)));
  125. addChild(Widget::create<as_HexScrew>(Vec(box.size.x - 2 * RACK_GRID_WIDTH, RACK_GRID_HEIGHT - RACK_GRID_WIDTH)));
  126. static const float posX[4] = {13,39,65,91};
  127. //SLIDERS
  128. addParam(ParamWidget::create<as_SlidePot>(Vec(posX[0]-3, 70), module, QuadVCA::GAIN1_PARAM, 0.0f, 1.0f, 0.5f));
  129. addParam(ParamWidget::create<as_SlidePot>(Vec(posX[1]-3, 70), module, QuadVCA::GAIN2_PARAM, 0.0f, 1.0f, 0.5f));
  130. addParam(ParamWidget::create<as_SlidePot>(Vec(posX[2]-3, 70), module, QuadVCA::GAIN3_PARAM, 0.0f, 1.0f, 0.5f));
  131. addParam(ParamWidget::create<as_SlidePot>(Vec(posX[3]-3, 70), module, QuadVCA::GAIN4_PARAM, 0.0f, 1.0f, 0.5f));
  132. //MODE SWITCHES
  133. addParam(ParamWidget::create<as_CKSS>(Vec(posX[0], 190), module, QuadVCA::MODE1_PARAM, 0.0f, 1.0f, 1.0f));
  134. addParam(ParamWidget::create<as_CKSS>(Vec(posX[1], 190), module, QuadVCA::MODE2_PARAM, 0.0f, 1.0f, 1.0f));
  135. addParam(ParamWidget::create<as_CKSS>(Vec(posX[2], 190), module, QuadVCA::MODE3_PARAM, 0.0f, 1.0f, 1.0f));
  136. addParam(ParamWidget::create<as_CKSS>(Vec(posX[3], 190), module, QuadVCA::MODE4_PARAM, 0.0f, 1.0f, 1.0f));
  137. //CV INPUTS
  138. addInput(Port::create<as_PJ301MPort>(Vec(posX[0]-4, 217), Port::INPUT, module, QuadVCA::GAIN1_CV_INPUT));
  139. addInput(Port::create<as_PJ301MPort>(Vec(posX[1]-4, 217), Port::INPUT, module, QuadVCA::GAIN2_CV_INPUT));
  140. addInput(Port::create<as_PJ301MPort>(Vec(posX[2]-4, 217), Port::INPUT, module, QuadVCA::GAIN3_CV_INPUT));
  141. addInput(Port::create<as_PJ301MPort>(Vec(posX[3]-4, 217), Port::INPUT, module, QuadVCA::GAIN4_CV_INPUT));
  142. //INPUTS
  143. addInput(Port::create<as_PJ301MPort>(Vec(posX[0]-4, 260), Port::INPUT, module, QuadVCA::IN1_INPUT));
  144. addInput(Port::create<as_PJ301MPort>(Vec(posX[1]-4, 260), Port::INPUT, module, QuadVCA::IN2_INPUT));
  145. addInput(Port::create<as_PJ301MPort>(Vec(posX[2]-4, 260), Port::INPUT, module, QuadVCA::IN3_INPUT));
  146. addInput(Port::create<as_PJ301MPort>(Vec(posX[3]-4, 260), Port::INPUT, module, QuadVCA::IN4_INPUT));
  147. //LEDS
  148. addChild(ModuleLightWidget::create<SmallLight<RedLight>>(Vec(posX[0]+5, 288), module, QuadVCA::GAIN1_LIGHT));//294
  149. addChild(ModuleLightWidget::create<SmallLight<RedLight>>(Vec(posX[1]+5, 288), module, QuadVCA::GAIN2_LIGHT));
  150. addChild(ModuleLightWidget::create<SmallLight<RedLight>>(Vec(posX[2]+5, 288), module, QuadVCA::GAIN3_LIGHT));
  151. addChild(ModuleLightWidget::create<SmallLight<RedLight>>(Vec(posX[3]+5, 288), module, QuadVCA::GAIN4_LIGHT));
  152. //OUTPUTS
  153. addOutput(Port::create<as_PJ301MPort>(Vec(posX[0]-4, 310), Port::OUTPUT, module, QuadVCA::OUT1_OUTPUT));
  154. addOutput(Port::create<as_PJ301MPort>(Vec(posX[1]-4, 310), Port::OUTPUT, module, QuadVCA::OUT2_OUTPUT));
  155. addOutput(Port::create<as_PJ301MPort>(Vec(posX[2]-4, 310), Port::OUTPUT, module, QuadVCA::OUT3_OUTPUT));
  156. addOutput(Port::create<as_PJ301MPort>(Vec(posX[3]-4, 310), Port::OUTPUT, module, QuadVCA::OUT4_OUTPUT));
  157. }
  158. RACK_PLUGIN_MODEL_INIT(AS, QuadVCA) {
  159. Model *modelQuadVCA = Model::create<QuadVCA, QuadVCAWidget>("AS", "QuadVCA", "Quad VCA/Mixer", AMPLIFIER_TAG, MIXER_TAG);
  160. return modelQuadVCA;
  161. }