#include #include #include #include #include #include #define SCREEN_ADDRESS 0x3C #define PPQN 24 #define PULSE_LENGTH 5 #define MAXBPM 250 //250 at 24ppqn with 5ms pulse will be 50/50 square wave #define MINBPM 20 #define INPUT_CONNECTED_PIN 1 #define INPUT_PIN 2 //needs to be an interrupt pin #define ENC_BTN_PIN 17 #define ENC_D1_PIN 4 #define ENC_D2_PIN 3 #define START_STOP_BTN_PIN 14 #define ANALOGUE_INPUT_1_PIN A1 #define ANALOGUE_INPUT_2_PIN A1 const int outsPins[6] = {5, 6, 7, 8, 9, 10}; const int outsModes[18] = {-24, -16, -12, -8, -6, -4, -3, -2, 1, 2, 3, 4, 5, 6, 7, 8, 16, 32}; //positive - divide, negative - multiply, 0 - off int outsModeIndex[6] = {8, 9, 10, 7, 2, 6}; //8 - 1 bool outsRandom[6] = {0,0,0,0,0,1}; bool outsMod[6] = {0,0,0,0,1,1}; //0 - A1, 1 - A2 int outsModRange[6] = {0,0,0,0,4,-4}; int outsPeriods[6]; int outsClocksCounts[6]; int outsModesPlay[6]; //actual channel modes array updated from outsModeIndex each beat bool clockMode; bool clockModeOld; int inputMode; //clock, reset, start/stop int clockCount = 0; int bpm = 130; int pulseClockCount = 0; int pulseCount = 0; int pulsePeriod; bool isPlaying = 0; int needToResetChannel; bool beatCounted = false; bool pulseCounted = false; int displayTab = 0; int displayTabOld; int insideTab = 0; bool playBtnPushed = false; int a1Input = 0; int a2Input = 0; bool changesSaved; int encPositionOld = 0; unsigned long encPressedTime; unsigned long encReleasedTime; bool encPressRegistered; Adafruit_SSD1306 display(128, 64, &Wire, -1); RotaryEncoder encoder(ENC_D1_PIN, ENC_D2_PIN, RotaryEncoder::LatchMode::TWO03); void setup() { Serial.begin(9600); pinMode(INPUT_CONNECTED_PIN, INPUT_PULLUP); pinMode(ENC_BTN_PIN, INPUT_PULLUP); pinMode(START_STOP_BTN_PIN, INPUT_PULLUP); pinMode(START_STOP_BTN_PIN, ANALOGUE_INPUT_1_PIN); pinMode(INPUT_PIN, INPUT_PULLUP); //probably will need interrupt for (int i=0; i<6; i++) { pinMode(outsPins[i], OUTPUT); } display.begin(SSD1306_SWITCHCAPVCC, SCREEN_ADDRESS); updateScreen(); updateTiming(); FlexiTimer2::set(1, 1.0/1000, internalClock); // 1.0/1000 = 1ms period FlexiTimer2::start(); } void loop() { checkInputs(); } void internalClock() { if (isPlaying) { // Action on each pulse if (pulseClockCount == 0 && !pulseCounted) { //modulation for (int i = 0; i<6; i++) { int mod; if (!outsMod[i]) { mod = a1Input; } else { mod = a2Input; } mod = map (mod, 0, 1023, 0, outsModRange[i]); outsModesPlay[i] = outsModes[outsModeIndex[i] - mod]; //subtrackting because the innitiall array is backwards } //divider if (pulseCount == 0 && !beatCounted) { for (int i = 0; i<6; i++) { outsModesPlay[i] = outsModes[outsModeIndex[i]]; //updated here to prevent sync problems for multipliers /*if (a1Input != 0 && i == 5) { //TESTING THE MODULATION ON 6th CHANNEL outsModesPlay[i] = outsModes[outsModeIndex[i] - a1Input]; }*/ if (outsModesPlay[i] > 0) { if (outsClocksCounts[i] == 0) { //Pulse on 0 if (outsRandom[i] == 0 || (outsRandom[i] == 1 && random(2))) { //random digitalWrite(outsPins[i], HIGH); } } if (outsClocksCounts[i] < (outsModesPlay[i] - 1)) { outsClocksCounts[i]++; } else { outsClocksCounts[i] = 0; } } } beatCounted = true; } //multiplier for (int i = 0; i<6; i++) { if (outsModesPlay[i] < 0) { if (outsClocksCounts[i] == 0) { //Pulse on 0 if (outsRandom[i] == 0 || (outsRandom[i] == 1 && random(2))) { //random digitalWrite(outsPins[i], HIGH); } } if (outsClocksCounts[i] < (PPQN / abs(outsModesPlay[i])) - 1) { outsClocksCounts[i]++; } else { outsClocksCounts[i] = 0; } } } pulseCounted = true; } //internal pulse if (pulseClockCount == 0) { pulseCount++; beatCounted = false; pulseCounted = false; } if (pulseClockCount < pulsePeriod) { pulseClockCount++; } else { pulseClockCount = 0; } if (pulseCount >= PPQN) { pulseCount = 0; } // pull low all outputs after set pulse length if (pulseClockCount >= PULSE_LENGTH) { for (int i = 0; i<6; i++) { digitalWrite(outsPins[i], LOW); } } } } void updateTiming() { pulsePeriod = 60000 / (bpm * PPQN); } void resetClocks() { for (int i = 0; i<6; i++) { outsClocksCounts[i] = 0; } } void checkInputs() { //input jack switcch clockMode = digitalRead(INPUT_CONNECTED_PIN); if (clockMode != clockModeOld) { updateScreen(); clockModeOld = clockMode; } //encoder button if (!digitalRead(ENC_BTN_PIN) && !encPressRegistered) { encPressRegistered = true; encPressedTime = millis(); } else if (digitalRead(ENC_BTN_PIN) && encPressRegistered) { encPressRegistered = false; encReleasedTime = millis(); Serial.println(encReleasedTime - encPressedTime); if (encReleasedTime - encPressedTime < 500) { // press shorter than .5s switches tabs if (insideTab == 0) { displayTabOld = displayTab; displayTab++; if (displayTab>6) { displayTab = 0; } } else if (insideTab < 2) { insideTab ++; } else { insideTab = 1; } updateScreen(); } else if (encReleasedTime - encPressedTime < 2000 && displayTab != 0) { // longer press (<2s) and switches random mode, longer than 2s presses are ignored if (insideTab == 0) { insideTab = 1; } else { insideTab = 0; } //outsRandom[displayTab-1] = !outsRandom[displayTab-1]; updateScreen(); } } //encoder encoder.tick(); int encPosition = encoder.getPosition(); if (encPositionOld != encPosition) { int change = encPositionOld - encPosition; if (displayTab == 0) { bpm = bpm + change; if (bpm > MAXBPM) { bpm = MAXBPM; } else if (bpm < MINBPM) { bpm = MINBPM; } updateTiming(); } else if (displayTab != 0 && insideTab == 0) { outsModeIndex[displayTab-1] = outsModeIndex[displayTab-1] - change; if (outsModeIndex[displayTab-1] < 0) { outsModeIndex[displayTab-1] = 0; } else if (outsModeIndex[displayTab-1] > (sizeof(outsModes)/sizeof(int)) - 1) { outsModeIndex[displayTab-1] = (sizeof(outsModes)/sizeof(int)) - 1; } needToResetChannel = displayTab-1; } else if (displayTab != 0 && insideTab == 1) { //random outsRandom[displayTab-1] = !outsRandom[displayTab-1]; } else if (displayTab != 0 && insideTab == 2) { //modulation outsModRange[displayTab-1] = outsModRange[displayTab-1] + change; if (outsModRange[displayTab-1] > 6 || outsModRange[displayTab-1] < -6) { outsMod[displayTab-1] = !outsMod[displayTab-1]; outsModRange[displayTab-1] = 0; } } updateScreen(); encPositionOld = encPosition; } //play button if (!digitalRead(START_STOP_BTN_PIN) && !playBtnPushed) { isPlaying = !isPlaying; resetClocks(); playBtnPushed = true; } else if (digitalRead(START_STOP_BTN_PIN) && playBtnPushed) { playBtnPushed = false; } //modulations a1Input = analogRead(ANALOGUE_INPUT_1_PIN); a2Input = analogRead(ANALOGUE_INPUT_2_PIN); } void updateScreen() { display.clearDisplay(); //Tabs display.drawRect(0, 0, 128, 2, SSD1306_WHITE); display.setCursor(0,2); display.setTextSize(1); if (displayTab == 0) { display.setTextColor(SSD1306_BLACK, SSD1306_WHITE); display.print(F(" ")); display.setTextColor(SSD1306_WHITE); display.print(F(" bpm ")); } else { display.setTextColor(SSD1306_BLACK, SSD1306_WHITE); display.print(F(" bpm")); } for (int i = 1; i <= 6; i++) { if (displayTab == i) { display.setTextColor(SSD1306_BLACK, SSD1306_WHITE); display.print(" "); display.setTextColor(SSD1306_WHITE); display.print(" "); display.print(i); display.print(" "); } else { display.setTextColor(SSD1306_BLACK, SSD1306_WHITE); display.print(" "); display.print(i); } } display.setTextColor(SSD1306_BLACK, SSD1306_WHITE); //display.println(F(" ")); display.fillRect(108, 2, 20, 8, SSD1306_WHITE); display.println(); display.println(); display.fillRect(0, 10, 128, 2, SSD1306_WHITE); //Content display.setTextSize(3); display.setTextColor(SSD1306_WHITE); if (displayTab == 0) { display.print(bpm); display.println(F("bpm")); } else { if (outsModes[outsModeIndex[displayTab-1]] == 0) { display.print(F(" OFF")); } else if (outsModes[outsModeIndex[displayTab-1]]>0) { display.print(F(" /")); display.print(abs(outsModes[outsModeIndex[displayTab-1]])); } else { display.print(F(" x")); display.print(abs(outsModes[outsModeIndex[displayTab-1]])); } } display.println(); display.setTextSize(1); display.println(); //Extra params display.setTextSize(1); if (displayTab != 0) { if (insideTab == 1) { display.setTextColor(SSD1306_BLACK, SSD1306_WHITE); } else { display.setTextColor(SSD1306_WHITE); } display.print(F(" RND:")); if (outsRandom[displayTab-1]) { display.print(F("On ")); } else { display.print(F("Off ")); } display.setCursor(60,50); if (insideTab == 2) { display.setTextColor(SSD1306_BLACK, SSD1306_WHITE); } else { display.setTextColor(SSD1306_WHITE); } display.print(F(" MOD:")); if (outsMod[displayTab-1] && outsModRange[displayTab-1] != 0) { display.print(F("A2 ")); if (outsModRange[displayTab-1] > 0) { display.print(F("+")); } display.print(outsModRange[displayTab-1]); display.print(F(" ")); } else if (!outsMod[displayTab-1] && outsModRange[displayTab-1] != 0) { display.print(F("A1 ")); if (outsModRange[displayTab-1] > 0) { display.print(F("+")); } display.print(outsModRange[displayTab-1]); display.print(F(" ")); } else { display.print(F("Off ")); } } display.display(); }