|
|
@ -17,16 +17,20 @@ |
|
|
|
#include <SoftwareSerial.h> |
|
|
|
#include <SoftwareSerial.h> |
|
|
|
#include <LiquidCrystal_I2C.h> |
|
|
|
#include <LiquidCrystal_I2C.h> |
|
|
|
#include <looper.h> |
|
|
|
#include <looper.h> |
|
|
|
|
|
|
|
#include <SPIFFS.h> |
|
|
|
|
|
|
|
#include <ESPFlash.h> |
|
|
|
|
|
|
|
|
|
|
|
#define MDNS_NAME "osc2midi" |
|
|
|
#define MDNS_NAME "osc2midi" |
|
|
|
#define AP_SSID_NAME "OSC2MIDI" |
|
|
|
#define AP_SSID_NAME "OSC2MIDI" |
|
|
|
#define AP_PASSWORD "osc2midi" |
|
|
|
#define AP_PASSWORD "osc2midi" |
|
|
|
#define AP_SSID_CONFIG_NAME "OSC2MIDI-Config" |
|
|
|
#define AP_SSID_CONFIG_NAME "OSC2MIDI-Config" |
|
|
|
#define AP_CONFIG_PASSWORD "osc2midi" |
|
|
|
#define AP_CONFIG_PASSWORD "osc2midi" |
|
|
|
|
|
|
|
#define WIFI_CONNECT_TIMEOUT 30 |
|
|
|
|
|
|
|
#define CFG_PORTAL_TIMEOUT 90 |
|
|
|
#define SOFT_SERIAL_RX 18 |
|
|
|
#define SOFT_SERIAL_RX 18 |
|
|
|
#define SOFT_SERIAL_TX 19 |
|
|
|
#define SOFT_SERIAL_TX 19 |
|
|
|
#define AP_DATA_RESET_PIN 35 |
|
|
|
#define AP_DATA_RESET_PIN 25 |
|
|
|
#define AP_MODE_PIN 34 |
|
|
|
#define AP_MODE_PIN 26 |
|
|
|
#define LCD_I2C_ADDR 0x27 |
|
|
|
#define LCD_I2C_ADDR 0x27 |
|
|
|
#define LCD_COL 16 |
|
|
|
#define LCD_COL 16 |
|
|
|
#define LCD_ROW 2 |
|
|
|
#define LCD_ROW 2 |
|
|
@ -34,13 +38,17 @@ |
|
|
|
#define UDP_SEND_PORT 9000 |
|
|
|
#define UDP_SEND_PORT 9000 |
|
|
|
#define KRATE_MODE 200 |
|
|
|
#define KRATE_MODE 200 |
|
|
|
#define KRATE_STATE 2000 |
|
|
|
#define KRATE_STATE 2000 |
|
|
|
|
|
|
|
#define KRATE_RESET_AP_DATA 5000 |
|
|
|
|
|
|
|
#define KRATE_CHECK_WRITE_STATE 10000 |
|
|
|
|
|
|
|
#define LAST_USAGE_TIMER 5000 |
|
|
|
|
|
|
|
#define FORMAT_SPIFFS_IF_FAILED true |
|
|
|
|
|
|
|
|
|
|
|
void OSCToMidiCC(OSCMessage &msg, int offset); |
|
|
|
void OSCToMidiCC(OSCMessage &msg, int offset); |
|
|
|
void MidiCCToOSC(uint8_t channel, uint8_t number, uint8_t value); |
|
|
|
void MidiCCToOSC(uint8_t channel, uint8_t number, uint8_t value); |
|
|
|
void change_midi_state(uint8_t midichannel, uint8_t cc, uint8_t value); |
|
|
|
/*void change_midi_state(uint8_t midichannel, uint8_t cc, uint8_t value);
|
|
|
|
void show_midi_state(void); |
|
|
|
void show_midi_state(void); |
|
|
|
void set_midi_state(void); |
|
|
|
void set_midi_state(void); |
|
|
|
void check_mode(void); |
|
|
|
void check_mode(void);*/ |
|
|
|
|
|
|
|
|
|
|
|
WiFiUDP udp; |
|
|
|
WiFiUDP udp; |
|
|
|
IPAddress clientIP; |
|
|
|
IPAddress clientIP; |
|
|
@ -54,19 +62,29 @@ HardwareSerial midi1(2); // RX: 16, TX: 17 |
|
|
|
//#define TX (1)
|
|
|
|
//#define TX (1)
|
|
|
|
#endif |
|
|
|
#endif |
|
|
|
SoftwareSerial midi2; |
|
|
|
SoftwareSerial midi2; |
|
|
|
bool ap_mode_state = digitalRead(AP_MODE_PIN); |
|
|
|
bool ap_mode_state; |
|
|
|
int8_t state[16][128]; |
|
|
|
int8_t midistate[16 * 128]; |
|
|
|
|
|
|
|
bool last_reset_ap_check = false; |
|
|
|
looper sched; |
|
|
|
looper sched; |
|
|
|
|
|
|
|
bool write_state = false; |
|
|
|
|
|
|
|
uint32_t last_usage = millis(); |
|
|
|
|
|
|
|
|
|
|
|
MIDI_CREATE_INSTANCE(HardwareSerial, midi1, MIDI1); |
|
|
|
MIDI_CREATE_INSTANCE(HardwareSerial, midi1, MIDI1); |
|
|
|
|
|
|
|
|
|
|
|
void setup() |
|
|
|
void setup() |
|
|
|
{ |
|
|
|
{ |
|
|
|
pinMode(AP_DATA_RESET_PIN, INPUT_PULLUP); |
|
|
|
|
|
|
|
pinMode(AP_MODE_PIN, INPUT_PULLUP); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Serial.begin(115200); |
|
|
|
Serial.begin(115200); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEBUG_MSG("<SETUP>\n"); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//Serial.print("FORMAT SPIFFS..."); SPIFFS.format(); Serial.println("done."); while (1);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
memset(midistate, -1, 16 * 128); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
pinMode(AP_DATA_RESET_PIN, INPUT_PULLDOWN); |
|
|
|
|
|
|
|
pinMode(AP_MODE_PIN, INPUT_PULLDOWN); |
|
|
|
|
|
|
|
ap_mode_state = digitalRead(AP_MODE_PIN); |
|
|
|
|
|
|
|
|
|
|
|
Serial.setDebugOutput(true); |
|
|
|
Serial.setDebugOutput(true); |
|
|
|
Serial.println(F("OSC2MIDI (c)2020 H. Wirtz <wirtz@parasitstudio.de>")); |
|
|
|
Serial.println(F("OSC2MIDI (c)2020 H. Wirtz <wirtz@parasitstudio.de>")); |
|
|
|
|
|
|
|
|
|
|
@ -80,6 +98,20 @@ void setup() |
|
|
|
lcd.print(F("(c)parasiTstudio")); |
|
|
|
lcd.print(F("(c)parasiTstudio")); |
|
|
|
delay(1000); |
|
|
|
delay(1000); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
WiFi.mode(WIFI_STA); // explicitly set mode, esp defaults to STA+AP
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!SPIFFS.begin(FORMAT_SPIFFS_IF_FAILED)) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
DEBUG_MSG("An Error has occurred while mounting SPIFFS!"); |
|
|
|
|
|
|
|
lcd.clear(); |
|
|
|
|
|
|
|
lcd.setCursor(0, 0); |
|
|
|
|
|
|
|
lcd.print(F("SPIFFS ERROR")); |
|
|
|
|
|
|
|
lcd.setCursor(0, 1); |
|
|
|
|
|
|
|
lcd.print(F("Restarting...")); |
|
|
|
|
|
|
|
delay(1000); |
|
|
|
|
|
|
|
ESP.restart(); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
if (ap_mode_state == LOW) |
|
|
|
if (ap_mode_state == LOW) |
|
|
|
{ |
|
|
|
{ |
|
|
|
DEBUG_MSG("Mode Access-Point\n"); |
|
|
|
DEBUG_MSG("Mode Access-Point\n"); |
|
|
@ -108,31 +140,18 @@ void setup() |
|
|
|
{ |
|
|
|
{ |
|
|
|
DEBUG_MSG("Mode Client\n"); |
|
|
|
DEBUG_MSG("Mode Client\n"); |
|
|
|
|
|
|
|
|
|
|
|
WiFi.mode(WIFI_STA); // explicitly set mode, esp defaults to STA+AP
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
WiFiManager wm; |
|
|
|
WiFiManager wm; |
|
|
|
|
|
|
|
|
|
|
|
if (digitalRead(AP_DATA_RESET_PIN) != LOW) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
wm.resetSettings(); |
|
|
|
|
|
|
|
DEBUG_MSG("Resetting AP data\n"); |
|
|
|
|
|
|
|
lcd.clear(); |
|
|
|
|
|
|
|
lcd.setCursor(0, 0); |
|
|
|
|
|
|
|
lcd.print(F("Resetting AP data")); |
|
|
|
|
|
|
|
delay(2000); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Automatically connect using saved credentials,
|
|
|
|
|
|
|
|
// if connection fails, it starts an access point with the specified name ( "AutoConnectAP"),
|
|
|
|
|
|
|
|
// if empty will auto generate SSID, if password is blank it will be anonymous AP (wm.autoConnect())
|
|
|
|
|
|
|
|
// then goes into a blocking loop awaiting configuration and will return success result
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
lcd.clear(); |
|
|
|
lcd.clear(); |
|
|
|
lcd.setCursor(0, 0); |
|
|
|
lcd.setCursor(0, 0); |
|
|
|
lcd.print(F("Mode Config-AP")); |
|
|
|
lcd.print(F("Mode Config-AP")); |
|
|
|
lcd.setCursor(0, 1); |
|
|
|
lcd.setCursor(0, 1); |
|
|
|
lcd.print(F("192.168.4.1")); |
|
|
|
lcd.print(F("192.168.4.1")); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
wm.setConnectTimeout(WIFI_CONNECT_TIMEOUT); |
|
|
|
|
|
|
|
wm.setConfigPortalTimeout(CFG_PORTAL_TIMEOUT); |
|
|
|
|
|
|
|
wm.setBreakAfterConfig(true); |
|
|
|
|
|
|
|
|
|
|
|
if (!wm.autoConnect(AP_SSID_CONFIG_NAME, AP_CONFIG_PASSWORD)) |
|
|
|
if (!wm.autoConnect(AP_SSID_CONFIG_NAME, AP_CONFIG_PASSWORD)) |
|
|
|
{ |
|
|
|
{ |
|
|
|
DEBUG_MSG("Failed to connect\n"); |
|
|
|
DEBUG_MSG("Failed to connect\n"); |
|
|
@ -163,8 +182,6 @@ void setup() |
|
|
|
lcd.print(WiFi.localIP()); |
|
|
|
lcd.print(WiFi.localIP()); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
memset(state, -1, 16 * 128); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
udp.begin(UDP_RECV_PORT); |
|
|
|
udp.begin(UDP_RECV_PORT); |
|
|
|
DEBUG_MSG("Listening for UDP packets on port %d\n", UDP_RECV_PORT); |
|
|
|
DEBUG_MSG("Listening for UDP packets on port %d\n", UDP_RECV_PORT); |
|
|
|
|
|
|
|
|
|
|
@ -176,10 +193,16 @@ void setup() |
|
|
|
MIDI1.setHandleControlChange(MidiCCToOSC); |
|
|
|
MIDI1.setHandleControlChange(MidiCCToOSC); |
|
|
|
MIDI1.turnThruOff(); |
|
|
|
MIDI1.turnThruOff(); |
|
|
|
|
|
|
|
|
|
|
|
//set_midi_state();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sched.addJob(check_mode, KRATE_MODE); |
|
|
|
sched.addJob(check_mode, KRATE_MODE); |
|
|
|
sched.addJob(show_midi_state, KRATE_STATE); |
|
|
|
sched.addJob(check_reset_ap_data, KRATE_RESET_AP_DATA); |
|
|
|
|
|
|
|
sched.addJob(check_write_state, KRATE_CHECK_WRITE_STATE); |
|
|
|
|
|
|
|
//sched.addJob(show_midi_state, KRATE_STATE);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
listDir(SPIFFS, "/", 1); |
|
|
|
|
|
|
|
//read_midistate(1);
|
|
|
|
|
|
|
|
show_midi_state(); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEBUG_MSG("<LOOP>\n"); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void loop() |
|
|
|
void loop() |
|
|
@ -190,10 +213,15 @@ void loop() |
|
|
|
|
|
|
|
|
|
|
|
size_t size = udp.parsePacket(); |
|
|
|
size_t size = udp.parsePacket(); |
|
|
|
|
|
|
|
|
|
|
|
while (udp.available()) |
|
|
|
if (udp.available()) |
|
|
|
{ |
|
|
|
{ |
|
|
|
IPAddress tmpIP = udp.remoteIP(); |
|
|
|
IPAddress tmpIP = udp.remoteIP(); |
|
|
|
|
|
|
|
// Keep track of the client IP address for "talking back"
|
|
|
|
|
|
|
|
if (clientIP != tmpIP) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
clientIP = tmpIP; |
|
|
|
|
|
|
|
DEBUG_MSG("New connection from: %d.%d.%d.%d\n", clientIP[0],clientIP[1],clientIP[2],clientIP[3]); |
|
|
|
|
|
|
|
} |
|
|
|
// Check if there are any OSC packets to handle
|
|
|
|
// Check if there are any OSC packets to handle
|
|
|
|
udp.read(buffer, size); |
|
|
|
udp.read(buffer, size); |
|
|
|
msg.fill(buffer, size); |
|
|
|
msg.fill(buffer, size); |
|
|
@ -202,6 +230,7 @@ void loop() |
|
|
|
{ |
|
|
|
{ |
|
|
|
DEBUG_OSC_MESSAGE(msg); |
|
|
|
DEBUG_OSC_MESSAGE(msg); |
|
|
|
msg.route("/midi/cc", OSCToMidiCC); |
|
|
|
msg.route("/midi/cc", OSCToMidiCC); |
|
|
|
|
|
|
|
msg.route("/ping", ping); |
|
|
|
//msg.route("/midi/sysex", OSCToMidiSYSEX);
|
|
|
|
//msg.route("/midi/sysex", OSCToMidiSYSEX);
|
|
|
|
//msg.route("/midi/note", OSCToMidiNote);
|
|
|
|
//msg.route("/midi/note", OSCToMidiNote);
|
|
|
|
} |
|
|
|
} |
|
|
@ -209,13 +238,6 @@ void loop() |
|
|
|
{ |
|
|
|
{ |
|
|
|
DEBUG_MSG("Error parsing OSC message: %d\n", msg.getError()); |
|
|
|
DEBUG_MSG("Error parsing OSC message: %d\n", msg.getError()); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// Keep track of the client IP address for "talking back"
|
|
|
|
|
|
|
|
if (clientIP == tmpIP) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
clientIP = tmpIP; |
|
|
|
|
|
|
|
DEBUG_MSG("Connection from: %s", clientIP); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// Check if there are any CC messages from synth itself
|
|
|
|
// Check if there are any CC messages from synth itself
|
|
|
@ -269,13 +291,13 @@ void OSCToMidiCC(OSCMessage & msg, int offset) |
|
|
|
// XY pad, two values
|
|
|
|
// XY pad, two values
|
|
|
|
cc = getVar(address, 1); |
|
|
|
cc = getVar(address, 1); |
|
|
|
value = round(msg.getFloat(0)); |
|
|
|
value = round(msg.getFloat(0)); |
|
|
|
value = value > 127 ? 127 : value; |
|
|
|
value = constrain(value, 0, 127); |
|
|
|
DEBUG_MSG("MSG: % s\tChannel: % u\t\tCC: % u\tValue: % u\n", address, midichannel, cc, value); |
|
|
|
DEBUG_MSG("MSG: % s\tChannel: % u\t\tCC: % u\tValue: % u\n", address, midichannel, cc, value); |
|
|
|
MIDI1.sendControlChange(cc, value, midichannel); |
|
|
|
MIDI1.sendControlChange(cc, value, midichannel); |
|
|
|
change_midi_state(midichannel, cc, value); |
|
|
|
change_midi_state(midichannel, cc, value); |
|
|
|
cc = getVar(address, 2); |
|
|
|
cc = getVar(address, 2); |
|
|
|
value = round(msg.getFloat(1)); |
|
|
|
value = round(msg.getFloat(1)); |
|
|
|
value = value > 127 ? 127 : value; |
|
|
|
value = constrain(value, 0, 127); |
|
|
|
DEBUG_MSG("MSG: % s\tChannel: % u\t\tCC: % u\tValue: % u\n", address, midichannel, cc, value); |
|
|
|
DEBUG_MSG("MSG: % s\tChannel: % u\t\tCC: % u\tValue: % u\n", address, midichannel, cc, value); |
|
|
|
MIDI1.sendControlChange(cc, value, midichannel); |
|
|
|
MIDI1.sendControlChange(cc, value, midichannel); |
|
|
|
change_midi_state(midichannel, cc, value); |
|
|
|
change_midi_state(midichannel, cc, value); |
|
|
@ -290,9 +312,17 @@ void MidiCCToOSC(uint8_t channel, uint8_t number, uint8_t val) |
|
|
|
{ |
|
|
|
{ |
|
|
|
char buffer[1024]; |
|
|
|
char buffer[1024]; |
|
|
|
|
|
|
|
|
|
|
|
snprintf(buffer, sizeof(buffer), " / midi / cc / % u / % u", channel, number); |
|
|
|
if (channel < 1 && channel > 16) |
|
|
|
|
|
|
|
return; |
|
|
|
|
|
|
|
if (number > 127) |
|
|
|
|
|
|
|
return; |
|
|
|
|
|
|
|
val = constrain(val, 0, 127); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
snprintf(buffer, sizeof(buffer), "/midi/cc/%d/%d", channel, number); |
|
|
|
|
|
|
|
|
|
|
|
DEBUG_MSG("MidiCCToOsc: % s % f\n", buffer, val * 1.0); |
|
|
|
DEBUG_MSG("MidiCCToOsc: %s %f\n", buffer, val); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
change_midi_state(channel, number, val); |
|
|
|
|
|
|
|
|
|
|
|
if (clientIP) |
|
|
|
if (clientIP) |
|
|
|
{ |
|
|
|
{ |
|
|
@ -302,12 +332,26 @@ void MidiCCToOSC(uint8_t channel, uint8_t number, uint8_t val) |
|
|
|
udp.beginPacket(clientIP, UDP_SEND_PORT); |
|
|
|
udp.beginPacket(clientIP, UDP_SEND_PORT); |
|
|
|
msg.send(udp); |
|
|
|
msg.send(udp); |
|
|
|
udp.endPacket(); |
|
|
|
udp.endPacket(); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
msg.empty(); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
else |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
DEBUG_MSG("No client IP.\n"); |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void change_midi_state(uint8_t midichannel, uint8_t cc, uint8_t value) |
|
|
|
void change_midi_state(uint8_t midichannel, uint8_t cc, uint8_t value) |
|
|
|
{ |
|
|
|
{ |
|
|
|
state[midichannel - 1][cc - 1] = value; |
|
|
|
last_usage = millis(); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEBUG_MSG("Setting internal state of MIDI Channel %2d CC#%02d to %d\n", midichannel, cc, int8_t(value)); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (midistate[(midichannel - 1) * 128 + cc - 1] != int8_t(value)) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
midistate[(midichannel - 1) * 128 + cc - 1] = int8_t(value); |
|
|
|
|
|
|
|
write_state = true; |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void show_midi_state(void) |
|
|
|
void show_midi_state(void) |
|
|
@ -316,52 +360,227 @@ void show_midi_state(void) |
|
|
|
|
|
|
|
|
|
|
|
DEBUG_MSG("Current MIDI state:\n"); |
|
|
|
DEBUG_MSG("Current MIDI state:\n"); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
listDir(SPIFFS, "/", 1); |
|
|
|
|
|
|
|
|
|
|
|
for (m = 0; m < 16; m++) |
|
|
|
for (m = 0; m < 16; m++) |
|
|
|
{ |
|
|
|
{ |
|
|
|
DEBUG_MSG("MIDI-Channel %d\n", m + 1); |
|
|
|
DEBUG_MSG("MIDI-Channel %2d\n", m + 1); |
|
|
|
for (c = 0; c < 128; c++) |
|
|
|
for (c = 0; c < 128; c++) |
|
|
|
{ |
|
|
|
{ |
|
|
|
if (state[m][c] >= 0) |
|
|
|
if (midistate[m * 128 + c] >= 0) |
|
|
|
{ |
|
|
|
{ |
|
|
|
DEBUG_MSG("\tCC: %03d = %03d\n", c, state[m][c]); |
|
|
|
DEBUG_MSG("\tCC: %3d = %3d\n", c + 1, midistate[m * 128 + c]); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void check_mode(void) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
if (ap_mode_state != digitalRead(AP_MODE_PIN)) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
lcd.clear(); |
|
|
|
|
|
|
|
lcd.setCursor(0, 0); |
|
|
|
|
|
|
|
lcd.print("Changing mode..."); |
|
|
|
|
|
|
|
delay(500); |
|
|
|
|
|
|
|
ESP.restart(); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
void check_write_state(void) |
|
|
|
void set_midi_state(void) |
|
|
|
|
|
|
|
{ |
|
|
|
{ |
|
|
|
uint8_t tmp; |
|
|
|
DEBUG_MSG("Checking midi state.\n"); |
|
|
|
uint8_t m; |
|
|
|
|
|
|
|
uint16_t i; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEBUG_MSG("Send MIDI state:\n"); |
|
|
|
if (write_state == true && millis() - last_usage > LAST_USAGE_TIMER) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
write_midistate(1); |
|
|
|
|
|
|
|
write_state = false; |
|
|
|
|
|
|
|
show_midi_state(); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
for (m = 0; m < 16; m++); |
|
|
|
void write_midistate(uint8_t setup_number) |
|
|
|
{ |
|
|
|
{ |
|
|
|
if (state[m].size() > 0) |
|
|
|
DEBUG_MSG("Write MIDI state\n"); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for (uint8_t m = 0; m < 16; m++) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
for (uint8_t c = 0; c < 128; c++) |
|
|
|
{ |
|
|
|
{ |
|
|
|
DEBUG_MSG("MIDI-Channel %d\n"); |
|
|
|
if (midistate[m * 128 + c] >= 0) |
|
|
|
for (i = 0; i < state[m].size(); i++) |
|
|
|
|
|
|
|
{ |
|
|
|
{ |
|
|
|
tmp = state[m].get(i); |
|
|
|
DEBUG_MSG("Found MIDI Channel %2d, CC#%2d = %3d\n", m, c, midistate[m * 128 + c]); |
|
|
|
MIDI1.sendControlChange(tmp >> 4, tmp & 0x0f, m); |
|
|
|
|
|
|
|
DEBUG_MSG("\tSet state of CC: %03d = %03d\n", tmp >> 4, tmp & 0x0f); |
|
|
|
char midi_cc_name[33]; |
|
|
|
|
|
|
|
int16_t tmp_val; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sprintf(midi_cc_name, "/%d/midistate/%d/%d", setup_number, m, c); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (SPIFFS.exists(midi_cc_name)) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
File midi_cc = SPIFFS.open(midi_cc_name, "r"); |
|
|
|
|
|
|
|
if (midi_cc) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
tmp_val = midi_cc.read(); |
|
|
|
|
|
|
|
DEBUG_MSG("Data for MIDI Channel %d, CC %d exists: %d\n", m, c, tmp_val); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
close(midi_cc); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
else |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
tmp_val = -1; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (midistate[m * 128 + c] != tmp_val) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
DEBUG_MSG(" Change from %d to %d detected\n", tmp_val, midistate[m * 128 + c]); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
File midi_cc = SPIFFS.open(midi_cc_name, "w"); |
|
|
|
|
|
|
|
if (midi_cc) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
midi_cc.write(midistate[m * 128 + c]); |
|
|
|
|
|
|
|
midi_cc.flush(); |
|
|
|
|
|
|
|
DEBUG_MSG("Wrote %d to %s.\n", midistate[m * 128 + c], midi_cc_name); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
else |
|
|
|
|
|
|
|
DEBUG_MSG("Cannot write to %s.\n", midi_cc_name); |
|
|
|
|
|
|
|
close(midi_cc); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
*/ |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void check_mode(void) |
|
|
|
void read_midistate(uint8_t setup_number) |
|
|
|
{ |
|
|
|
{ |
|
|
|
if (ap_mode_state != digitalRead(AP_MODE_PIN)) |
|
|
|
char setup_dir_name[33]; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEBUG_MSG("Read MIDI state\n"); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sprintf(setup_dir_name, "/%d/midistate", setup_number); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
File midistate_dir = SPIFFS.open(setup_dir_name); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!midistate_dir) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
DEBUG_MSG("Failed to open directory %s.\n", setup_dir_name); |
|
|
|
|
|
|
|
return; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!midistate_dir.isDirectory()) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
DEBUG_MSG("%s is not a directory.\n", setup_dir_name); |
|
|
|
|
|
|
|
return; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
File channel_cc = midistate_dir.openNextFile(); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
while (channel_cc) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
DEBUG_MSG("Trying %s (Size: %d)\n", channel_cc.name(), channel_cc.size()); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (uint8_t(channel_cc.size()) == 1) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
if (!channel_cc.isDirectory()) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
char tmp_name[33]; |
|
|
|
|
|
|
|
uint8_t midi_channel; |
|
|
|
|
|
|
|
uint8_t midi_cc; |
|
|
|
|
|
|
|
uint8_t count = 0; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEBUG_MSG("Using %s\n", channel_cc.name()); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
strcpy(tmp_name, channel_cc.name()); |
|
|
|
|
|
|
|
for (String part = strtok(tmp_name, "/"); part; part = strtok(NULL, "/")) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
count++; |
|
|
|
|
|
|
|
if (count == 3) |
|
|
|
|
|
|
|
midi_channel = atoi(part.c_str()); |
|
|
|
|
|
|
|
else if (count == 4) |
|
|
|
|
|
|
|
midi_cc = atoi(part.c_str()); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
File cc = SPIFFS.open(channel_cc.name(), "r"); |
|
|
|
|
|
|
|
if (cc) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
int8_t val = cc.read(); |
|
|
|
|
|
|
|
DEBUG_MSG(" MIDI-Channel %d CC#%d = %d\n", midi_channel + 1, midi_cc + 1, val); |
|
|
|
|
|
|
|
midistate[midi_channel * 128 + midi_cc] = val; |
|
|
|
|
|
|
|
MidiCCToOSC(midi_channel + 1, midi_cc + 1, val); |
|
|
|
|
|
|
|
cc.close(); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
else |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
DEBUG_MSG("Removing %s: not the right size.\n", channel_cc.name()); |
|
|
|
|
|
|
|
SPIFFS.remove(channel_cc.name()); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
channel_cc = midistate_dir.openNextFile(); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void check_reset_ap_data(void) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
if (digitalRead(AP_DATA_RESET_PIN) == HIGH && last_reset_ap_check == true) |
|
|
|
{ |
|
|
|
{ |
|
|
|
|
|
|
|
DEBUG_MSG("Reset AP data\n"); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
WiFiManager wm; |
|
|
|
|
|
|
|
|
|
|
|
lcd.clear(); |
|
|
|
lcd.clear(); |
|
|
|
lcd.setCursor(0, 0); |
|
|
|
lcd.setCursor(0, 0); |
|
|
|
lcd.print("Changing mode..."); |
|
|
|
lcd.print("Firmware reset"); |
|
|
|
delay(500); |
|
|
|
wm.resetSettings(); |
|
|
|
|
|
|
|
SPIFFS.format(); |
|
|
|
|
|
|
|
lcd.setCursor(0, 1); |
|
|
|
|
|
|
|
lcd.print("Done."); |
|
|
|
|
|
|
|
delay(1000); |
|
|
|
ESP.restart(); |
|
|
|
ESP.restart(); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
else if (digitalRead(AP_DATA_RESET_PIN) == HIGH) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
if (digitalRead(AP_DATA_RESET_PIN) == HIGH) |
|
|
|
|
|
|
|
DEBUG_MSG("Reset AP data pressed\n"); |
|
|
|
|
|
|
|
last_reset_ap_check = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
else |
|
|
|
|
|
|
|
last_reset_ap_check = false; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ping(OSCMessage & msg, int offset) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
if (clientIP) |
|
|
|
|
|
|
|
read_midistate(1); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void listDir(fs::FS & fs, const char * dirname, uint8_t levels) { |
|
|
|
|
|
|
|
Serial.printf("Listing directory: %s\r\n", dirname); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
File root = fs.open(dirname); |
|
|
|
|
|
|
|
if (!root) { |
|
|
|
|
|
|
|
Serial.println("- failed to open directory"); |
|
|
|
|
|
|
|
return; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if (!root.isDirectory()) { |
|
|
|
|
|
|
|
Serial.println(" - not a directory"); |
|
|
|
|
|
|
|
return; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
File file = root.openNextFile(); |
|
|
|
|
|
|
|
while (file) { |
|
|
|
|
|
|
|
if (file.isDirectory()) { |
|
|
|
|
|
|
|
Serial.print(" DIR : "); |
|
|
|
|
|
|
|
Serial.println(file.name()); |
|
|
|
|
|
|
|
if (levels) { |
|
|
|
|
|
|
|
listDir(fs, file.name(), levels - 1); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
Serial.print(" FILE: "); |
|
|
|
|
|
|
|
Serial.print(file.name()); |
|
|
|
|
|
|
|
Serial.print("\tSIZE: "); |
|
|
|
|
|
|
|
Serial.println(file.size()); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
file = root.openNextFile(); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|