282 lines
6.4 KiB
Arduino
282 lines
6.4 KiB
Arduino
|
#include <Audio.h>
|
||
|
#include <Wire.h>
|
||
|
#include <Adafruit_NeoTrellisM4.h>
|
||
|
#include <SFE_MicroOLED.h>
|
||
|
#include <Adafruit_MPR121.h>
|
||
|
#include "synth.h"
|
||
|
#include "patches.h"
|
||
|
#include "notes.h"
|
||
|
#include "fingering.h"
|
||
|
|
||
|
#define KNEE_OFFSET 0
|
||
|
#define KEY_OFFSET 2
|
||
|
|
||
|
float cmaj_low[8] = { 130.81, 146.83, 164.81, 174.61, 196.00, 220.00, 246.94, 261.63 };
|
||
|
float cmaj_high[8] = { 261.6, 293.7, 329.6, 349.2, 392.0, 440.0, 493.9, 523.3 };
|
||
|
|
||
|
AudioEffectEnvelope *envs[] = {&env1, &env2, &env3, &env4};
|
||
|
AudioSynthWaveformSineModulated *oscs[] = {&osc1, &osc2, &osc3, &osc4};
|
||
|
|
||
|
int currentPatch = 0;
|
||
|
|
||
|
Adafruit_MPR121 cap = Adafruit_MPR121();
|
||
|
Adafruit_NeoTrellisM4 trellis = Adafruit_NeoTrellisM4();
|
||
|
MicroOLED oled(9, 1);
|
||
|
|
||
|
// Hat tip to Kyle Gann
|
||
|
// https://www.kylegann.com/tuning.html
|
||
|
float JustPitches[MaxNote + 1];
|
||
|
void setupJustPitches(uint8_t baseNote, float basePitch) {
|
||
|
JustPitches[baseNote + 0] = basePitch * 1 / 1;
|
||
|
JustPitches[baseNote + 1] = basePitch * 16 / 15;
|
||
|
JustPitches[baseNote + 2] = basePitch * 9 / 8;
|
||
|
JustPitches[baseNote + 3] = basePitch * 6 / 5;
|
||
|
JustPitches[baseNote + 4] = basePitch * 5 / 4;
|
||
|
JustPitches[baseNote + 5] = basePitch * 4 / 3;
|
||
|
JustPitches[baseNote + 6] = basePitch * 45 / 32;
|
||
|
JustPitches[baseNote + 7] = basePitch * 3 / 2;
|
||
|
JustPitches[baseNote + 8] = basePitch * 8 / 5;
|
||
|
JustPitches[baseNote + 9] = basePitch * 5 / 3;
|
||
|
JustPitches[baseNote + 10] = basePitch * 9 / 5;
|
||
|
JustPitches[baseNote + 11] = basePitch * 15 / 8;
|
||
|
|
||
|
// Octaves
|
||
|
for (int note = baseNote; note < baseNote + 12; note++) {
|
||
|
for (int i = 1; i < 9; i++) {
|
||
|
int multiplier = 1<<i;
|
||
|
int shift = i*12;
|
||
|
int upNote = note + shift;
|
||
|
int dnNote = note - shift;
|
||
|
|
||
|
if (upNote <= MaxNote) {
|
||
|
JustPitches[upNote] = JustPitches[note] * multiplier;
|
||
|
}
|
||
|
if (dnNote >= 0) {
|
||
|
JustPitches[dnNote] = JustPitches[note] / multiplier;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void loadPatch(Patch p) {
|
||
|
oled.clear(PAGE);
|
||
|
oled.setFontType(0);
|
||
|
oled.setCursor(0, 0);
|
||
|
oled.print(p.name);
|
||
|
oled.display();
|
||
|
|
||
|
for (int i=0; i<4; i++) {
|
||
|
Operator op = p.operators[i];
|
||
|
|
||
|
oscs[i]->amplitude(op.gain);
|
||
|
envs[i]->delay(op.delay);
|
||
|
envs[i]->attack(op.attack);
|
||
|
envs[i]->hold(op.hold);
|
||
|
envs[i]->decay(op.decay);
|
||
|
envs[i]->sustain(op.sustain);
|
||
|
envs[i]->release(op.release);
|
||
|
}
|
||
|
feedback.gain(0, p.feedback);
|
||
|
}
|
||
|
|
||
|
void setup(){
|
||
|
setupJustPitches(NOTE_D4, PITCH_D4);
|
||
|
pinMode(LED_BUILTIN, OUTPUT);
|
||
|
|
||
|
// Wire.begin needs a moment
|
||
|
delay(100);
|
||
|
Wire.begin();
|
||
|
|
||
|
// Initialize OLED display
|
||
|
oled.begin();
|
||
|
oled.clear(ALL);
|
||
|
oled.display();
|
||
|
|
||
|
// Initialize touch sensor
|
||
|
if (!cap.begin(0x5A)) {
|
||
|
oled.clear(PAGE);
|
||
|
oled.print("Can't find MPR121");
|
||
|
oled.display();
|
||
|
|
||
|
bool on = HIGH;
|
||
|
while (1) {
|
||
|
digitalWrite(LED_BUILTIN, on);
|
||
|
on = ! on;
|
||
|
delay(200);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Initialize the Trellis
|
||
|
trellis.begin();
|
||
|
trellis.setBrightness(20);
|
||
|
|
||
|
AudioMemory(120);
|
||
|
|
||
|
// Turn it down, man
|
||
|
mixL.gain(0, 0.05);
|
||
|
mixR.gain(0, 0.05);
|
||
|
|
||
|
// Set up high shelf filter, for vibrato effects
|
||
|
biquad1.setHighShelf(0, 1100, 1.0, 1);
|
||
|
|
||
|
// load the patch
|
||
|
loadPatch(Bank[0]);
|
||
|
|
||
|
// Initialize processor and memory measurements
|
||
|
AudioProcessorUsageMaxReset();
|
||
|
AudioMemoryUsageMaxReset();
|
||
|
}
|
||
|
|
||
|
void wtfSetPitch(float freq) {
|
||
|
for (int i=0; i<4; i++) {
|
||
|
Operator op = Bank[currentPatch].operators[i];
|
||
|
oscs[i]->frequency(op.baseFrequency + freq*op.multiplier);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void setPitch(float freq) {
|
||
|
for (int i=0; i<4; i++) {
|
||
|
Operator op = Bank[currentPatch].operators[i];
|
||
|
oscs[i]->frequency(op.baseFrequency + freq*op.multiplier);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void noteOn(float freq) {
|
||
|
AudioNoInterrupts();
|
||
|
for (int i=0; i<4; i++) {
|
||
|
Operator op = Bank[currentPatch].operators[i];
|
||
|
oscs[i]->frequency(op.baseFrequency + freq*op.multiplier);
|
||
|
envs[i]->noteOn();
|
||
|
}
|
||
|
AudioInterrupts();
|
||
|
}
|
||
|
|
||
|
void noteOff() {
|
||
|
AudioNoInterrupts();
|
||
|
for (int i=0; i<4; i++) {
|
||
|
envs[i]->noteOff();
|
||
|
}
|
||
|
AudioInterrupts();
|
||
|
}
|
||
|
|
||
|
void trellisLoop() {
|
||
|
trellis.tick();
|
||
|
|
||
|
while (trellis.available()) {
|
||
|
keypadEvent e = trellis.read();
|
||
|
int keyindex = e.bit.KEY;
|
||
|
if (e.bit.EVENT == KEY_JUST_PRESSED) {
|
||
|
trellis.setPixelColor(keyindex, 0x600000);
|
||
|
noteOn(JustPitches[NOTE_D3 + keyindex]);
|
||
|
} else if (e.bit.EVENT == KEY_JUST_RELEASED){
|
||
|
noteOff();
|
||
|
trellis.setPixelColor(keyindex, 0);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#define CLOSEDVAL 0x30
|
||
|
#define OPENVAL 0x70
|
||
|
|
||
|
bool playing = false;
|
||
|
|
||
|
void loop() {
|
||
|
uint8_t keys = 0;
|
||
|
uint8_t note;
|
||
|
uint8_t gkeys = 0;
|
||
|
uint8_t gnote;
|
||
|
uint8_t gaffinity = 0;
|
||
|
bool knee = true;
|
||
|
bool bag = false;
|
||
|
bool silent = false;
|
||
|
|
||
|
trellisLoop();
|
||
|
|
||
|
oled.clear(PAGE);
|
||
|
for (int i = 0; i < 8; i++) {
|
||
|
uint16_t val = cap.filteredData(i+KEY_OFFSET);
|
||
|
uint8_t c = 0;
|
||
|
if (val < OPENVAL) {
|
||
|
bitSet(keys, i);
|
||
|
c = 7;
|
||
|
|
||
|
// If they're just sort of touching it, we're doing a glissando!
|
||
|
if (val > CLOSEDVAL) {
|
||
|
int aff = val - CLOSEDVAL;
|
||
|
|
||
|
gaffinity = max(gaffinity, aff);
|
||
|
c = 7 - (7 * aff / (OPENVAL - CLOSEDVAL));
|
||
|
} else {
|
||
|
bitSet(gkeys, i);
|
||
|
}
|
||
|
}
|
||
|
// print key states
|
||
|
oled.rectFill(32 - 4*i, 40, 3, c);
|
||
|
}
|
||
|
|
||
|
note = uilleann_matrix[keys];
|
||
|
gnote = uilleann_matrix[gkeys];
|
||
|
|
||
|
knee = cap.filteredData(KNEE_OFFSET) < CLOSEDVAL;
|
||
|
bool alt = note & 0x80;
|
||
|
bool galt = gnote & 0x80;
|
||
|
note = note & 0x7f;
|
||
|
gnote = gnote & 0x7f;
|
||
|
|
||
|
// All keys closed + knee = no sound
|
||
|
if (knee) {
|
||
|
oled.rectFill(36, 42, 3, 3);
|
||
|
if (keys == 0xff) {
|
||
|
silent = true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Jump octave if the bag is squished
|
||
|
//bag = !digitalRead(BAG);
|
||
|
oled.setCursor(9*6, 10);
|
||
|
if (bag) {
|
||
|
oled.print("^");
|
||
|
if (keys & bit(7)) {
|
||
|
note += 12;
|
||
|
gnote += 12;
|
||
|
}
|
||
|
} else {
|
||
|
oled.print(" ");
|
||
|
}
|
||
|
|
||
|
if (silent) {
|
||
|
noteOff();
|
||
|
playing = false;
|
||
|
} else {
|
||
|
// Calculate pitch, and glissando pitch
|
||
|
uint16_t pitch = JustPitches[note];
|
||
|
uint16_t gpitch = JustPitches[gnote];
|
||
|
|
||
|
if (alt) {
|
||
|
biquad1.setHighShelf(0, 1000, 0.5, 1);
|
||
|
} else {
|
||
|
biquad1.setHighShelf(0, 1000, 1.0, 1);
|
||
|
}
|
||
|
|
||
|
// Bend pitch
|
||
|
if (gaffinity && (abs(gnote - note) < 3)) {
|
||
|
uint32_t sum = (pitch * (OPENVAL-CLOSEDVAL)) + (gpitch * gaffinity);
|
||
|
pitch = sum / ((OPENVAL-CLOSEDVAL) + gaffinity);
|
||
|
}
|
||
|
|
||
|
if (playing) {
|
||
|
setPitch(pitch);
|
||
|
} else {
|
||
|
noteOn(pitch);
|
||
|
}
|
||
|
playing = true;
|
||
|
}
|
||
|
|
||
|
// Print status
|
||
|
oled.setCursor(0*6, 20);
|
||
|
oled.print(NoteNames[note % 12]);
|
||
|
oled.print(alt?".":" ");
|
||
|
|
||
|
//oled.display();
|
||
|
}
|