#include QMK_KEYBOARD_H #ifdef WEBUSB_ENABLE #include "webusb.h" #endif extern inline void ergodox_board_led_on(void); extern inline void ergodox_right_led_1_on(void); extern inline void ergodox_right_led_2_on(void); extern inline void ergodox_right_led_3_on(void); extern inline void ergodox_right_led_on(uint8_t led); extern inline void ergodox_board_led_off(void); extern inline void ergodox_right_led_1_off(void); extern inline void ergodox_right_led_2_off(void); extern inline void ergodox_right_led_3_off(void); extern inline void ergodox_right_led_off(uint8_t led); extern inline void ergodox_led_all_on(void); extern inline void ergodox_led_all_off(void); extern inline void ergodox_right_led_1_set(uint8_t n); extern inline void ergodox_right_led_2_set(uint8_t n); extern inline void ergodox_right_led_3_set(uint8_t n); extern inline void ergodox_right_led_set(uint8_t led, uint8_t n); extern inline void ergodox_led_all_set(uint8_t n); keyboard_config_t keyboard_config; bool i2c_initialized = 0; i2c_status_t mcp23018_status = 0x20; void matrix_init_kb(void) { // keyboard LEDs (see "PWM on ports OC1(A|B|C)" in "teensy-2-0.md") TCCR1A = 0b10101001; // set and configure fast PWM TCCR1B = 0b00001001; // set and configure fast PWM // (tied to Vcc for hardware convenience) DDRB &= ~(1<<4); // set B(4) as input PORTB &= ~(1<<4); // set B(4) internal pull-up disabled // unused pins - C7, D4, D5, D7, E6 // set as input with internal pull-up enabled DDRC &= ~(1<<7); DDRD &= ~(1<<5 | 1<<4); DDRE &= ~(1<<6); PORTC |= (1<<7); PORTD |= (1<<5 | 1<<4); PORTE |= (1<<6); keyboard_config.raw = eeconfig_read_kb(); ergodox_led_all_set((uint8_t)keyboard_config.led_level * 255 / 4 ); #ifdef RGB_MATRIX_ENABLE if (keyboard_config.rgb_matrix_enable) { rgb_matrix_set_flags(LED_FLAG_ALL); } else { rgb_matrix_set_flags(LED_FLAG_NONE); } #endif ergodox_blink_all_leds(); matrix_init_user(); } void ergodox_blink_all_leds(void) { ergodox_led_all_off(); ergodox_led_all_set(LED_BRIGHTNESS_DEFAULT); ergodox_right_led_1_on(); _delay_ms(50); ergodox_right_led_2_on(); _delay_ms(50); ergodox_right_led_3_on(); _delay_ms(50); #ifdef LEFT_LEDS ergodox_left_led_1_on(); _delay_ms(50); if (!mcp23018_status) { mcp23018_status = ergodox_left_leds_update(); } ergodox_left_led_2_on(); _delay_ms(50); if (!mcp23018_status) { mcp23018_status = ergodox_left_leds_update(); } ergodox_left_led_3_on(); _delay_ms(50); if (!mcp23018_status) { mcp23018_status = ergodox_left_leds_update(); } #endif ergodox_right_led_1_off(); _delay_ms(50); ergodox_right_led_2_off(); _delay_ms(50); ergodox_right_led_3_off(); #ifdef LEFT_LEDS _delay_ms(50); ergodox_left_led_1_off(); if (!mcp23018_status) { mcp23018_status = ergodox_left_leds_update(); } _delay_ms(50); ergodox_left_led_2_off(); if (!mcp23018_status) { mcp23018_status = ergodox_left_leds_update(); } _delay_ms(50); ergodox_left_led_3_off(); if (!mcp23018_status) { mcp23018_status = ergodox_left_leds_update(); } #endif //ergodox_led_all_on(); //_delay_ms(333); ergodox_led_all_set((uint8_t)keyboard_config.led_level * 255 / 4 ); ergodox_led_all_off(); } uint8_t init_mcp23018(void) { mcp23018_status = 0x20; // I2C subsystem // uint8_t sreg_prev; // sreg_prev=SREG; // cli(); if (i2c_initialized == 0) { i2c_init(); // on pins D(1,0) i2c_initialized = true; _delay_ms(1000); } // i2c_init(); // on pins D(1,0) // _delay_ms(1000); // set pin direction // - unused : input : 1 // - input : input : 1 // - driving : output : 0 mcp23018_status = i2c_start(I2C_ADDR_WRITE, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out; mcp23018_status = i2c_write(IODIRA, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out; mcp23018_status = i2c_write(0b00000000, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out; mcp23018_status = i2c_write(0b00111111, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out; i2c_stop(); // set pull-up // - unused : on : 1 // - input : on : 1 // - driving : off : 0 mcp23018_status = i2c_start(I2C_ADDR_WRITE, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out; mcp23018_status = i2c_write(GPPUA, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out; mcp23018_status = i2c_write(0b00000000, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out; mcp23018_status = i2c_write(0b00111111, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out; out: i2c_stop(); #ifdef LEFT_LEDS if (!mcp23018_status) mcp23018_status = ergodox_left_leds_update(); #endif // LEFT_LEDS // SREG=sreg_prev; return mcp23018_status; } #ifdef LEFT_LEDS uint8_t ergodox_left_leds_update(void) { if (mcp23018_status) { // if there was an error return mcp23018_status; } #define LEFT_LED_1_SHIFT 7 // in MCP23018 port B #define LEFT_LED_2_SHIFT 6 // in MCP23018 port B #define LEFT_LED_3_SHIFT 7 // in MCP23018 port A // set logical value (doesn't matter on inputs) // - unused : hi-Z : 1 // - input : hi-Z : 1 // - driving : hi-Z : 1 mcp23018_status = i2c_start(I2C_ADDR_WRITE, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out; mcp23018_status = i2c_write(OLATA, ERGODOX_EZ_I2C_TIMEOUT); if (mcp23018_status) goto out; mcp23018_status = i2c_write(0b11111111 & ~(ergodox_left_led_3<event.pressed ? WEBUSB_EVT_KEYDOWN : WEBUSB_EVT_KEYUP; event[2] = record->event.key.col; event[3] = record->event.key.row; event[4] = WEBUSB_STOP_BIT; webusb_send(event, sizeof(event)); } #endif switch (keycode) { case LED_LEVEL: if (record->event.pressed) { keyboard_config.led_level++; if (keyboard_config.led_level > 4) { keyboard_config.led_level = 0; } ergodox_led_all_set((uint8_t)keyboard_config.led_level * 255 / 4 ); eeconfig_update_kb(keyboard_config.raw); layer_state_set_kb(layer_state); } break; #ifdef RGB_MATRIX_ENABLE case TOGGLE_LAYER_COLOR: if (record->event.pressed) { keyboard_config.disable_layer_led ^= 1; if (keyboard_config.disable_layer_led) rgb_matrix_set_color_all(0, 0, 0); eeconfig_update_kb(keyboard_config.raw); } break; case RGB_TOG: if (record->event.pressed) { switch (rgb_matrix_get_flags()) { case LED_FLAG_ALL: { rgb_matrix_set_flags(LED_FLAG_NONE); keyboard_config.rgb_matrix_enable = false; rgb_matrix_set_color_all(0, 0, 0); } break; default: { rgb_matrix_set_flags(LED_FLAG_ALL); keyboard_config.rgb_matrix_enable = true; } break; } eeconfig_update_kb(keyboard_config.raw); } return false; #endif } return process_record_user(keycode, record); } void eeconfig_init_kb(void) { // EEPROM is getting reset! keyboard_config.raw = 0; keyboard_config.led_level = 4; keyboard_config.rgb_matrix_enable = true; eeconfig_update_kb(keyboard_config.raw); eeconfig_init_user(); } #ifdef WEBUSB_ENABLE static uint16_t loops = 0; static bool is_on = false; void matrix_scan_kb(void) { if(webusb_state.pairing == true) { if(loops == 0) { ergodox_right_led_1_off(); ergodox_right_led_2_off(); ergodox_right_led_3_off(); } if(loops % WEBUSB_BLINK_STEPS == 0) { if(is_on) { ergodox_right_led_2_off(); } else { ergodox_right_led_2_on(); } is_on ^= 1; } if(loops > WEBUSB_BLINK_END) { webusb_state.pairing = false; layer_state_set_user(layer_state); loops = 0; } loops++; } else if(loops > 0) { loops = 0; layer_state_set_user(layer_state); } matrix_scan_user(); } uint32_t layer_state_set_kb(uint32_t state) { state = layer_state_set_user(state); uint8_t layer = biton32(state); if(webusb_state.paired == true) { uint8_t event[4]; event[0] = WEBUSB_STATUS_OK; event[1] = WEBUSB_EVT_LAYER; event[2] = layer; event[3] = WEBUSB_STOP_BIT; webusb_send(event, sizeof(event)); } return state; } #endif