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.

446 lines
16KB

  1. /******************************************************************************
  2. * Copyright 2017-2018 Valerio Orlandini / Sonus Dept. <sonusdept@gmail.com>
  3. *
  4. * This program is free software: you can redistribute it and/or modify
  5. * it under the terms of the GNU General Public License as published by
  6. * the Free Software Foundation, either version 3 of the License, or
  7. * (at your option) any later version.
  8. *
  9. * This program is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. * GNU General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU General Public License
  15. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  16. *****************************************************************************/
  17. #include "sonusmodular.hpp"
  18. namespace rack_plugin_SonusModular {
  19. struct Luppolo3 : Module
  20. {
  21. enum ParamIds
  22. {
  23. DIRECTION,
  24. ERASE,
  25. CLEAR_M,
  26. CLEAR_S1,
  27. CLEAR_S2,
  28. GAIN_M,
  29. GAIN_S1,
  30. GAIN_S2,
  31. TRIGGER_M,
  32. TRIGGER_S1,
  33. TRIGGER_S2,
  34. OVERDUB_M,
  35. OVERDUB_S1,
  36. OVERDUB_S2,
  37. NUM_PARAMS
  38. };
  39. enum InputIds
  40. {
  41. INPUT_L,
  42. INPUT_R,
  43. TRIGGER_DIR,
  44. CV_TRIGGER_M,
  45. CV_TRIGGER_S1,
  46. CV_TRIGGER_S2,
  47. CV_OVERDUB_M,
  48. CV_OVERDUB_S1,
  49. CV_OVERDUB_S2,
  50. CV_ERASE,
  51. CV_CLEAR_M,
  52. CV_CLEAR_S1,
  53. CV_CLEAR_S2,
  54. NUM_INPUTS
  55. };
  56. enum OutputIds
  57. {
  58. OUTPUT_L,
  59. OUTPUT_R,
  60. OUTPUT_ML,
  61. OUTPUT_MR,
  62. OUTPUT_S1L,
  63. OUTPUT_S1R,
  64. OUTPUT_S2L,
  65. OUTPUT_S2R,
  66. NUM_OUTPUTS
  67. };
  68. enum LightIds
  69. {
  70. REC_LIGHT_M,
  71. PLAY_LIGHT_M,
  72. REC_LIGHT_S1,
  73. PLAY_LIGHT_S1,
  74. REC_LIGHT_S2,
  75. PLAY_LIGHT_S2,
  76. NUM_LIGHTS
  77. };
  78. Luppolo3() : Module(NUM_PARAMS, NUM_INPUTS, NUM_OUTPUTS, NUM_LIGHTS) {}
  79. ~Luppolo3()
  80. {
  81. for (int c = 0; c < 2; c++)
  82. {
  83. master_loop[c].clear();
  84. slave_loop_1[c].clear();
  85. slave_loop_2[c].clear();
  86. }
  87. }
  88. void step() override;
  89. std::deque<float> master_loop[2];
  90. std::deque<float> slave_loop_1[2];
  91. std::deque<float> slave_loop_2[2];
  92. bool is_recording[3] = {false, false, false};
  93. bool master_rec = false;
  94. bool overdubbing[3] = {false, false, false};
  95. int sample = 0;
  96. float trig_last_value[3][2] = {{0.0, 0.0}, {0.0, 0.0}, {0.0, 0.0}};
  97. float overdub_last_value[3][2] = {{0.0, 0.0}, {0.0, 0.0}, {0.0, 0.0}};
  98. float clear_last_value[3][2] = {{0.0, 0.0}, {0.0, 0.0}, {0.0, 0.0}};
  99. float erase_last_value[2] = {0.0, 0.0};
  100. float dir_last_value[2] = {0.0, 0.0};
  101. bool forward = true;
  102. int loop_size = 0;
  103. };
  104. void Luppolo3::step()
  105. {
  106. float in_l = inputs[INPUT_L].value;
  107. float in_r = inputs[INPUT_R].value;
  108. float out_l = 0.0;
  109. float out_r = 0.0;
  110. if (((inputs[TRIGGER_DIR].value != 0.0) && dir_last_value[0] == 0.0) || ((params[DIRECTION].value != 0.0) && (dir_last_value[1] == 0.0)))
  111. {
  112. forward = !forward;
  113. }
  114. dir_last_value[0] = inputs[TRIGGER_DIR].value;
  115. dir_last_value[1] = params[DIRECTION].value;
  116. // Master track
  117. if (((inputs[CV_TRIGGER_M].value != trig_last_value[0][0]) && (trig_last_value[0][0] == 0.0)) || ((params[TRIGGER_M].value != trig_last_value[0][1]) && (trig_last_value[0][1] == 0.0)))
  118. {
  119. if (!is_recording[0])
  120. {
  121. for (int c = 0; c < 2; c++)
  122. {
  123. master_loop[c].clear();
  124. slave_loop_1[c].clear();
  125. slave_loop_2[c].clear();
  126. }
  127. sample = 0;
  128. loop_size = 0;
  129. master_rec = false;
  130. overdubbing[0] = false;
  131. }
  132. else
  133. {
  134. master_rec = true;
  135. }
  136. is_recording[0] = !is_recording[0];
  137. }
  138. trig_last_value[0][0] = inputs[CV_TRIGGER_M].value;
  139. trig_last_value[0][1] = params[TRIGGER_M].value;
  140. if (((inputs[CV_OVERDUB_M].value != overdub_last_value[0][0]) && (overdub_last_value[0][0] == 0.0)) || ((params[OVERDUB_M].value != overdub_last_value[0][1]) && (overdub_last_value[0][1] == 0.0)))
  141. {
  142. if (!overdubbing[0] && master_rec)
  143. {
  144. overdubbing[0] = true;
  145. }
  146. else if (overdubbing && master_rec)
  147. {
  148. overdubbing[0] = false;
  149. }
  150. }
  151. overdub_last_value[0][0] = inputs[CV_OVERDUB_M].value;
  152. overdub_last_value[0][1] = params[OVERDUB_M].value;
  153. if (((inputs[CV_CLEAR_M].value != 0.0) && clear_last_value[0][0] == 0.0) || ((params[CLEAR_M].value != 0.0) && (clear_last_value[0][1] == 0.0)))
  154. {
  155. master_rec = false;
  156. is_recording[0] = false;
  157. overdubbing[0] = false;
  158. for (int c = 0; c < 2; c++)
  159. {
  160. master_loop[c].assign(master_loop[c].size(), 0.0);
  161. }
  162. }
  163. clear_last_value[0][0] = inputs[CV_CLEAR_M].value;
  164. clear_last_value[0][1] = params[CLEAR_M].value;
  165. if (is_recording[0])
  166. {
  167. ++loop_size;
  168. if (loop_size < INT_MAX)
  169. {
  170. master_loop[0].push_back(in_l);
  171. master_loop[1].push_back(in_r);
  172. slave_loop_1[0].push_back(0.0);
  173. slave_loop_1[1].push_back(0.0);
  174. slave_loop_2[0].push_back(0.0);
  175. slave_loop_2[1].push_back(0.0);
  176. }
  177. else
  178. {
  179. --loop_size;
  180. is_recording[0] = false;
  181. master_rec = true;
  182. }
  183. }
  184. else
  185. {
  186. if (!master_loop[0].empty())
  187. {
  188. if (overdubbing[0])
  189. {
  190. master_loop[0].at(sample) += in_l;
  191. master_loop[1].at(sample) += in_r;
  192. }
  193. out_l += master_loop[0].at(sample) * params[GAIN_M].value;
  194. out_r += master_loop[1].at(sample) * params[GAIN_M].value;
  195. }
  196. else
  197. {
  198. out_l += 0.0;
  199. out_r += 0.0;
  200. }
  201. }
  202. // Slave tracks
  203. if (master_rec)
  204. {
  205. for (int t = 1; t < 3; t++)
  206. {
  207. if (((inputs[CV_TRIGGER_M + t].value != trig_last_value[t][0]) && (trig_last_value[t][0] == 0.0)) || ((params[TRIGGER_M + t].value != trig_last_value[t][1]) && (trig_last_value[t][1] == 0.0)))
  208. {
  209. if (!is_recording[t])
  210. {
  211. for (int c = 0; c < 2; c++)
  212. {
  213. t < 2 ? slave_loop_1[c].assign(slave_loop_1[c].size(), 0.0) : slave_loop_2[c].assign(slave_loop_2[c].size(), 0.0);
  214. }
  215. overdubbing[t] = false;
  216. }
  217. is_recording[t] = !is_recording[t];
  218. }
  219. trig_last_value[t][0] = inputs[CV_TRIGGER_M + t].value;
  220. trig_last_value[t][1] = params[TRIGGER_M + t].value;
  221. if (((inputs[CV_OVERDUB_M + t].value != overdub_last_value[t][0]) && (overdub_last_value[t][0] == 0.0)) || ((params[OVERDUB_M + t].value != overdub_last_value[t][1]) && (overdub_last_value[t][1] == 0.0)))
  222. {
  223. if (!overdubbing[t] && master_rec)
  224. {
  225. overdubbing[t] = true;
  226. }
  227. else if (overdubbing && master_rec)
  228. {
  229. overdubbing[t] = false;
  230. }
  231. }
  232. overdub_last_value[t][0] = inputs[CV_OVERDUB_M + t].value;
  233. overdub_last_value[t][1] = params[OVERDUB_M + t].value;
  234. if (((inputs[CV_CLEAR_M + t].value != 0.0) && clear_last_value[t][0] == 0.0) || ((params[CLEAR_M].value != 0.0) && (clear_last_value[t][1] == 0.0)))
  235. {
  236. is_recording[t] = false;
  237. overdubbing[t] = false;
  238. for (int c = 0; c < 2; c++)
  239. {
  240. t < 2 ? slave_loop_1[c].assign(slave_loop_1[c].size(), 0.0) : slave_loop_2[c].assign(slave_loop_2[c].size(), 0.0);
  241. }
  242. sample = 0;
  243. }
  244. clear_last_value[t][0] = inputs[CV_CLEAR_M + t].value;
  245. clear_last_value[t][1] = params[CLEAR_M + t].value;
  246. if (is_recording[t])
  247. {
  248. t < 2 ? slave_loop_1[0].at(sample) = in_l : slave_loop_2[0].at(sample) = in_l;
  249. t < 2 ? slave_loop_1[1].at(sample) = in_r : slave_loop_2[1].at(sample) = in_r;
  250. }
  251. else
  252. {
  253. if (!master_loop[0].empty())
  254. {
  255. if (overdubbing[t])
  256. {
  257. t < 2 ? slave_loop_1[0].at(sample) += in_l : slave_loop_2[0].at(sample) += in_l;
  258. t < 2 ? slave_loop_1[1].at(sample) += in_r : slave_loop_2[1].at(sample) += in_r;
  259. }
  260. t < 2 ? out_l += slave_loop_1[0].at(sample) * params[GAIN_S1].value : out_l += slave_loop_2[0].at(sample) * params[GAIN_S2].value;
  261. t < 2 ? out_r += slave_loop_1[1].at(sample) * params[GAIN_S1].value : out_r += slave_loop_2[1].at(sample) * params[GAIN_S2].value;
  262. }
  263. else
  264. {
  265. out_l += 0.0;
  266. out_r += 0.0;
  267. }
  268. }
  269. }
  270. }
  271. (is_recording[0] || overdubbing[0]) ? lights[REC_LIGHT_M].value = 1.0 : lights[REC_LIGHT_M].value = 0.0;
  272. (is_recording[1] || overdubbing[1]) ? lights[REC_LIGHT_S1].value = 1.0 : lights[REC_LIGHT_S1].value = 0.0;
  273. (is_recording[2] || overdubbing[2]) ? lights[REC_LIGHT_S2].value = 1.0 : lights[REC_LIGHT_S2].value = 0.0;
  274. master_rec ? lights[PLAY_LIGHT_M].value = 1.0 : lights[PLAY_LIGHT_M].value = 0.0;
  275. master_rec ? lights[PLAY_LIGHT_S1].value = 1.0 : lights[PLAY_LIGHT_S1].value = 0.0;
  276. master_rec ? lights[PLAY_LIGHT_S2].value = 1.0 : lights[PLAY_LIGHT_S2].value = 0.0;
  277. if (is_recording[0] || is_recording[1] || is_recording[2])
  278. {
  279. out_l += in_l;
  280. out_r += in_r;
  281. }
  282. outputs[OUTPUT_L].value = out_l;
  283. outputs[OUTPUT_R].value = out_r;
  284. if (master_rec)
  285. {
  286. outputs[OUTPUT_ML].value = master_loop[0].at(sample) * params[GAIN_M].value;
  287. outputs[OUTPUT_MR].value = master_loop[1].at(sample) * params[GAIN_M].value;
  288. outputs[OUTPUT_S1L].value = slave_loop_1[0].at(sample) * params[GAIN_S1].value;
  289. outputs[OUTPUT_S1R].value = slave_loop_1[1].at(sample) * params[GAIN_S1].value;
  290. outputs[OUTPUT_S2L].value = slave_loop_2[0].at(sample) * params[GAIN_S2].value;
  291. outputs[OUTPUT_S2R].value = slave_loop_2[1].at(sample) * params[GAIN_S2].value;
  292. }
  293. if (!is_recording[0])
  294. {
  295. if (forward)
  296. {
  297. if (++sample >= (int)master_loop[0].size())
  298. {
  299. sample = 0;
  300. }
  301. }
  302. else
  303. {
  304. if (--sample < 0)
  305. {
  306. sample = master_loop[0].size() - 1;
  307. }
  308. }
  309. }
  310. if (((inputs[CV_ERASE].value != 0.0) && erase_last_value[0] == 0.0) || ((params[ERASE].value != 0.0) && (erase_last_value[1] == 0.0)))
  311. {
  312. master_rec = false;
  313. for (int t = 0; t < 3; t++)
  314. {
  315. is_recording[t] = false;
  316. overdubbing[t] = false;
  317. }
  318. for (int c = 0; c < 2; c++)
  319. {
  320. master_loop[c].clear();
  321. slave_loop_1[c].clear();
  322. slave_loop_2[c].clear();
  323. }
  324. sample = 0;
  325. loop_size = 0;
  326. }
  327. erase_last_value[0] = inputs[CV_ERASE].value;
  328. erase_last_value[1] = params[ERASE].value;
  329. }
  330. struct Luppolo3Widget : ModuleWidget
  331. {
  332. Luppolo3Widget(Luppolo3 *module);
  333. };
  334. Luppolo3Widget::Luppolo3Widget(Luppolo3 *module) : ModuleWidget(module)
  335. {
  336. box.size = Vec(15 * 30, 380);
  337. {
  338. SVGPanel *panel = new SVGPanel();
  339. panel->box.size = box.size;
  340. panel->setBackground(SVG::load(assetPlugin(plugin, "res/luppolo3.svg")));
  341. addChild(panel);
  342. }
  343. addChild(Widget::create<SonusScrew>(Vec(0, 0)));
  344. addChild(Widget::create<SonusScrew>(Vec(box.size.x - 15, 0)));
  345. addChild(Widget::create<SonusScrew>(Vec(0, 365)));
  346. addChild(Widget::create<SonusScrew>(Vec(box.size.x - 15, 365)));
  347. addInput(Port::create<PJ301MPort>(Vec(14, 92), Port::INPUT, module, Luppolo3::INPUT_L));
  348. addInput(Port::create<PJ301MPort>(Vec(52, 92), Port::INPUT, module, Luppolo3::INPUT_R));
  349. addInput(Port::create<PJ301MPort>(Vec(14, 215), Port::INPUT, module, Luppolo3::TRIGGER_DIR));
  350. addParam(ParamWidget::create<CKD6>(Vec(50, 213), module, Luppolo3::DIRECTION, 0.0, 1.0, 0.0));
  351. addInput(Port::create<PJ301MPort>(Vec(14,272), Port::INPUT, module, Luppolo3::CV_ERASE));
  352. addParam(ParamWidget::create<CKD6>(Vec(50,270), module, Luppolo3::ERASE, 0.0, 1.0, 0.0));
  353. addParam(ParamWidget::create<SonusKnob>(Vec(117, 85), module, Luppolo3::GAIN_M, 0.0, 2.0, 1.0));
  354. addInput(Port::create<PJ301MPort>(Vec(104, 155), Port::INPUT, module, Luppolo3::CV_TRIGGER_M));
  355. addParam(ParamWidget::create<CKD6>(Vec(140, 153), module, Luppolo3::TRIGGER_M, 0.0, 1.0, 0.0));
  356. addInput(Port::create<PJ301MPort>(Vec(104, 215), Port::INPUT, module, Luppolo3::CV_OVERDUB_M));
  357. addParam(ParamWidget::create<CKD6>(Vec(140, 213), module, Luppolo3::OVERDUB_M, 0.0, 1.0, 0.0));
  358. addInput(Port::create<PJ301MPort>(Vec(104,272), Port::INPUT, module, Luppolo3::CV_CLEAR_M));
  359. addParam(ParamWidget::create<CKD6>(Vec(140,270), module, Luppolo3::CLEAR_M, 0.0, 1.0, 0.0));
  360. addParam(ParamWidget::create<SonusKnob>(Vec(207, 85), module, Luppolo3::GAIN_S1, 0.0, 2.0, 1.0));
  361. addInput(Port::create<PJ301MPort>(Vec(194, 155), Port::INPUT, module, Luppolo3::CV_TRIGGER_S1));
  362. addParam(ParamWidget::create<CKD6>(Vec(230, 153), module, Luppolo3::TRIGGER_S1, 0.0, 1.0, 0.0));
  363. addInput(Port::create<PJ301MPort>(Vec(194, 215), Port::INPUT, module, Luppolo3::CV_OVERDUB_S1));
  364. addParam(ParamWidget::create<CKD6>(Vec(230, 213), module, Luppolo3::OVERDUB_S1, 0.0, 1.0, 0.0));
  365. addInput(Port::create<PJ301MPort>(Vec(194, 272), Port::INPUT, module, Luppolo3::CV_CLEAR_S1));
  366. addParam(ParamWidget::create<CKD6>(Vec(230,270), module, Luppolo3::CLEAR_S1, 0.0, 1.0, 0.0));
  367. addParam(ParamWidget::create<SonusKnob>(Vec(297, 85), module, Luppolo3::GAIN_S2, 0.0, 2.0, 1.0));
  368. addInput(Port::create<PJ301MPort>(Vec(284, 155), Port::INPUT, module, Luppolo3::CV_TRIGGER_S2));
  369. addParam(ParamWidget::create<CKD6>(Vec(320, 153), module, Luppolo3::TRIGGER_S2, 0.0, 1.0, 0.0));
  370. addInput(Port::create<PJ301MPort>(Vec(284, 215), Port::INPUT, module, Luppolo3::CV_OVERDUB_S2));
  371. addParam(ParamWidget::create<CKD6>(Vec(320, 213), module, Luppolo3::OVERDUB_S2, 0.0, 1.0, 0.0));
  372. addInput(Port::create<PJ301MPort>(Vec(284,272), Port::INPUT, module, Luppolo3::CV_CLEAR_S2));
  373. addParam(ParamWidget::create<CKD6>(Vec(320,270), module, Luppolo3::CLEAR_S2, 0.0, 1.0, 0.0));
  374. addOutput(Port::create<PJ301MPort>(Vec(374, 92), Port::OUTPUT, module, Luppolo3::OUTPUT_L));
  375. addOutput(Port::create<PJ301MPort>(Vec(412, 92), Port::OUTPUT, module, Luppolo3::OUTPUT_R));
  376. addOutput(Port::create<PJ301MPort>(Vec(374, 175), Port::OUTPUT, module, Luppolo3::OUTPUT_ML));
  377. addOutput(Port::create<PJ301MPort>(Vec(412, 175), Port::OUTPUT, module, Luppolo3::OUTPUT_MR));
  378. addOutput(Port::create<PJ301MPort>(Vec(374, 230), Port::OUTPUT, module, Luppolo3::OUTPUT_S1L));
  379. addOutput(Port::create<PJ301MPort>(Vec(412, 230), Port::OUTPUT, module, Luppolo3::OUTPUT_S1R));
  380. addOutput(Port::create<PJ301MPort>(Vec(374, 286), Port::OUTPUT, module, Luppolo3::OUTPUT_S2L));
  381. addOutput(Port::create<PJ301MPort>(Vec(412, 286), Port::OUTPUT, module, Luppolo3::OUTPUT_S2R));
  382. addChild(ModuleLightWidget::create<MediumLight<RedLight>>(Vec(113, 65), module, Luppolo3::REC_LIGHT_M));
  383. addChild(ModuleLightWidget::create<MediumLight<GreenLight>>(Vec(148, 65), module, Luppolo3::PLAY_LIGHT_M));
  384. addChild(ModuleLightWidget::create<MediumLight<RedLight>>(Vec(203, 65), module, Luppolo3::REC_LIGHT_S1));
  385. addChild(ModuleLightWidget::create<MediumLight<GreenLight>>(Vec(238, 65), module, Luppolo3::PLAY_LIGHT_S1));
  386. addChild(ModuleLightWidget::create<MediumLight<RedLight>>(Vec(293, 65), module, Luppolo3::REC_LIGHT_S2));
  387. addChild(ModuleLightWidget::create<MediumLight<GreenLight>>(Vec(328, 65), module, Luppolo3::PLAY_LIGHT_S2));
  388. }
  389. } // namespace rack_plugin_SonusModular
  390. using namespace rack_plugin_SonusModular;
  391. RACK_PLUGIN_MODEL_INIT(SonusModular, Luppolo3) {
  392. Model *modelLuppolo3 = Model::create<Luppolo3, Luppolo3Widget>("Sonus Modular", "Luppolo3", "Luppolo3 | Loop Station", SAMPLER_TAG);
  393. return modelLuppolo3;
  394. }