mirror of
https://github.com/Keychron/qmk_firmware.git
synced 2025-01-01 06:18:52 +06:00
263 lines
7.9 KiB
C
263 lines
7.9 KiB
C
// Copyright 2022 Sergey Vlasov (@sigprof)
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#include "winry315.h"
|
|
|
|
#include "via.h"
|
|
|
|
#if !defined(WINRY315_DEFAULT_ORIENTATION)
|
|
# define WINRY315_DEFAULT_ORIENTATION WINRY315_ORIENTATION_TOP
|
|
#endif
|
|
|
|
#if !defined(VIA_ENABLE) && defined(ENCODER_ENABLE)
|
|
# ifndef MEDIA_KEY_DELAY
|
|
# define MEDIA_KEY_DELAY 10
|
|
# endif
|
|
bool encoder_update_kb(uint8_t index, bool clockwise) {
|
|
if (!encoder_update_user(index, clockwise)) {
|
|
return false;
|
|
}
|
|
if (index == 0) {
|
|
// Left encoder (assuming the default "top" orientation)
|
|
if (clockwise) {
|
|
tap_code(KC_PGDN);
|
|
} else {
|
|
tap_code(KC_PGUP);
|
|
}
|
|
} else if (index == 1) {
|
|
// Center encoder
|
|
if (clockwise) {
|
|
tap_code_delay(KC_VOLU, MEDIA_KEY_DELAY);
|
|
} else {
|
|
tap_code_delay(KC_VOLD, MEDIA_KEY_DELAY);
|
|
}
|
|
} else if (index == 2) {
|
|
// Right encoder
|
|
if (clockwise) {
|
|
tap_code_delay(KC_MNXT, MEDIA_KEY_DELAY);
|
|
} else {
|
|
tap_code_delay(KC_MPRV, MEDIA_KEY_DELAY);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
#endif // !defined(VIA_ENABLE) && defined(ENCODER_ENABLE)
|
|
|
|
#if defined(RGB_MATRIX_ENABLE)
|
|
|
|
// LED mapping (assuming the default "top" orientation):
|
|
// 0 - right encoder, top right
|
|
// 1 - right encoder, top left
|
|
// 2 - center encoder, top right
|
|
// 3 - center encoder, top left
|
|
// 4 - left encoder, top right
|
|
// 5 - left encoder, top left
|
|
// 6 - row 0, column 0
|
|
// 7 - row 1, column 0
|
|
// 8 - row 2, column 0
|
|
// 9 - row 2, column 1
|
|
// 10 - row 1, column 1
|
|
// 11 - row 0, column 1
|
|
// 12 - row 0, column 2
|
|
// 13 - row 1, column 2
|
|
// 14 - row 2, column 2
|
|
// 15 - row 2, column 3
|
|
// 16 - row 1, column 3
|
|
// 17 - row 0, column 3
|
|
// 18 - row 0, column 4
|
|
// 19 - row 1, column 4
|
|
// 20 - row 2, column 4
|
|
// 21 - underglow, right bottom
|
|
// 22 - underglow, left bottom
|
|
// 23 - underglow, left middle
|
|
// 24 - underglow, left top
|
|
// 25 - underglow, right top
|
|
// 26 - underglow, right middle
|
|
|
|
# define X_MM_MIN (-42)
|
|
# define X_MM_MAX 42
|
|
# define Y_MM_MIN (-40) // actually -35, but adjusted to get height = width
|
|
# define Y_MM_MAX 44 // actually 40, but adjusted to get height = width
|
|
# define WIDTH_MM (X_MM_MAX - X_MM_MIN)
|
|
# define HEIGHT_MM (Y_MM_MAX - Y_MM_MIN)
|
|
# define WIDTH_UNITS (35 * 2) // needs to match RGB_MATRIX_CENTER
|
|
# define HEIGHT_UNITS (35 * 2) // needs to match RGB_MATRIX_CENTER
|
|
|
|
// Convert the LED physical coordinates from millimeters with the origin at the
|
|
// PCB center to the form expected by the RGB Matrix code.
|
|
# define LED(x_mm, y_mm) \
|
|
{ ((x_mm - X_MM_MIN) * WIDTH_UNITS + WIDTH_MM / 2) / WIDTH_MM, ((Y_MM_MAX - y_mm) * HEIGHT_UNITS + HEIGHT_MM / 2) / HEIGHT_MM }
|
|
|
|
// clang-format off
|
|
static const led_config_t PROGMEM initial_led_config = {
|
|
{
|
|
{ 6, 11, 12, 17, 18, 7, 10, 13, 16, 19, 8, 9, 14, 15, 20, 2, 0, 4, NO_LED, NO_LED, NO_LED, NO_LED, NO_LED, NO_LED }
|
|
},
|
|
{
|
|
LED( 35, 36),
|
|
LED( 21, 36),
|
|
LED( 8, 34),
|
|
LED(-8, 34),
|
|
LED(-21, 36),
|
|
LED(-35, 36),
|
|
LED(-38, 5),
|
|
LED(-38, -14),
|
|
LED(-38, -33),
|
|
LED(-19, -33),
|
|
LED(-19, -14),
|
|
LED(-19, 5),
|
|
LED( 0, 5),
|
|
LED( 0, -14),
|
|
LED( 0, -33),
|
|
LED( 19, -33),
|
|
LED( 19, -14),
|
|
LED( 19, 5),
|
|
LED( 38, 5),
|
|
LED( 38, -14),
|
|
LED( 38, -33),
|
|
LED( 28, -35),
|
|
LED(-28, -35),
|
|
LED(-37, -9),
|
|
LED(-42, 40),
|
|
LED( 42, 40),
|
|
LED( 37, -9)
|
|
},
|
|
{
|
|
1, 1, 1, 1, 1, 1, // encoders (colored as modifiers)
|
|
4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, // regular keys
|
|
2, 2, 2, 2, 2, 2 // underglow
|
|
}
|
|
};
|
|
// clang-format on
|
|
|
|
led_config_t g_led_config;
|
|
|
|
void keyboard_pre_init_kb(void) {
|
|
// To be safe against any possible changes to rgb_matrix_init(),
|
|
// g_led_config should be initialized before rgb_matrix_init() is called;
|
|
// matrix_init_kb() would be too late, so keyboard_pre_init_kb() is used.
|
|
memcpy_P(&g_led_config, &initial_led_config, sizeof(g_led_config));
|
|
winry315_set_orientation(WINRY315_DEFAULT_ORIENTATION);
|
|
|
|
keyboard_pre_init_user();
|
|
}
|
|
|
|
// Encoders have two associated LEDs on this board; supporting more than one
|
|
// LED per key requires defining rgb_matrix_map_row_column_to_led_kb() to
|
|
// report any extra LEDs.
|
|
uint8_t rgb_matrix_map_row_column_to_led_kb(uint8_t row, uint8_t column, uint8_t *led_i) {
|
|
if (row == 0) {
|
|
switch (column) {
|
|
case 15: // center encoder
|
|
led_i[0] = 3;
|
|
return 1;
|
|
|
|
case 16: // right encoder
|
|
led_i[0] = 1;
|
|
return 1;
|
|
|
|
case 17: // left encoder
|
|
led_i[0] = 5;
|
|
return 1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
# if defined(VIA_ENABLE) && defined(VIA_CUSTOM_LIGHTING_ENABLE)
|
|
|
|
// VIA supports only 4 discrete values for effect speed; map these to some
|
|
// useful speed values for RGB Matrix.
|
|
enum speed_values {
|
|
RGBLIGHT_SPEED_0 = UINT8_MAX / 16, // not 0 to avoid really slow effects
|
|
RGBLIGHT_SPEED_1 = UINT8_MAX / 4,
|
|
RGBLIGHT_SPEED_2 = UINT8_MAX / 2, // matches the default value
|
|
RGBLIGHT_SPEED_3 = UINT8_MAX / 4 * 3, // UINT8_MAX is really fast
|
|
};
|
|
|
|
static uint8_t speed_from_rgblight(uint8_t rgblight_speed) {
|
|
switch (rgblight_speed) {
|
|
case 0:
|
|
return RGBLIGHT_SPEED_0;
|
|
case 1:
|
|
return RGBLIGHT_SPEED_1;
|
|
case 2:
|
|
default:
|
|
return RGBLIGHT_SPEED_2;
|
|
case 3:
|
|
return RGBLIGHT_SPEED_3;
|
|
}
|
|
}
|
|
|
|
static uint8_t speed_to_rgblight(uint8_t rgb_matrix_speed) {
|
|
if (rgb_matrix_speed < ((RGBLIGHT_SPEED_0 + RGBLIGHT_SPEED_1) / 2)) {
|
|
return 0;
|
|
} else if (rgb_matrix_speed < ((RGBLIGHT_SPEED_1 + RGBLIGHT_SPEED_2) / 2)) {
|
|
return 1;
|
|
} else if (rgb_matrix_speed < ((RGBLIGHT_SPEED_2 + RGBLIGHT_SPEED_3) / 2)) {
|
|
return 2;
|
|
} else {
|
|
return 3;
|
|
}
|
|
}
|
|
|
|
void raw_hid_receive_kb(uint8_t *data, uint8_t length) {
|
|
switch (data[0]) {
|
|
case id_lighting_get_value:
|
|
if (data[1] == id_qmk_rgblight_effect_speed) {
|
|
data[2] = speed_to_rgblight(rgb_matrix_get_speed());
|
|
}
|
|
break;
|
|
case id_lighting_set_value:
|
|
if (data[1] == id_qmk_rgblight_effect_speed) {
|
|
rgb_matrix_set_speed_noeeprom(speed_from_rgblight(data[2]));
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
# endif // defined(VIA_ENABLE) && defined(VIA_CUSTOM_LIGHTING_ENABLE)
|
|
|
|
#endif // defined(RGB_MATRIX_ENABLE)
|
|
|
|
void winry315_set_orientation(uint8_t orientation) {
|
|
(void)orientation;
|
|
#if defined(RGB_MATRIX_ENABLE)
|
|
for (uint8_t i = 0; i < DRIVER_LED_TOTAL; ++i) {
|
|
led_point_t * dst_point = &g_led_config.point[i];
|
|
const led_point_t *src_point = &initial_led_config.point[i];
|
|
uint8_t x = pgm_read_byte(&src_point->x);
|
|
uint8_t y = pgm_read_byte(&src_point->y);
|
|
|
|
switch (orientation) {
|
|
case WINRY315_ORIENTATION_TOP:
|
|
default:
|
|
dst_point->x = x;
|
|
dst_point->y = y;
|
|
break;
|
|
|
|
case WINRY315_ORIENTATION_LEFT:
|
|
dst_point->x = y;
|
|
dst_point->y = WIDTH_UNITS - x;
|
|
break;
|
|
|
|
case WINRY315_ORIENTATION_RIGHT:
|
|
dst_point->x = HEIGHT_UNITS - y;
|
|
dst_point->y = x;
|
|
break;
|
|
|
|
case WINRY315_ORIENTATION_BOTTOM:
|
|
dst_point->x = WIDTH_UNITS - x;
|
|
dst_point->y = HEIGHT_UNITS - y;
|
|
break;
|
|
}
|
|
}
|
|
#endif // defined(RGB_MATRIX_ENABLE)
|
|
}
|
|
|
|
#if defined(VIA_ENABLE)
|
|
void via_set_layout_options_kb(uint32_t value) {
|
|
winry315_set_orientation(value & 0x03);
|
|
}
|
|
#endif // defined(VIA_ENABLE)
|