123 lines
4.7 KiB
123 lines
4.7 KiB
6 years ago
|
/* Kaleidoscope - Firmware for computer input devices
|
||
|
* Copyright (C) 2013-2018 Keyboard.io, Inc.
|
||
|
*
|
||
|
* This program is free software: you can redistribute it and/or modify it under
|
||
|
* the terms of the GNU General Public License as published by the Free Software
|
||
|
* Foundation, version 3.
|
||
|
*
|
||
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
||
|
* FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
|
||
|
* details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License along with
|
||
|
* this program. If not, see <http://www.gnu.org/licenses/>.
|
||
|
*/
|
||
|
|
||
6 years ago
|
#include "kaleidoscope/Kaleidoscope.h"
|
||
7 years ago
|
#include "kaleidoscope/hooks.h"
|
||
6 years ago
|
#include "kaleidoscope/plugin.h"
|
||
8 years ago
|
|
||
8 years ago
|
static bool handleSyntheticKeyswitchEvent(Key mappedKey, uint8_t keyState) {
|
||
8 years ago
|
if (mappedKey.flags & RESERVED)
|
||
|
return false;
|
||
8 years ago
|
|
||
8 years ago
|
if (!(mappedKey.flags & SYNTHETIC))
|
||
|
return false;
|
||
8 years ago
|
|
||
7 years ago
|
if (mappedKey.flags & IS_CONSUMER) {
|
||
7 years ago
|
if (keyIsPressed(keyState))
|
||
8 years ago
|
kaleidoscope::hid::pressConsumerControl(mappedKey);
|
||
8 years ago
|
} else if (mappedKey.flags & IS_SYSCTL) {
|
||
8 years ago
|
if (keyIsPressed(keyState)) {
|
||
|
} else if (keyWasPressed(keyState)) {
|
||
8 years ago
|
kaleidoscope::hid::pressSystemControl(mappedKey);
|
||
8 years ago
|
kaleidoscope::hid::releaseSystemControl(mappedKey);
|
||
8 years ago
|
}
|
||
7 years ago
|
} else if (mappedKey.flags & IS_INTERNAL) {
|
||
|
return false;
|
||
8 years ago
|
} else if (mappedKey.flags & SWITCH_TO_KEYMAP) {
|
||
|
// Should not happen, handled elsewhere.
|
||
|
}
|
||
8 years ago
|
|
||
8 years ago
|
return true;
|
||
9 years ago
|
}
|
||
8 years ago
|
|
||
8 years ago
|
static bool handleKeyswitchEventDefault(Key mappedKey, byte row, byte col, uint8_t keyState) {
|
||
8 years ago
|
//for every newly pressed button, figure out what logical key it is and send a key down event
|
||
|
// for every newly released button, figure out what logical key it is and send a key up event
|
||
9 years ago
|
|
||
8 years ago
|
if (mappedKey.flags & SYNTHETIC) {
|
||
8 years ago
|
handleSyntheticKeyswitchEvent(mappedKey, keyState);
|
||
7 years ago
|
} else if (keyToggledOn(keyState)) {
|
||
8 years ago
|
kaleidoscope::hid::pressKey(mappedKey);
|
||
7 years ago
|
} else if (keyIsPressed(keyState)) {
|
||
|
kaleidoscope::hid::pressKey(mappedKey, false);
|
||
8 years ago
|
} else if (keyToggledOff(keyState) && (keyState & INJECTED)) {
|
||
8 years ago
|
kaleidoscope::hid::releaseKey(mappedKey);
|
||
8 years ago
|
}
|
||
|
return true;
|
||
9 years ago
|
}
|
||
|
|
||
8 years ago
|
void handleKeyswitchEvent(Key mappedKey, byte row, byte col, uint8_t keyState) {
|
||
7 years ago
|
/* These first steps are only done for keypresses that have a real (row,col).
|
||
|
* In particular, doing them for keypresses with out-of-bounds (row,col)
|
||
|
* would cause out-of-bounds array accesses in Layer.lookup(),
|
||
|
* Layer.updateLiveCompositeKeymap(), etc.
|
||
7 years ago
|
* Note that many INJECTED keypresses use the UNKNOWN_KEYSWITCH_LOCATION macro
|
||
|
* which gives us row==255, col==255 here. Therefore, it's legitimate that
|
||
|
* we may have keypresses with out-of-bounds (row, col).
|
||
|
* However, some INJECTED keypresses do have valid (row, col) if they are
|
||
|
* injecting an event tied to a physical keyswitch - and we want them to go
|
||
|
* through this lookup.
|
||
|
* So we can't just test for INJECTED here, we need to test the row and col
|
||
|
* directly.
|
||
|
* Note also that this (row, col) test avoids out-of-bounds accesses in *core*,
|
||
|
* but doesn't guarantee anything about event handlers - event handlers may
|
||
|
* still receive out-of-bounds (row, col), and handling that properly is on
|
||
|
* them.
|
||
8 years ago
|
*/
|
||
7 years ago
|
if (row < ROWS && col < COLS) {
|
||
|
|
||
7 years ago
|
/* If a key had an on event, we update the live composite keymap. See
|
||
7 years ago
|
* layers.h for an explanation about the different caches we have. */
|
||
7 years ago
|
if (keyToggledOn(keyState))
|
||
7 years ago
|
Layer.updateLiveCompositeKeymap(row, col);
|
||
|
|
||
|
/* If the key we are dealing with is masked, ignore it until it is released.
|
||
|
* When releasing it, clear the mask, so future key events can be handled
|
||
|
* appropriately.
|
||
|
*
|
||
|
* See layers.cpp for an example that masks keys, and the reason why it does
|
||
|
* so.
|
||
|
*/
|
||
|
if (KeyboardHardware.isKeyMasked(row, col)) {
|
||
|
if (keyToggledOff(keyState)) {
|
||
|
KeyboardHardware.unMaskKey(row, col);
|
||
|
} else {
|
||
|
return;
|
||
|
}
|
||
8 years ago
|
}
|
||
|
|
||
7 years ago
|
/* Convert (row, col) to the correct mappedKey
|
||
|
* The condition here means that if mappedKey and (row, col) are both valid,
|
||
|
* the mappedKey wins - we don't re-look-up the mappedKey
|
||
|
*/
|
||
|
if (mappedKey.raw == Key_NoKey.raw) {
|
||
7 years ago
|
mappedKey = Layer.lookup(row, col);
|
||
|
}
|
||
|
|
||
|
} // row < ROWS && col < COLS
|
||
|
|
||
|
// Keypresses with out-of-bounds (row,col) start here in the processing chain
|
||
8 years ago
|
|
||
7 years ago
|
// New event handler interface
|
||
|
if (kaleidoscope::Hooks::onKeyswitchEvent(mappedKey, row, col, keyState) != kaleidoscope::EventHandlerResult::OK)
|
||
|
return;
|
||
|
|
||
8 years ago
|
mappedKey = Layer.eventHandler(mappedKey, row, col, keyState);
|
||
|
if (mappedKey.raw == Key_NoKey.raw)
|
||
|
return;
|
||
|
handleKeyswitchEventDefault(mappedKey, row, col, keyState);
|
||
|
}
|