/* USB MIDI Sync Slave Box Monitor * * This example demonstrates how to create a * MIDI hid compilant slave clock box with * monitor support using oled displays * * You need the following libraries to make it work * - Midi Library * - USB-MIDI and MIDIUSB * - u8g2 * - uClock * This example code is in the public domain. */ #include #include // // BPM Clock support // #include USBMIDI_CREATE_DEFAULT_INSTANCE(); U8X8 * u8x8; // MIDI clock, start, stop, note on and note off byte definitions - based on MIDI 1.0 Standards. #define MIDI_CLOCK 0xF8 #define MIDI_START 0xFA #define MIDI_STOP 0xFC char bpm_str[4]; float bpm = 126.0; uint8_t bpm_blink_timer = 1; uint8_t clock_state = 1; uint8_t clock_mode = 0; void handle_bpm_led(uint32_t * tick) { // BPM led indicator if ( !(*tick % (96)) || (*tick == 1) ) { // first compass step will flash longer bpm_blink_timer = 8; TXLED1; } else if ( !(*tick % (24)) ) { // each quarter led on TXLED1; } else if ( !(*tick % bpm_blink_timer) ) { // get led off TXLED0; bpm_blink_timer = 1; } } // Internal clock handlers void ClockOut96PPQN(uint32_t * tick) { // Send MIDI_CLOCK to external gears MIDI.sendRealTime(MIDI_CLOCK); handle_bpm_led(tick); } void onClockStart() { MIDI.sendRealTime(MIDI_START); } void onClockStop() { MIDI.sendRealTime(MIDI_STOP); } // External clock handlers void onExternalClock() { uClock.clockMe(); } void onExternalStart() { uClock.start(); } void onExternalStop() { uClock.stop(); } void setup() { // // MIDI setup // MIDI.begin(); MIDI.setHandleClock(onExternalClock); MIDI.setHandleStart(onExternalStart); MIDI.setHandleStop(onExternalStop); // // OLED setup // Please check you oled model to correctly init him // //u8x8 = new U8X8_SH1106_128X64_NONAME_HW_I2C(U8X8_PIN_NONE); u8x8 = new U8X8_SSD1306_128X64_NONAME_HW_I2C(U8X8_PIN_NONE); u8x8->begin(); u8x8->setFont(u8x8_font_pressstart2p_r); u8x8->clear(); u8x8->setFlipMode(true); u8x8->drawUTF8(0, 0, "uClock"); // // uClock Setup // // Drift for arudino leonardo over USB as MIDI HID uClock.setSlaveDrift(10); uClock.init(); uClock.setClock96PPQNOutput(ClockOut96PPQN); // For MIDI Sync Start and Stop uClock.setOnClockStartOutput(onClockStart); uClock.setOnClockStopOutput(onClockStop); uClock.setMode(uClock.EXTERNAL_CLOCK); //uClock.setTempo(136.5); //uClock.start(); } void printBpm(float _bpm, uint8_t col, uint8_t line) { int b = (int)_bpm; //int c = (int)((_bpm-b)*100); int c = (int)((_bpm-b)*10); itoa(b, bpm_str, 10); if (b > 99) { u8x8->drawUTF8(col, line, bpm_str); } else { bpm_str[2] = "\0"; u8x8->drawUTF8(col, line, " "); u8x8->drawUTF8(col+1, line, bpm_str); } u8x8->drawUTF8(col+3, line, "."); itoa(c, bpm_str, 10); u8x8->drawUTF8(col+4, line, bpm_str); u8x8->drawUTF8(col+5, line, "bpm"); } void loop() { while(MIDI.read()) {} // DO NOT ADD MORE PROCESS HERE AT THE COST OF LOSING CLOCK SYNC // Since arduino make use of Serial RX interruption we need to // read Serial as fast as we can on the loop if (bpm != uClock.getTempo()) { bpm = uClock.getTempo(); printBpm(bpm, 8, 7); } if (clock_state != uClock.state) { clock_state = uClock.state; if (clock_state >= 1) { u8x8->drawUTF8(0, 7, "playing"); } else { u8x8->drawUTF8(0, 7, "stoped "); } } if (clock_mode != uClock.getMode()) { clock_mode = uClock.getMode(); if (clock_mode == uClock.EXTERNAL_CLOCK) { u8x8->drawUTF8(10, 0, "slave "); } else { u8x8->drawUTF8(10, 0, "master"); } } }