678 lines
21 KiB
C++
678 lines
21 KiB
C++
#include <Wire.h>
|
|
#include <Adafruit_GFX.h>
|
|
#include <Adafruit_SSD1306.h>
|
|
#include <RotaryEncoder.h>
|
|
#include <FlexiTimer2.h>
|
|
#include <EEPROM.h>
|
|
|
|
#define SCREEN_ADDRESS 0x3C
|
|
|
|
#define PPQN 24
|
|
#define PULSE_LENGTH 12 //ms (with 12 ms you can't get higher than 208bpm)
|
|
#define MAXBPM 200 //250 at 24ppqn with 5ms pulse will be 50/50 square wave
|
|
#define MINBPM 20
|
|
#define SCREEN_TIMEOUT 120000 //Turn display off after 2 min
|
|
|
|
#define ENC_BTN_PIN 14
|
|
#define ENC_D1_PIN 17
|
|
#define ENC_D2_PIN 4
|
|
#define START_STOP_BTN_PIN 5
|
|
|
|
#define EXT_INPUT_PIN 2 //needs to be an interrupt pin
|
|
#define ANALOGUE_INPUT_1_PIN A2
|
|
#define ANALOGUE_INPUT_2_PIN A1
|
|
|
|
const int outsPins[6] = {6, 11, 7, 10, 8, 9};
|
|
|
|
const int clockModes[17] = {-24, -12, -8, -6, -4, -3, -2, 1, 2, 3, 4, 5, 6, 7, 8, 16, 32}; //positive - divide, negative - multiply, 0 - off
|
|
|
|
unsigned int bpm = 130;
|
|
bool bpmModulationChannel; //0 - CV1, 1 - CV2
|
|
byte bpmModulationRange = 0;
|
|
|
|
struct channel {
|
|
unsigned int mode;
|
|
unsigned int random;
|
|
bool modulationChannel; //0 - CV1, 1 - CV2
|
|
int modulationRange;
|
|
unsigned int offset;
|
|
};
|
|
|
|
channel channels[6] = { //array of channel settings
|
|
{ 7, 0, 0, 0, 0 },
|
|
{ 7, 0, 0, 0, 0 },
|
|
{ 7, 0, 0, 0, 0 },
|
|
{ 7, 0, 0, 0, 0 },
|
|
{ 7, 0, 0, 0, 0 },
|
|
{ 7, 0, 0, 0, 0 }
|
|
};
|
|
|
|
int channelPulseCount[6];
|
|
int channelPulsesPerCycle[6];
|
|
int playingModes[6]; //actual channel modes array updated from channels object on each beat
|
|
|
|
unsigned int pulsePeriod;
|
|
bool isPlaying = false;
|
|
|
|
unsigned int tickCount = 0;
|
|
unsigned int pulseCount = 0;
|
|
unsigned int extTriggerCount = 0;
|
|
byte extResetCountdown = 0;
|
|
|
|
byte masterClockMode = 0; // 0 - internal, 1 - external 24ppqn, 2 - external beat
|
|
unsigned long lastExtPulseTime;
|
|
unsigned long newExtPulseTime;
|
|
|
|
bool needPulseReset[6] = {true, true, true, true, true, true};
|
|
|
|
unsigned int displayTab = 0;
|
|
unsigned int displayTabOld;
|
|
unsigned int insideTab = 0;
|
|
bool playBtnPushed = false;
|
|
|
|
int a1Input = 0;
|
|
int a2Input = 0;
|
|
|
|
int encPositionOld = 0;
|
|
unsigned long encPressedTime;
|
|
unsigned long encReleasedTime;
|
|
bool encPressRegistered;
|
|
|
|
unsigned long lastInteractionTime; // used for display timeout
|
|
|
|
Adafruit_SSD1306 display(128, 64, &Wire, -1);
|
|
RotaryEncoder encoder(ENC_D1_PIN, ENC_D2_PIN, RotaryEncoder::LatchMode::TWO03);
|
|
|
|
const unsigned char splash_logo [] PROGMEM = {
|
|
0x00, 0x1f, 0xfc, 0x00, 0x00, 0x00, 0x7f, 0xff, 0x00, 0x00, 0x01, 0xff, 0xff, 0xc0, 0x00, 0x03,
|
|
0xfc, 0x1f, 0xe0, 0x00, 0x07, 0xe0, 0x03, 0xf0, 0x00, 0x0f, 0x80, 0x00, 0xf8, 0x00, 0x1f, 0x00,
|
|
0x00, 0x7c, 0x00, 0x1e, 0x00, 0x00, 0x3c, 0x00, 0x3c, 0x00, 0x00, 0x1e, 0x00, 0x3c, 0x00, 0x00,
|
|
0x1e, 0x00, 0x78, 0x0c, 0x18, 0x0f, 0x00, 0x78, 0x1e, 0x3c, 0x0f, 0x00, 0x70, 0x1e, 0x3c, 0x07,
|
|
0x00, 0xf0, 0x1e, 0x3c, 0x07, 0x80, 0xf0, 0x1e, 0x3c, 0x07, 0x80, 0xf0, 0x1e, 0x3c, 0x07, 0x80,
|
|
0xf0, 0x1e, 0x3c, 0x07, 0x80, 0xf0, 0x0c, 0x18, 0x07, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x04, 0x0c, 0x18, 0x10, 0x00, 0x0f, 0x1e, 0x3c, 0x78, 0x00, 0x0f, 0x1e,
|
|
0x3c, 0x78, 0x00, 0x1e, 0x0c, 0x18, 0x3c, 0x00, 0x1e, 0x00, 0x00, 0x3c, 0x00, 0x3c, 0x00, 0x00,
|
|
0x1e, 0x00, 0x3c, 0x00, 0x00, 0x1e, 0x00, 0x78, 0x00, 0x00, 0x0f, 0x00, 0x78, 0x00, 0x00, 0x0f,
|
|
0x00, 0xf0, 0x00, 0x00, 0x07, 0x80, 0xf0, 0x00, 0x00, 0x07, 0x80, 0xf0, 0x00, 0x00, 0x07, 0x80,
|
|
0x78, 0x00, 0x00, 0x0f, 0x00, 0x7c, 0x00, 0x00, 0x1f, 0x00, 0x3e, 0x00, 0x00, 0x3e, 0x00, 0x3f,
|
|
0xff, 0xff, 0xfe, 0x00, 0x1f, 0xff, 0xff, 0xfc, 0x00, 0x07, 0xff, 0xff, 0xf0, 0x00, 0x01, 0xff,
|
|
0xff, 0xc0, 0x00
|
|
};
|
|
|
|
void setup() {
|
|
//Serial.begin(9600);
|
|
|
|
//check last bit in eeprom to know if the correct settings were stored
|
|
if (EEPROM.read(1023) == 'S') {
|
|
int addr = 0;
|
|
EEPROM.get(addr, bpm);
|
|
addr = addr + sizeof(bpm);
|
|
EEPROM.get(addr, bpmModulationChannel);
|
|
addr = addr + sizeof(bpmModulationChannel);
|
|
EEPROM.get(addr, bpmModulationRange);
|
|
addr = addr + sizeof(bpmModulationRange);
|
|
EEPROM.get(addr, masterClockMode);
|
|
addr = addr + sizeof(masterClockMode);
|
|
EEPROM.get(addr, channels);
|
|
} else {
|
|
saveState();
|
|
EEPROM.write(1023, 'S');
|
|
}
|
|
|
|
pinMode(ENC_BTN_PIN, INPUT_PULLUP);
|
|
pinMode(START_STOP_BTN_PIN, INPUT_PULLUP);
|
|
pinMode(START_STOP_BTN_PIN, ANALOGUE_INPUT_1_PIN);
|
|
pinMode(EXT_INPUT_PIN, INPUT_PULLUP);
|
|
attachInterrupt(digitalPinToInterrupt(EXT_INPUT_PIN), externalClock, FALLING);
|
|
|
|
for (int i=0; i<6; i++) {
|
|
pinMode(outsPins[i], OUTPUT);
|
|
}
|
|
|
|
display.begin(SSD1306_SWITCHCAPVCC, SCREEN_ADDRESS);
|
|
|
|
//Splash screen
|
|
display.clearDisplay();
|
|
display.drawBitmap(48, 18, splash_logo, 33, 39, 1);
|
|
display.setCursor(90,1);
|
|
display.setTextSize(1);
|
|
display.setTextColor(SSD1306_WHITE);
|
|
display.println(F("V:0.1a")); //Version
|
|
display.display();
|
|
delay(800);
|
|
|
|
updateScreen();
|
|
calculateCycles();
|
|
calculateBPMTiming();
|
|
|
|
FlexiTimer2::set(1, 1.0/1000, clock); // 1.0/1000 = 1ms period. If other than 1ms calculateBPMTiming() might need tweaking
|
|
FlexiTimer2::start();
|
|
}
|
|
|
|
void loop() {
|
|
checkInputs();
|
|
if ((millis() - lastInteractionTime) > SCREEN_TIMEOUT) {
|
|
display.clearDisplay();
|
|
display.display();
|
|
}
|
|
if (masterClockMode == 2 || masterClockMode == 3) {
|
|
calculateBPMTiming();
|
|
}
|
|
}
|
|
|
|
void clock() {
|
|
if (isPlaying) {
|
|
|
|
// Action on each pulse
|
|
if (tickCount == 0) {
|
|
sendTriggers();
|
|
}
|
|
|
|
//this part gets the Pulse and Ticks ticking
|
|
//it's placed after the triggers to avoid problems on the start (when pulseCount==0)
|
|
tickCount++;
|
|
if (masterClockMode == 0) {
|
|
if (tickCount >= pulsePeriod) {
|
|
tickCount = 0;
|
|
if (pulseCount < (PPQN-1)) { //-1 is here to avoid extra IF to reset to 0
|
|
pulseCount++;
|
|
} else {
|
|
pulseCount = 0;
|
|
}
|
|
if (bpmModulationRange != 0) {
|
|
calculateBPMTiming();
|
|
}
|
|
}
|
|
}
|
|
|
|
//EXT-B
|
|
if (masterClockMode == 2) {
|
|
if (tickCount >= pulsePeriod && pulseCount < (PPQN-1)) {
|
|
tickCount = 0;
|
|
pulseCount++;
|
|
}
|
|
}
|
|
|
|
//EXT-16
|
|
if (masterClockMode == 3) {
|
|
if (tickCount >= pulsePeriod && pulseCount < ((6 * (extTriggerCount + 1))-1)) { //hardcoded for 24ppqn
|
|
tickCount = 0;
|
|
pulseCount++;
|
|
}
|
|
}
|
|
|
|
// pull low all outputs after set pulse length
|
|
if (tickCount >= PULSE_LENGTH) {
|
|
for (int i = 0; i<6; i++) {
|
|
digitalWrite(outsPins[i], LOW);
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
void externalClock() {
|
|
lastExtPulseTime = newExtPulseTime;
|
|
newExtPulseTime = millis();
|
|
|
|
if (masterClockMode == 1) { // EXT-24
|
|
//reset cycles if there was no pulses for a while
|
|
if ((newExtPulseTime - lastExtPulseTime) > 125) { //125ms is 20bpm
|
|
for (int i = 0; i<6; i++) {
|
|
channelPulseCount[i] = 0;
|
|
}
|
|
}
|
|
|
|
if (!isPlaying) {
|
|
isPlaying = true;
|
|
}
|
|
tickCount = 0; //to make things happen in the main clock function
|
|
if (pulseCount < (PPQN-1)) {
|
|
pulseCount++;
|
|
} else {
|
|
pulseCount = 0;
|
|
}
|
|
|
|
} else if (masterClockMode == 2) { // EXT-Beat
|
|
if (!isPlaying) {
|
|
isPlaying = true;
|
|
}
|
|
if ((newExtPulseTime - lastExtPulseTime) > 3000) { //3000ms is 1/4 at 20bpm
|
|
resetClocks();
|
|
extResetCountdown = 0;
|
|
}
|
|
tickCount = 0;
|
|
pulseCount = 0;
|
|
if (extResetCountdown < 1) { //reset on the second pulse, so that BPM is already calculated correctly
|
|
extResetCountdown++;
|
|
} else if (extResetCountdown == 1) {
|
|
resetClocks();
|
|
extResetCountdown++; //to get out of the loop
|
|
}
|
|
|
|
} else if (masterClockMode == 3) { // EXT-1/16
|
|
if (!isPlaying) {
|
|
isPlaying = true;
|
|
}
|
|
if ((newExtPulseTime - lastExtPulseTime) > 1000) { //750ms is 1/16 at 20bpm
|
|
resetClocks();
|
|
extResetCountdown = 0;
|
|
}
|
|
tickCount = 0;
|
|
if (extTriggerCount == 0) {
|
|
pulseCount = 0;
|
|
}
|
|
|
|
if (extTriggerCount < 2) {
|
|
extTriggerCount++;
|
|
} else {
|
|
extTriggerCount = 0;
|
|
}
|
|
if (extResetCountdown < 4) { //reset on the first beat (5th pulse), so that BPM is already calculated correctly
|
|
extResetCountdown++;
|
|
} else if (extResetCountdown == 4) {
|
|
resetClocks();
|
|
extResetCountdown++; //to get out of the loop
|
|
}
|
|
}
|
|
}
|
|
|
|
void sendTriggers() {
|
|
|
|
for (int i = 0; i<6; i++) {
|
|
if (playingModes[i] != clockModes[channels[i].mode]) {
|
|
needPulseReset[i] = true;
|
|
}
|
|
}
|
|
|
|
//switching modes on the beat and resetting channel clock
|
|
if (pulseCount == 0) {
|
|
calculateCycles();
|
|
for (int i = 0; i<6; i++) {
|
|
if (needPulseReset[i] == true) {
|
|
channelPulseCount[i] = 0;
|
|
needPulseReset[i] = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
//multiplier
|
|
for (int i = 0; i<6; i++) {
|
|
if (channelPulseCount[i] == channels[i].offset) { //Pulse on 0
|
|
if (channels[i].random == 0 || random(10) > channels[i].random) { //random
|
|
digitalWrite(outsPins[i], HIGH);
|
|
}
|
|
}
|
|
if (channelPulseCount[i] < channelPulsesPerCycle[i]) {
|
|
channelPulseCount[i]++;
|
|
} else {
|
|
channelPulseCount[i] = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
void calculateCycles() {
|
|
|
|
for (int i = 0; i<6; i++) {
|
|
if (channels[i].modulationRange == 0) {
|
|
playingModes[i] = clockModes[channels[i].mode];
|
|
} else { //modulation happens here
|
|
int mod;
|
|
if (!channels[i].modulationChannel) {
|
|
mod = a1Input;
|
|
} else {
|
|
mod = a2Input;
|
|
}
|
|
mod = map (mod, 0, 1023, 0, channels[i].modulationRange);
|
|
playingModes[i] = clockModes[channels[i].mode - mod]; //subtracting because the innitiall array is backwards
|
|
}
|
|
|
|
if (playingModes[i] > 0) {
|
|
channelPulsesPerCycle[i] = (playingModes[i] * PPQN) - 1;
|
|
} else {
|
|
channelPulsesPerCycle[i] = (PPQN / abs(playingModes[i])) - 1;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
void calculateBPMTiming() {
|
|
int mod = 0;
|
|
if (masterClockMode == 0) {
|
|
if (bpmModulationRange != 0 && !bpmModulationChannel) {
|
|
mod = map (a1Input, 0, 1023, 0, bpmModulationRange*10);
|
|
} else if (bpmModulationRange != 0 && bpmModulationChannel) {
|
|
mod = map (a2Input, 0, 1023, 0, bpmModulationRange*10);
|
|
}
|
|
pulsePeriod = 60000 / ((bpm + mod) * PPQN);
|
|
|
|
} else if (masterClockMode == 2) { //for external beat clock
|
|
pulsePeriod = (newExtPulseTime - lastExtPulseTime) / PPQN;
|
|
|
|
} else if (masterClockMode == 3) { //for external 1/16 clock
|
|
pulsePeriod = (newExtPulseTime - lastExtPulseTime) / 6; //6 is hardcoded 1/16 at 24ppqn
|
|
}
|
|
}
|
|
|
|
void resetClocks() {
|
|
for (int i = 0; i<6; i++) {
|
|
channelPulseCount[i] = 0;
|
|
digitalWrite(outsPins[i], LOW); //to avoid stuck leds
|
|
}
|
|
pulseCount = 0;
|
|
tickCount = 0;
|
|
}
|
|
|
|
void saveState() {
|
|
int addr = 0;
|
|
EEPROM.put(addr, bpm);
|
|
addr = addr + sizeof(bpm);
|
|
EEPROM.put(addr, bpmModulationChannel);
|
|
addr = addr + sizeof(bpmModulationChannel);
|
|
EEPROM.put(addr, bpmModulationRange);
|
|
addr = addr + sizeof(bpmModulationRange);
|
|
EEPROM.put(addr, masterClockMode);
|
|
addr = addr + sizeof(masterClockMode);
|
|
EEPROM.put(addr, channels);
|
|
}
|
|
|
|
void checkInputs() {
|
|
|
|
//encoder button
|
|
if (!digitalRead(ENC_BTN_PIN) && !encPressRegistered) {
|
|
encPressRegistered = true;
|
|
encPressedTime = millis();
|
|
} else if (digitalRead(ENC_BTN_PIN) && encPressRegistered) {
|
|
encPressRegistered = false;
|
|
encReleasedTime = millis();
|
|
|
|
if (encReleasedTime - encPressedTime < 500) { // press shorter than .5s switches tabs
|
|
if (insideTab == 0) {
|
|
displayTabOld = displayTab;
|
|
displayTab++;
|
|
if (displayTab>6) {
|
|
displayTab = 0;
|
|
}
|
|
} else if (insideTab < 3 && displayTab != 0) {
|
|
insideTab ++;
|
|
} else if (insideTab < 2 && displayTab == 0) {
|
|
insideTab ++;
|
|
} else {
|
|
insideTab = 1;
|
|
}
|
|
updateScreen();
|
|
} else if (encReleasedTime - encPressedTime < 2000) { // longer press (<2s) and switches random mode, longer than 2s presses are ignored
|
|
if (insideTab == 0) {
|
|
insideTab = 1;
|
|
} else {
|
|
insideTab = 0;
|
|
}
|
|
updateScreen();
|
|
}
|
|
}
|
|
|
|
//encoder
|
|
encoder.tick();
|
|
int encPosition = encoder.getPosition();
|
|
if (encPositionOld != encPosition) {
|
|
int change = encPositionOld - encPosition;
|
|
if (displayTab == 0 && insideTab == 0 && masterClockMode == 0) {
|
|
bpm = bpm + change;
|
|
if (bpm > MAXBPM) {
|
|
bpm = MAXBPM;
|
|
} else if (bpm < MINBPM) {
|
|
bpm = MINBPM;
|
|
}
|
|
calculateBPMTiming();
|
|
|
|
} else if (displayTab == 0 && insideTab == 1) { //Clock mode
|
|
masterClockMode = masterClockMode + change;
|
|
if (masterClockMode > 250) {
|
|
masterClockMode = 0;
|
|
} else if (masterClockMode > 3) {
|
|
masterClockMode = 3;
|
|
}
|
|
|
|
} else if (displayTab == 0 && insideTab == 2 && masterClockMode == 0) { //bpm modulation
|
|
bpmModulationRange = bpmModulationRange + change;
|
|
if (bpmModulationRange == 255 && bpmModulationChannel == 0) {
|
|
bpmModulationRange = 0;
|
|
} else if (bpmModulationRange == 255 && bpmModulationChannel == 1) {
|
|
bpmModulationRange = 9;
|
|
bpmModulationChannel = 0;
|
|
} else if (bpmModulationRange > 9 && bpmModulationChannel == 0) {
|
|
bpmModulationRange = 0;
|
|
bpmModulationChannel = 1;
|
|
} else if (bpmModulationRange > 9 && bpmModulationChannel == 1) {
|
|
bpmModulationRange = 9;
|
|
}
|
|
|
|
} else if (displayTab != 0 && insideTab == 0) { //subdivision
|
|
channels[displayTab-1].mode = channels[displayTab-1].mode - change;
|
|
if (channels[displayTab-1].mode == 65535) { //65535 is 0-1 for unsigned vars
|
|
channels[displayTab-1].mode = 0;
|
|
} else if (channels[displayTab-1].mode > (sizeof(clockModes)/sizeof(int)) - 1) {
|
|
channels[displayTab-1].mode = (sizeof(clockModes)/sizeof(int)) - 1;
|
|
}
|
|
if (!isPlaying) {
|
|
calculateCycles();
|
|
}
|
|
|
|
} else if (displayTab != 0 && insideTab == 1) { //random
|
|
channels[displayTab-1].random = channels[displayTab-1].random + change;
|
|
if (channels[displayTab-1].random == 65535) {
|
|
channels[displayTab-1].random = 0;
|
|
} else if (channels[displayTab-1].random > 9) {
|
|
channels[displayTab-1].random = 9;
|
|
}
|
|
|
|
} else if (displayTab != 0 && insideTab == 2) { //modulation
|
|
channels[displayTab-1].modulationRange = channels[displayTab-1].modulationRange + change;
|
|
if (channels[displayTab-1].modulationRange < 0 && channels[displayTab-1].modulationChannel == 0) {
|
|
channels[displayTab-1].modulationRange = 0;
|
|
} else if (channels[displayTab-1].modulationRange > 6 && channels[displayTab-1].modulationChannel == 0) {
|
|
channels[displayTab-1].modulationChannel = 1;
|
|
channels[displayTab-1].modulationRange = 0;
|
|
} else if (channels[displayTab-1].modulationRange < 0 && channels[displayTab-1].modulationChannel == 1) {
|
|
channels[displayTab-1].modulationChannel = 0;
|
|
channels[displayTab-1].modulationRange = 6;
|
|
} else if (channels[displayTab-1].modulationRange > 6 && channels[displayTab-1].modulationChannel == 1) {
|
|
channels[displayTab-1].modulationRange = 6;
|
|
}
|
|
|
|
} else if (displayTab != 0 && insideTab == 3) { //offset
|
|
channels[displayTab-1].offset = channels[displayTab-1].offset + change;
|
|
if (channels[displayTab-1].offset == 65535) {
|
|
channels[displayTab-1].offset = 0;
|
|
} else if (channels[displayTab-1].offset >= channelPulsesPerCycle[displayTab-1]) {
|
|
channels[displayTab-1].offset = channelPulsesPerCycle[displayTab-1];
|
|
}
|
|
}
|
|
updateScreen();
|
|
encPositionOld = encPosition;
|
|
}
|
|
|
|
//play button
|
|
if (!digitalRead(START_STOP_BTN_PIN) && !playBtnPushed) {
|
|
if (masterClockMode == 0) {
|
|
calculateBPMTiming();
|
|
resetClocks();
|
|
isPlaying = !isPlaying;
|
|
}
|
|
playBtnPushed = true;
|
|
saveState();
|
|
updateScreen(); //to wake up the screen if turned off
|
|
} 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();
|
|
|
|
lastInteractionTime = millis(); //not sure if it's a right place for this, but should do for now
|
|
|
|
//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.fillRect(108, 2, 20, 8, SSD1306_WHITE);
|
|
display.println();
|
|
display.println();
|
|
display.fillRect(0, 10, 128, 2, SSD1306_WHITE);
|
|
|
|
//Content
|
|
display.setCursor(4,16);
|
|
display.setTextSize(3);
|
|
display.setTextColor(SSD1306_WHITE);
|
|
if (displayTab == 0 && masterClockMode == 0) {
|
|
if (bpm<100) {
|
|
display.print(" ");
|
|
}
|
|
display.println(bpm);
|
|
display.setCursor(21,40);
|
|
display.setTextSize(2);
|
|
display.println(F("bpm"));
|
|
} else if (displayTab == 0 && masterClockMode == 1) {
|
|
display.println(F(" 24"));
|
|
display.setCursor(8,40);
|
|
display.setTextSize(2);
|
|
display.println(F("PPQN"));
|
|
} else if (displayTab == 0 && masterClockMode == 2) {
|
|
display.setCursor(8,16);
|
|
display.setTextSize(2);
|
|
display.println(F("BEAT"));
|
|
} else if (displayTab == 0 && masterClockMode == 3) {
|
|
display.setCursor(8,16);
|
|
display.setTextSize(2);
|
|
display.println(F("1/16"));
|
|
} else {
|
|
if (clockModes[channels[displayTab-1].mode] == 0) {
|
|
display.print(F("OFF"));
|
|
} else if (clockModes[channels[displayTab-1].mode]>0) {
|
|
display.print(F("/"));
|
|
display.print(abs(clockModes[channels[displayTab-1].mode]));
|
|
} else {
|
|
display.print(F("x"));
|
|
display.print(abs(clockModes[channels[displayTab-1].mode]));
|
|
}
|
|
}
|
|
display.println();
|
|
display.setTextSize(1);
|
|
display.println();
|
|
|
|
//Extra params
|
|
display.setCursor(58,16);
|
|
display.setTextSize(1);
|
|
if (displayTab == 0) {
|
|
if (insideTab == 1) {
|
|
display.setTextColor(SSD1306_BLACK, SSD1306_WHITE);
|
|
} else {
|
|
display.setTextColor(SSD1306_WHITE);
|
|
}
|
|
display.print(F(" CLK:"));
|
|
if (masterClockMode == 1) {
|
|
display.println(F("EXT24 "));
|
|
} else if (masterClockMode == 2) {
|
|
display.println(F("EXT-B "));
|
|
} else if (masterClockMode == 3) {
|
|
display.println(F("EXT16 "));
|
|
} else {
|
|
display.println(F("INT "));
|
|
}
|
|
if (insideTab == 2) {
|
|
display.setTextColor(SSD1306_BLACK, SSD1306_WHITE);
|
|
} else {
|
|
display.setTextColor(SSD1306_WHITE);
|
|
}
|
|
display.setCursor(58,30);
|
|
display.print(F(" MOD:"));
|
|
if (bpmModulationRange != 0 && masterClockMode == 0) {
|
|
display.print(F("CV"));
|
|
display.print(bpmModulationChannel+1);
|
|
display.print("+");
|
|
display.print(bpmModulationRange * 10);
|
|
} else {
|
|
display.print(F("Off "));
|
|
}
|
|
} else {
|
|
if (insideTab == 1) {
|
|
display.setTextColor(SSD1306_BLACK, SSD1306_WHITE);
|
|
} else {
|
|
display.setTextColor(SSD1306_WHITE);
|
|
}
|
|
display.print(F(" RND:"));
|
|
if (channels[displayTab-1].random > 0) {
|
|
display.print(channels[displayTab-1].random);
|
|
display.print(F("0% "));
|
|
} else {
|
|
display.print(F("Off "));
|
|
}
|
|
if (insideTab == 2) {
|
|
display.setTextColor(SSD1306_BLACK, SSD1306_WHITE);
|
|
} else {
|
|
display.setTextColor(SSD1306_WHITE);
|
|
}
|
|
display.setCursor(58,30);
|
|
display.print(F(" MOD:"));
|
|
if (channels[displayTab-1].modulationChannel && channels[displayTab-1].modulationRange != 0) {
|
|
display.print(F("CV2"));
|
|
if (channels[displayTab-1].modulationRange > 0) {
|
|
display.print(F("+"));
|
|
}
|
|
display.print(channels[displayTab-1].modulationRange);
|
|
display.print(F(" "));
|
|
} else if (!channels[displayTab-1].modulationChannel && channels[displayTab-1].modulationRange != 0) {
|
|
display.print(F("CV1"));
|
|
if (channels[displayTab-1].modulationRange > 0) {
|
|
display.print(F("+"));
|
|
}
|
|
display.print(channels[displayTab-1].modulationRange);
|
|
display.print(F(" "));
|
|
} else {
|
|
display.print(F("Off "));
|
|
}
|
|
if (insideTab == 3) {
|
|
display.setTextColor(SSD1306_BLACK, SSD1306_WHITE);
|
|
} else {
|
|
display.setTextColor(SSD1306_WHITE);
|
|
}
|
|
display.setCursor(58,44);
|
|
display.print(F(" OFT:"));
|
|
display.print(channels[displayTab-1].offset);
|
|
display.print(F("/"));
|
|
display.print(channelPulsesPerCycle[displayTab-1]+1);
|
|
display.print(F(" "));
|
|
}
|
|
|
|
display.display();
|
|
} |