Skip to content

My collection of (c++) tools: Keytool, Debug macro's and more

License

Notifications You must be signed in to change notification settings

knifter/lib-Tools

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

50 Commits
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Common c++ Tools Library for my projects

KeyTools Function to debounce and detect short/long/simultaneous presses

NonCopyable Disables copy- and assignemnt constructors when inherited from

log.h Log formatter helpers

tools-keys.cpp: key2event

Convert raw key scan bits to events: debounce, detect simultaneous presses and all with short, long, long repeat, release events.

Usage

  • Define enum with key bits, combinations etc
  • Scan keys into bits of a uint32_t (or the keys)
  • feed these bits through key2event() often
  • key2event uses internal time-tracking to generate events for each key and combinations

Example

Define your enum for used combinations:

typedef enum : uint32_t
{
  KEY_NONE = 0x00,
  KEY_RELEASED = KEYTOOL_RELEASED,
  KEY_PRESSED = KEYTOOL_SHORT | KEYTOOL_RELEASED,

  // key defs
  KEY_A = 0x01,
  KEY_B = 0x02,
  KEY_C = 0x04,

  // single key modes
  KEY_A_SHORT = KEY_A | KEYTOOL_SHORT,
  KEY_A_PRESSED = KEY_A | KEY_PRESSED,
  KEY_B_PRESSED = KEY_B | KEY_PRESSED,
  KEY_C_PRESSED = KEY_C | KEY_PRESSED,
  KEY_A_LONG = KEY_A | KEYTOOL_LONG,
  KEY_B_LONG = KEY_B | KEYTOOL_LONG,
  KEY_C_LONG = KEY_C | KEYTOOL_LONG,
  KEY_A_LONG_REPEAT = KEY_A | KEYTOOL_LONG_REPEAT,
  KEY_B_LONG_REPEAT = KEY_B | KEYTOOL_LONG_REPEAT,
  KEY_C_LONG_REPEAT = KEY_C | KEYTOOL_LONG_REPEAT,
  KEY_A_LONG_RELEASED = KEY_A | KEYTOOL_LONG | KEYTOOL_RELEASED,

  // combinations of 2
  KEY_AB_SHORT = KEY_A | KEY_B | KEYTOOL_SHORT,
  KEY_AC_PRESSED = KEY_A | KEY_C | KEY_PRESSED,
  KEY_AC_LONG = KEY_A | KEY_C | KEYTOOL_LONG,
  KEY_BC_LONG = KEY_B | KEY_C | KEYTOOL_LONG,
  KEY_AB_LONG_REPEAT = KEY_A | KEY_B | KEYTOOL_LONG_REPEAT,
  KEY_AC_LONG_REPEAT = KEY_A | KEY_C | KEYTOOL_LONG_REPEAT,
  KEY_BC_LONG_REPEAT = KEY_B | KEY_C | KEYTOOL_LONG_REPEAT,

  // combinations of 3
  KEY_ABC_SHORT = KEY_A | KEY_B | KEY_C | KEYTOOL_SHORT,
  KEY_ABC_LONG = KEY_A | KEY_B | KEY_C | KEYTOOL_LONG,
  KEY_ABC_LONG_REPEAT = KEY_A | KEY_B | KEY_C | KEYTOOL_LONG_REPEAT,
  KEY_ABC_LONG_RELEASED = KEY_A | KEY_B | KEY_C | KEYTOOL_LONG | KEYTOOL_LONG_RELEASED

  etc.. only combinations used in application have to be defined.
} keyevent_t;

and use key2event() like this:

void test_keys()
{
  // Scan keys (note the OR'ing!)
  uint32_t pressed = KEY_NONE;
  if(digitalRead(PIN_BTN_A) == LOW)
    pressed |= KEY_A;
  if(digitalRead(PIN_BTN_B) == LOW)
    pressed |= KEY_B;
  if(digitalRead(PIN_BTN_C) == LOW)
    pressed |= KEY_C;

  // feed through key2event
  uint32_t k = key2event(pressed);
  
  // Check results
  switch(k)
  {
    case KEY_A_SHORT:           print("A short-pressed (still holding)"); break;
    case KEY_A_PRESSED:         print("A pressed and released."); break;
    case KEY_A_LONG:            print("A long press"); break;
    case KEY_A_LONG_REPEAT:     print("A long repeated event"); break;
    case KEY_A_LONG_RELEASE:    print("A was released after long press"); break;
    case KEY_B_SHORT:           print("B short-pressed"); break;
    case KEY_B_LONG:            print("B long press"); break;
    case KEY_B_LONG_REPEAT:     print("B long press is held"); break;
    case KEY_AB_SHORT:          print("A+B combined"); break;
    case KEY_AB_LONG:           print("A+B long pressed"); break;
    case KEY_AB_LONG_REPEAT:    print("A+B long held"); break;
    case KEY_RELEASED:          print("All keys are released"); break;
  };
};

If you need immidiate key responses as well you can let the keys 'fall-through' too.

#define KEYTOOL_FALLTHROUGH
#include "keys-tools.h"

Do not OR the keys like this:
  keyevent_t k = key2event(pressed) | pressed;
as it will not be able to emit KEY_RELEASED (might be or-ed with current press)

And check for raw keys:

  // Check results
  switch(e)
  {
    case KEY_A:                 print("A is pressed"); break;
    case KEY_A_SHORT:           print("A short-pressed"); break;
    case KEY_A_LONG:            print("A long press"); break;
    case KEY_A_LONG_REPEAT:     print("A long repeated event"); break;
    case KEY_B:                 print("B is pressed"); break;
    ....
    
  };

And another example of how events can be used (for some rudimentary acceleration):

  // Scan keys
  ...
  
  uint32_t k = key2event(pressed);

  static int val = 0;
  static time_t start;
  switch(k)
  {
    case KEY_A_SHORT: val -= 1; break;
    case KEY_A_LONG:  
      start = millis();
      // no break
    case KEY_A_LONG_REPEAT:
      val -= 5*(1 + (millis() - start) / 500);
      break;
    case KEY_B_LONG: val = 0; break;
    case KEY_C_SHORT: val += 1; break;
    case KEY_C_LONG:
      start = millis();
    case KEY_C_LONG_REPEAT:
      val += 5*(1 + (millis() - start) / 500);
      break;
  };
  static int val_prv = val;
  if(val != val_prv)
  {
      DBG("val = val + %d = %d", val - val_prv, val);
      val_prv = val;
  };

About

My collection of (c++) tools: Keytool, Debug macro's and more

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published