2017-02-19 18:55:54 -05:00
|
|
|
#include "applekeyboard.h"
|
|
|
|
#include "physicalkeyboard.h" // for LA/RA constants
|
|
|
|
|
|
|
|
#include "applemmu.h"
|
|
|
|
|
|
|
|
#include "globals.h"
|
|
|
|
|
|
|
|
// How many CPU cycles before we begin repeating a key?
|
|
|
|
#define STARTREPEAT 700000
|
|
|
|
// How many CPU cycles between repeats of a key?
|
|
|
|
#define REPEATAGAIN 66667
|
|
|
|
|
|
|
|
AppleKeyboard::AppleKeyboard(AppleMMU *m)
|
|
|
|
{
|
|
|
|
this->mmu = m;
|
|
|
|
|
|
|
|
for (uint16_t i=0; i<sizeof(keysDown); i++) {
|
|
|
|
keysDown[i] = false;
|
|
|
|
}
|
|
|
|
anyKeyIsDown = false;
|
|
|
|
startRepeatTimer = 0;
|
|
|
|
repeatTimer = 0;
|
|
|
|
|
|
|
|
capsLockEnabled = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
AppleKeyboard::~AppleKeyboard()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AppleKeyboard::isVirtualKey(uint8_t kc)
|
|
|
|
{
|
|
|
|
if (kc >= 0x81 && kc <= 0x86) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// apply the apple keymap.
|
|
|
|
// FIXME: easier with an array, but is that better?
|
|
|
|
uint8_t AppleKeyboard::translateKeyWithModifiers(uint8_t k)
|
|
|
|
{
|
|
|
|
// tolower, so we know what we're working with...
|
|
|
|
if (k >= 'A' && k <= 'Z') {
|
|
|
|
k = k - 'A' + 'a';
|
|
|
|
}
|
|
|
|
|
2018-02-07 10:20:26 -05:00
|
|
|
if (keysDown[PK_CTRL]) {
|
2017-02-19 18:55:54 -05:00
|
|
|
if (k >= 'a' && k <= 'z') {
|
|
|
|
return k - 'a' + 1;
|
|
|
|
}
|
|
|
|
// FIXME: any other control keys honored on the //e keyboard?
|
|
|
|
}
|
|
|
|
|
|
|
|
if (capsLockEnabled && k >= 'a' && k <= 'z') {
|
|
|
|
return k - 'a' + 'A';
|
|
|
|
}
|
|
|
|
|
2018-02-07 10:20:26 -05:00
|
|
|
if (keysDown[PK_LSHFT] || keysDown[PK_RSHFT]) {
|
2017-02-19 18:55:54 -05:00
|
|
|
if (k >= 'a' && k <= 'z') {
|
|
|
|
return k - 'a' + 'A';
|
|
|
|
}
|
|
|
|
switch (k) {
|
|
|
|
case '1':
|
|
|
|
return '!';
|
|
|
|
case '2':
|
|
|
|
return '@';
|
|
|
|
case '3':
|
|
|
|
return '#';
|
|
|
|
case '4':
|
|
|
|
return '$';
|
|
|
|
case '5':
|
|
|
|
return '%';
|
|
|
|
case '6':
|
|
|
|
return '^';
|
|
|
|
case '7':
|
|
|
|
return '&';
|
|
|
|
case '8':
|
|
|
|
return '*';
|
|
|
|
case '9':
|
|
|
|
return '(';
|
|
|
|
case '0':
|
|
|
|
return ')';
|
|
|
|
case '-':
|
|
|
|
return '_';
|
|
|
|
case '=':
|
|
|
|
return '+';
|
|
|
|
case '[':
|
|
|
|
return '{';
|
|
|
|
case ']':
|
|
|
|
return '}';
|
|
|
|
case '\\':
|
|
|
|
return '|';
|
|
|
|
case '`':
|
|
|
|
return '~';
|
|
|
|
case ';':
|
|
|
|
return ':';
|
|
|
|
case '\'':
|
|
|
|
return '"';
|
|
|
|
case ',':
|
|
|
|
return '<';
|
|
|
|
case '.':
|
|
|
|
return '>';
|
|
|
|
case '/':
|
|
|
|
return '?';
|
|
|
|
}
|
|
|
|
// FIXME: what the heck is it? I guess we don't need to shift it?
|
|
|
|
}
|
|
|
|
|
|
|
|
// And if we fall through, then just return it as-is
|
|
|
|
return k;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AppleKeyboard::keyDepressed(uint8_t k)
|
|
|
|
{
|
2021-12-12 12:10:49 -05:00
|
|
|
if (k >= 'A' && k <= 'Z') {
|
|
|
|
k = k - 'A' + 'a';
|
|
|
|
}
|
|
|
|
|
2017-02-19 18:55:54 -05:00
|
|
|
keysDown[k] = true;
|
|
|
|
|
|
|
|
// If it's not a virtual key, then set the anyKeyDown flag
|
|
|
|
// (the VM will see this as a keyboard key)
|
|
|
|
if (!isVirtualKey(k)) {
|
|
|
|
if (!anyKeyIsDown) {
|
|
|
|
mmu->setKeyDown(true);
|
|
|
|
anyKeyIsDown = true;
|
|
|
|
}
|
|
|
|
keyThatIsRepeating = translateKeyWithModifiers(k);
|
|
|
|
startRepeatTimer = g_cpu->cycles + STARTREPEAT;
|
|
|
|
mmu->keyboardInput(keyThatIsRepeating);
|
2018-02-07 10:20:26 -05:00
|
|
|
} else if (k == PK_LA) {
|
2017-02-19 18:55:54 -05:00
|
|
|
// Special handling: apple keys
|
2018-01-10 04:13:04 -05:00
|
|
|
mmu->setAppleKey(0, true);
|
2017-02-19 18:55:54 -05:00
|
|
|
return;
|
2018-02-07 10:20:26 -05:00
|
|
|
} else if (k == PK_RA) {
|
2017-02-19 18:55:54 -05:00
|
|
|
// Special handling: apple keys
|
2018-01-10 04:13:04 -05:00
|
|
|
mmu->setAppleKey(1, true);
|
2017-02-19 18:55:54 -05:00
|
|
|
return;
|
2018-02-07 10:20:26 -05:00
|
|
|
} else if (k == PK_LOCK) {
|
2017-02-19 18:55:54 -05:00
|
|
|
// Special handling: caps lock
|
|
|
|
capsLockEnabled = !capsLockEnabled;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AppleKeyboard::keyReleased(uint8_t k)
|
|
|
|
{
|
2021-12-12 12:10:49 -05:00
|
|
|
if (k >= 'A' && k <= 'Z') {
|
|
|
|
k = k - 'A' + 'a';
|
|
|
|
}
|
|
|
|
|
2017-02-19 18:55:54 -05:00
|
|
|
keysDown[k] = false;
|
|
|
|
|
|
|
|
// Special handling: apple keys
|
2018-02-07 10:20:26 -05:00
|
|
|
if (k == PK_LA) {
|
2018-01-10 04:13:04 -05:00
|
|
|
mmu->setAppleKey(0, false);
|
2017-02-19 18:55:54 -05:00
|
|
|
return;
|
|
|
|
}
|
2018-02-07 10:20:26 -05:00
|
|
|
if (k == PK_RA) {
|
2018-01-10 04:13:04 -05:00
|
|
|
mmu->setAppleKey(1, false);
|
2017-02-19 18:55:54 -05:00
|
|
|
return;
|
|
|
|
}
|
2018-02-07 10:20:26 -05:00
|
|
|
if (k == PK_LOCK) {
|
2017-02-19 18:55:54 -05:00
|
|
|
// Nothing to do when the caps lock key is released.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (anyKeyIsDown) {
|
|
|
|
anyKeyIsDown = false;
|
|
|
|
for (int i=0; i<sizeof(keysDown); i++) {
|
|
|
|
if (keysDown[i] && !isVirtualKey(i)) {
|
|
|
|
anyKeyIsDown = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!anyKeyIsDown) {
|
|
|
|
mmu->setKeyDown(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-02 09:06:15 -04:00
|
|
|
void AppleKeyboard::maintainKeyboard(int64_t cycleCount)
|
2017-02-19 18:55:54 -05:00
|
|
|
{
|
|
|
|
if (anyKeyIsDown) {
|
|
|
|
if (startRepeatTimer) {
|
|
|
|
if (cycleCount >= startRepeatTimer) {
|
|
|
|
// waiting to start repeating
|
|
|
|
startRepeatTimer = 0;
|
|
|
|
repeatTimer = 0;
|
|
|
|
// Will fall through...
|
|
|
|
} else {
|
|
|
|
// Don't fall through; not time to start repeating yet
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// already repeating; keep it up
|
|
|
|
if (cycleCount >= repeatTimer) {
|
|
|
|
mmu->keyboardInput(keyThatIsRepeating);
|
|
|
|
repeatTimer = cycleCount + REPEATAGAIN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|