2021-01-29 12:31:58 +01:00
|
|
|
//
|
|
|
|
// FILE: I2CKeyPad.cpp
|
|
|
|
// AUTHOR: Rob Tillaart
|
2022-09-29 16:17:30 +02:00
|
|
|
// VERSION: 0.3.2
|
2021-11-08 13:13:29 +01:00
|
|
|
// PURPOSE: Arduino library for 4x4 KeyPad connected to an I2C PCF8574
|
2021-01-29 12:31:58 +01:00
|
|
|
// URL: https://github.com/RobTillaart/I2CKeyPad
|
|
|
|
//
|
|
|
|
// HISTORY:
|
|
|
|
// 0.0.1 2019-10-01 initial version
|
|
|
|
// 0.1.0 2020-06-26 first release
|
|
|
|
// 0.1.1 2020-07-05 fix compilation for ESP32
|
2021-11-08 13:13:29 +01:00
|
|
|
// 0.1.2 2020-12-27 Arduino-CI + unit test
|
2021-05-08 09:52:18 +02:00
|
|
|
//
|
|
|
|
// 0.2.0 2021-05-06 MultiWire ... (breaking interface)
|
|
|
|
// 0.2.1 2021-05-06 add _read(0xF0) to begin() to enable PCF8574
|
|
|
|
// interrupts. (#5 thanks to JohnMac1234)
|
2022-09-29 16:17:30 +02:00
|
|
|
//
|
2021-11-08 13:13:29 +01:00
|
|
|
// 0.3.0 2021-11-04 add key mapping functions.
|
2021-12-19 20:23:20 +01:00
|
|
|
// 0.3.1 2021-12-19 update library.json, license, minor edits
|
2022-09-29 16:17:30 +02:00
|
|
|
// 0.3.2 2022-09-19 experimental version
|
|
|
|
// add 5x3, 6x2 and 8x1 support
|
|
|
|
// moved all code to .cpp file.
|
2021-11-08 13:13:29 +01:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
#include "I2CKeyPad.h"
|
|
|
|
|
2021-11-08 13:13:29 +01:00
|
|
|
|
2021-05-08 09:52:18 +02:00
|
|
|
I2CKeyPad::I2CKeyPad(const uint8_t deviceAddress, TwoWire *wire)
|
2021-01-29 12:31:58 +01:00
|
|
|
{
|
2021-05-08 09:52:18 +02:00
|
|
|
_lastKey = I2C_KEYPAD_NOKEY;
|
|
|
|
_address = deviceAddress;
|
|
|
|
_wire = wire;
|
2022-09-29 16:17:30 +02:00
|
|
|
_mode = I2C_KEYPAD_4x4;
|
2021-01-29 12:31:58 +01:00
|
|
|
}
|
|
|
|
|
2021-11-08 13:13:29 +01:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
#if defined(ESP8266) || defined(ESP32)
|
2021-05-08 09:52:18 +02:00
|
|
|
bool I2CKeyPad::begin(uint8_t sda, uint8_t scl)
|
2021-01-29 12:31:58 +01:00
|
|
|
{
|
2021-05-08 09:52:18 +02:00
|
|
|
_wire->begin(sda, scl);
|
2021-12-19 20:23:20 +01:00
|
|
|
// enable interrupts
|
|
|
|
_read(0xF0);
|
2021-01-29 12:31:58 +01:00
|
|
|
return isConnected();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2021-11-08 13:13:29 +01:00
|
|
|
|
2021-05-08 09:52:18 +02:00
|
|
|
bool I2CKeyPad::begin()
|
2021-01-29 12:31:58 +01:00
|
|
|
{
|
2021-05-08 09:52:18 +02:00
|
|
|
_wire->begin();
|
2021-12-19 20:23:20 +01:00
|
|
|
// enable interrupts
|
2021-11-08 13:13:29 +01:00
|
|
|
_read(0xF0);
|
2021-01-29 12:31:58 +01:00
|
|
|
return isConnected();
|
|
|
|
}
|
|
|
|
|
2021-11-08 13:13:29 +01:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
uint8_t I2CKeyPad::getKey()
|
|
|
|
{
|
2022-09-29 16:17:30 +02:00
|
|
|
if (_mode == I2C_KEYPAD_5x3) return _getKey5x3();
|
|
|
|
if (_mode == I2C_KEYPAD_6x2) return _getKey6x2();
|
|
|
|
if (_mode == I2C_KEYPAD_8x1) return _getKey8x1();
|
|
|
|
// default.
|
2021-11-08 13:13:29 +01:00
|
|
|
return _getKey4x4();
|
2021-01-29 12:31:58 +01:00
|
|
|
}
|
|
|
|
|
2021-11-08 13:13:29 +01:00
|
|
|
|
2022-09-29 16:17:30 +02:00
|
|
|
uint8_t I2CKeyPad::getLastKey()
|
|
|
|
{
|
|
|
|
return _lastKey;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2021-12-19 20:23:20 +01:00
|
|
|
// to check "press any key"
|
2021-01-29 12:31:58 +01:00
|
|
|
bool I2CKeyPad::isPressed()
|
|
|
|
{
|
|
|
|
uint8_t a = _read(0xF0);
|
|
|
|
if (a == 0xFF) return false;
|
|
|
|
return (a != 0xF0);
|
|
|
|
}
|
|
|
|
|
2021-11-08 13:13:29 +01:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
bool I2CKeyPad::isConnected()
|
|
|
|
{
|
2021-05-08 09:52:18 +02:00
|
|
|
_wire->beginTransmission(_address);
|
|
|
|
return (_wire->endTransmission() == 0);
|
2021-01-29 12:31:58 +01:00
|
|
|
}
|
|
|
|
|
2021-11-08 13:13:29 +01:00
|
|
|
|
2022-09-29 16:17:30 +02:00
|
|
|
uint8_t I2CKeyPad::getChar()
|
|
|
|
{
|
|
|
|
return _keyMap[getKey()];
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t I2CKeyPad::getLastChar()
|
|
|
|
{
|
|
|
|
return _keyMap[_lastKey];
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2021-11-08 13:13:29 +01:00
|
|
|
void I2CKeyPad::loadKeyMap(char * keyMap)
|
|
|
|
{
|
|
|
|
_keyMap = keyMap;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-29 16:17:30 +02:00
|
|
|
void I2CKeyPad::setKeyPadMode(uint8_t mode)
|
|
|
|
{
|
|
|
|
if ((mode == I2C_KEYPAD_5x3) ||
|
|
|
|
(mode == I2C_KEYPAD_6x2) ||
|
|
|
|
(mode == I2C_KEYPAD_8x1))
|
|
|
|
{
|
|
|
|
_mode = mode;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
_mode = I2C_KEYPAD_4x4;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t I2CKeyPad::getKeyPadMode()
|
|
|
|
{
|
|
|
|
return _mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-08 13:13:29 +01:00
|
|
|
//////////////////////////////////////////////////////
|
|
|
|
//
|
2022-09-29 16:17:30 +02:00
|
|
|
// PROTECTED
|
2021-11-08 13:13:29 +01:00
|
|
|
//
|
2021-01-29 12:31:58 +01:00
|
|
|
uint8_t I2CKeyPad::_read(uint8_t mask)
|
|
|
|
{
|
2021-12-19 20:23:20 +01:00
|
|
|
// improve the odds that IO will not interrupted.
|
|
|
|
yield();
|
2021-05-08 09:52:18 +02:00
|
|
|
|
|
|
|
_wire->beginTransmission(_address);
|
|
|
|
_wire->write(mask);
|
|
|
|
if (_wire->endTransmission() != 0)
|
2021-01-29 12:31:58 +01:00
|
|
|
{
|
2021-12-19 20:23:20 +01:00
|
|
|
// set communication error
|
2021-01-29 12:31:58 +01:00
|
|
|
return 0xFF;
|
|
|
|
}
|
2021-05-08 09:52:18 +02:00
|
|
|
_wire->requestFrom(_address, (uint8_t)1);
|
|
|
|
return _wire->read();
|
2021-01-29 12:31:58 +01:00
|
|
|
}
|
|
|
|
|
2021-11-08 13:13:29 +01:00
|
|
|
|
|
|
|
uint8_t I2CKeyPad::_getKey4x4()
|
|
|
|
{
|
2021-12-19 20:23:20 +01:00
|
|
|
// key = row + 4 x col
|
2021-11-08 13:13:29 +01:00
|
|
|
uint8_t key = 0;
|
|
|
|
|
2021-12-19 20:23:20 +01:00
|
|
|
// mask = 4 rows as input pull up, 4 columns as output
|
2021-11-08 13:13:29 +01:00
|
|
|
uint8_t rows = _read(0xF0);
|
2021-12-19 20:23:20 +01:00
|
|
|
// check if single line has gone low.
|
2021-11-08 13:13:29 +01:00
|
|
|
if (rows == 0xF0) return I2C_KEYPAD_NOKEY;
|
|
|
|
else if (rows == 0xE0) key = 0;
|
|
|
|
else if (rows == 0xD0) key = 1;
|
|
|
|
else if (rows == 0xB0) key = 2;
|
|
|
|
else if (rows == 0x70) key = 3;
|
|
|
|
else return I2C_KEYPAD_FAIL;
|
|
|
|
|
|
|
|
// 4 columns as input pull up, 4 rows as output
|
|
|
|
uint8_t cols = _read(0x0F);
|
|
|
|
// check if single line has gone low.
|
|
|
|
if (cols == 0x0F) return I2C_KEYPAD_NOKEY;
|
|
|
|
else if (cols == 0x0E) key += 0;
|
|
|
|
else if (cols == 0x0D) key += 4;
|
|
|
|
else if (cols == 0x0B) key += 8;
|
|
|
|
else if (cols == 0x07) key += 12;
|
|
|
|
else return I2C_KEYPAD_FAIL;
|
|
|
|
|
|
|
|
_lastKey = key;
|
|
|
|
|
|
|
|
return key; // 0..15
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-29 16:17:30 +02:00
|
|
|
// not tested
|
|
|
|
uint8_t I2CKeyPad::_getKey5x3()
|
|
|
|
{
|
|
|
|
// key = row + 5 x col
|
|
|
|
uint8_t key = 0;
|
|
|
|
|
|
|
|
// mask = 5 rows as input pull up, 3 columns as output
|
|
|
|
uint8_t rows = _read(0xF8);
|
|
|
|
// check if single line has gone low.
|
|
|
|
if (rows == 0xF8) return I2C_KEYPAD_NOKEY;
|
|
|
|
else if (rows == 0xF0) key = 0;
|
|
|
|
else if (rows == 0xE8) key = 1;
|
|
|
|
else if (rows == 0xD8) key = 2;
|
|
|
|
else if (rows == 0xB8) key = 3;
|
|
|
|
else if (rows == 0x78) key = 4;
|
|
|
|
else return I2C_KEYPAD_FAIL;
|
|
|
|
|
|
|
|
// 3 columns as input pull up, 5 rows as output
|
|
|
|
uint8_t cols = _read(0x07);
|
|
|
|
// check if single line has gone low.
|
|
|
|
if (cols == 0x07) return I2C_KEYPAD_NOKEY;
|
|
|
|
else if (cols == 0x06) key += 0;
|
|
|
|
else if (cols == 0x05) key += 5;
|
|
|
|
else if (cols == 0x03) key += 10;
|
|
|
|
else return I2C_KEYPAD_FAIL;
|
|
|
|
|
|
|
|
_lastKey = key;
|
|
|
|
|
|
|
|
return key; // 0..14
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// not tested
|
|
|
|
uint8_t I2CKeyPad::_getKey6x2()
|
|
|
|
{
|
|
|
|
// key = row + 6 x col
|
|
|
|
uint8_t key = 0;
|
|
|
|
|
|
|
|
// mask = 6 rows as input pull up, 2 columns as output
|
|
|
|
uint8_t rows = _read(0xFC);
|
|
|
|
// check if single line has gone low.
|
|
|
|
if (rows == 0xFC) return I2C_KEYPAD_NOKEY;
|
|
|
|
else if (rows == 0xF8) key = 0;
|
|
|
|
else if (rows == 0xF4) key = 1;
|
|
|
|
else if (rows == 0xEC) key = 2;
|
|
|
|
else if (rows == 0xDC) key = 3;
|
|
|
|
else if (rows == 0xBC) key = 4;
|
|
|
|
else if (rows == 0x7C) key = 5;
|
|
|
|
else return I2C_KEYPAD_FAIL;
|
|
|
|
|
|
|
|
// 2 columns as input pull up, 6 rows as output
|
|
|
|
uint8_t cols = _read(0x03);
|
|
|
|
// check if single line has gone low.
|
|
|
|
if (cols == 0x03) return I2C_KEYPAD_NOKEY;
|
|
|
|
else if (cols == 0x02) key += 0;
|
|
|
|
else if (cols == 0x01) key += 6;
|
|
|
|
else return I2C_KEYPAD_FAIL;
|
|
|
|
|
|
|
|
_lastKey = key;
|
|
|
|
|
|
|
|
return key; // 0..11
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// not tested
|
|
|
|
uint8_t I2CKeyPad::_getKey8x1()
|
|
|
|
{
|
|
|
|
// key = row
|
|
|
|
uint8_t key = 0;
|
|
|
|
|
|
|
|
// mask = 8 rows as input pull up, 0 columns as output
|
|
|
|
uint8_t rows = _read(0xFF);
|
|
|
|
// check if single line has gone low.
|
|
|
|
if (rows == 0xFF) return I2C_KEYPAD_NOKEY;
|
|
|
|
else if (rows == 0xFE) key = 0;
|
|
|
|
else if (rows == 0xFD) key = 1;
|
|
|
|
else if (rows == 0xFB) key = 2;
|
|
|
|
else if (rows == 0xF7) key = 3;
|
|
|
|
else if (rows == 0xEF) key = 4;
|
|
|
|
else if (rows == 0xDF) key = 5;
|
|
|
|
else if (rows == 0xBF) key = 6;
|
|
|
|
else if (rows == 0x7F) key = 7;
|
|
|
|
else return I2C_KEYPAD_FAIL;
|
|
|
|
|
|
|
|
_lastKey = key;
|
|
|
|
|
|
|
|
return key; // 0..7
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
// -- END OF FILE --
|
2021-11-08 13:13:29 +01:00
|
|
|
|