#include "plugin.hpp" struct SEQ3 : Module { enum ParamIds { CLOCK_PARAM, RUN_PARAM, RESET_PARAM, STEPS_PARAM, ENUMS(ROW1_PARAM, 8), ENUMS(ROW2_PARAM, 8), ENUMS(ROW3_PARAM, 8), ENUMS(GATE_PARAM, 8), NUM_PARAMS }; enum InputIds { CLOCK_INPUT, EXT_CLOCK_INPUT, RESET_INPUT, STEPS_INPUT, NUM_INPUTS }; enum OutputIds { GATES_OUTPUT, ROW1_OUTPUT, ROW2_OUTPUT, ROW3_OUTPUT, ENUMS(GATE_OUTPUT, 8), NUM_OUTPUTS }; enum LightIds { RUNNING_LIGHT, RESET_LIGHT, GATES_LIGHT, ENUMS(ROW_LIGHTS, 3), ENUMS(GATE_LIGHTS, 8), NUM_LIGHTS }; bool running = true; dsp::SchmittTrigger clockTrigger; dsp::SchmittTrigger runningTrigger; dsp::SchmittTrigger resetTrigger; dsp::SchmittTrigger gateTriggers[8]; /** Phase of internal LFO */ float phase = 0.f; int index = 0; bool gates[8] = {}; SEQ3() { config(NUM_PARAMS, NUM_INPUTS, NUM_OUTPUTS, NUM_LIGHTS); configParam(CLOCK_PARAM, -2.f, 6.f, 2.f, "Clock tempo", " bpm", 2.f, 60.f); configParam(RUN_PARAM, 0.f, 1.f, 0.f); configParam(RESET_PARAM, 0.f, 1.f, 0.f); configParam(STEPS_PARAM, 1.f, 8.f, 8.f); for (int i = 0; i < 8; i++) { configParam(ROW1_PARAM + i, 0.f, 10.f, 0.f); configParam(ROW2_PARAM + i, 0.f, 10.f, 0.f); configParam(ROW3_PARAM + i, 0.f, 10.f, 0.f); configParam(GATE_PARAM + i, 0.f, 1.f, 0.f); } onReset(); } void onReset() override { for (int i = 0; i < 8; i++) { gates[i] = true; } } void onRandomize() override { for (int i = 0; i < 8; i++) { gates[i] = (random::uniform() > 0.5f); } } json_t *dataToJson() override { json_t *rootJ = json_object(); // running json_object_set_new(rootJ, "running", json_boolean(running)); // gates json_t *gatesJ = json_array(); for (int i = 0; i < 8; i++) { json_array_insert_new(gatesJ, i, json_integer((int) gates[i])); } json_object_set_new(rootJ, "gates", gatesJ); return rootJ; } void dataFromJson(json_t *rootJ) override { // running json_t *runningJ = json_object_get(rootJ, "running"); if (runningJ) running = json_is_true(runningJ); // gates json_t *gatesJ = json_object_get(rootJ, "gates"); if (gatesJ) { for (int i = 0; i < 8; i++) { json_t *gateJ = json_array_get(gatesJ, i); if (gateJ) gates[i] = !!json_integer_value(gateJ); } } } void setIndex(int index) { int numSteps = (int) clamp(std::round(params[STEPS_PARAM].getValue() + inputs[STEPS_INPUT].getVoltage()), 1.f, 8.f); phase = 0.f; this->index = index; if (this->index >= numSteps) this->index = 0; } void process(const ProcessArgs &args) override { // Run if (runningTrigger.process(params[RUN_PARAM].getValue())) { running = !running; } bool gateIn = false; if (running) { if (inputs[EXT_CLOCK_INPUT].isConnected()) { // External clock if (clockTrigger.process(inputs[EXT_CLOCK_INPUT].getVoltage())) { setIndex(index + 1); } gateIn = clockTrigger.isHigh(); } else { // Internal clock float clockTime = std::pow(2.f, params[CLOCK_PARAM].getValue() + inputs[CLOCK_INPUT].getVoltage()); phase += clockTime * args.sampleTime; if (phase >= 1.f) { setIndex(index + 1); } gateIn = (phase < 0.5f); } } // Reset if (resetTrigger.process(params[RESET_PARAM].getValue() + inputs[RESET_INPUT].getVoltage())) { setIndex(0); } // Gate buttons for (int i = 0; i < 8; i++) { if (gateTriggers[i].process(params[GATE_PARAM + i].getValue())) { gates[i] = !gates[i]; } outputs[GATE_OUTPUT + i].setVoltage((running && gateIn && i == index && gates[i]) ? 10.f : 0.f); lights[GATE_LIGHTS + i].setSmoothBrightness((gateIn && i == index) ? (gates[i] ? 1.f : 0.33) : (gates[i] ? 0.66 : 0.0), args.sampleTime); } // Outputs outputs[ROW1_OUTPUT].setVoltage(params[ROW1_PARAM + index].getValue()); outputs[ROW2_OUTPUT].setVoltage(params[ROW2_PARAM + index].getValue()); outputs[ROW3_OUTPUT].setVoltage(params[ROW3_PARAM + index].getValue()); outputs[GATES_OUTPUT].setVoltage((gateIn && gates[index]) ? 10.f : 0.f); lights[RUNNING_LIGHT].value = (running); lights[RESET_LIGHT].setSmoothBrightness(resetTrigger.isHigh(), args.sampleTime); lights[GATES_LIGHT].setSmoothBrightness(gateIn, args.sampleTime); lights[ROW_LIGHTS].value = outputs[ROW1_OUTPUT].value / 10.f; lights[ROW_LIGHTS + 1].value = outputs[ROW2_OUTPUT].value / 10.f; lights[ROW_LIGHTS + 2].value = outputs[ROW3_OUTPUT].value / 10.f; } }; struct SEQ3Widget : ModuleWidget { SEQ3Widget(SEQ3 *module) { setModule(module); setPanel(APP->window->loadSvg(asset::plugin(pluginInstance, "res/SEQ3.svg"))); addChild(createWidget(Vec(15, 0))); addChild(createWidget(Vec(box.size.x-30, 0))); addChild(createWidget(Vec(15, 365))); addChild(createWidget(Vec(box.size.x-30, 365))); addParam(createParam(Vec(18, 56), module, SEQ3::CLOCK_PARAM)); addParam(createParam(Vec(60, 61-1), module, SEQ3::RUN_PARAM)); addChild(createLight>(Vec(64.4f, 64.4f), module, SEQ3::RUNNING_LIGHT)); addParam(createParam(Vec(99, 61-1), module, SEQ3::RESET_PARAM)); addChild(createLight>(Vec(103.4f, 64.4f), module, SEQ3::RESET_LIGHT)); addParam(createParam(Vec(132, 56), module, SEQ3::STEPS_PARAM)); addChild(createLight>(Vec(179.4f, 64.4f), module, SEQ3::GATES_LIGHT)); addChild(createLight>(Vec(218.4f, 64.4f), module, SEQ3::ROW_LIGHTS)); addChild(createLight>(Vec(256.4f, 64.4f), module, SEQ3::ROW_LIGHTS + 1)); addChild(createLight>(Vec(295.4f, 64.4f), module, SEQ3::ROW_LIGHTS + 2)); static const float portX[8] = {20, 58, 96, 135, 173, 212, 250, 289}; addInput(createInput(Vec(portX[0]-1, 98), module, SEQ3::CLOCK_INPUT)); addInput(createInput(Vec(portX[1]-1, 98), module, SEQ3::EXT_CLOCK_INPUT)); addInput(createInput(Vec(portX[2]-1, 98), module, SEQ3::RESET_INPUT)); addInput(createInput(Vec(portX[3]-1, 98), module, SEQ3::STEPS_INPUT)); addOutput(createOutput(Vec(portX[4]-1, 98), module, SEQ3::GATES_OUTPUT)); addOutput(createOutput(Vec(portX[5]-1, 98), module, SEQ3::ROW1_OUTPUT)); addOutput(createOutput(Vec(portX[6]-1, 98), module, SEQ3::ROW2_OUTPUT)); addOutput(createOutput(Vec(portX[7]-1, 98), module, SEQ3::ROW3_OUTPUT)); for (int i = 0; i < 8; i++) { addParam(createParam(Vec(portX[i]-2, 157), module, SEQ3::ROW1_PARAM + i)); addParam(createParam(Vec(portX[i]-2, 198), module, SEQ3::ROW2_PARAM + i)); addParam(createParam(Vec(portX[i]-2, 240), module, SEQ3::ROW3_PARAM + i)); addParam(createParam(Vec(portX[i]+2, 278-1), module, SEQ3::GATE_PARAM + i)); addChild(createLight>(Vec(portX[i]+6.4f, 281.4f), module, SEQ3::GATE_LIGHTS + i)); addOutput(createOutput(Vec(portX[i]-1, 307), module, SEQ3::GATE_OUTPUT + i)); } } }; Model *modelSEQ3 = createModel("SEQ3");