// // Keyboard/event test program for the Fast Light Tool Kit (FLTK). // // Copyright 1998-2024 by Bill Spitzak and others. // // This library is free software. Distribution and use rights are outlined in // the file "COPYING" which should have been included with this file. If this // file is missing or damaged, see the license at: // // https://www.fltk.org/COPYING.php // // Please see the following page on how to report bugs and issues: // // https://www.fltk.org/bugs.php // // // Continuously display FLTK's event state. // // Known bugs: // // X insists on reporting the state *before* the shift key was // pressed, rather than after, on shift key events. I fixed this for // the mouse buttons, but it did not seem worth it for shift. // // X servers do not agree about any shift flags after except shift, ctrl, // lock, and alt. They may also not agree about the symbols for the extra // keys Micro$oft put on the keyboard. // // On IRIX the backslash key does not work. A bug in XKeysymToKeycode? // #include "keyboard_ui.h" #include // these are used to identify which buttons are which: void key_cb(Fl_Button*, void*) {} void shift_cb(Fl_Button*, void*) {} void wheel_cb(Fl_Dial*, void*) {} // This is used to stop Esc from exiting the program. // Other keystrokes like zoom keys (ctrl/+/-/0) must pass though. int handle(int e) { if (e == FL_SHORTCUT && Fl::event_key() == FL_Escape) return 1; return 0; } int MyWindow::handle(int msg) { if (msg == FL_MOUSEWHEEL) { roller_x->value( roller_x->value() + Fl::e_dx * roller_x->step() ); roller_y->value( roller_y->value() + Fl::e_dy * roller_y->step() ); return 1; } return 0; } struct keycode_table { int n; // key code const char* text; // key name } key_table[] = { { FL_Escape, "FL_Escape"}, { FL_BackSpace, "FL_BackSpace"}, { FL_Tab, "FL_Tab"}, { FL_Iso_Key, "FL_Iso_Key"}, { FL_Enter, "FL_Enter"}, { FL_Print, "FL_Print"}, { FL_Scroll_Lock, "FL_Scroll_Lock"}, { FL_Pause, "FL_Pause"}, { FL_Insert, "FL_Insert"}, { FL_Home, "FL_Home"}, { FL_Page_Up, "FL_Page_Up"}, { FL_Delete, "FL_Delete"}, { FL_End, "FL_End"}, { FL_Page_Down, "FL_Page_Down"}, { FL_Left, "FL_Left"}, { FL_Up, "FL_Up"}, { FL_Right, "FL_Right"}, { FL_Down, "FL_Down"}, { FL_Shift_L, "FL_Shift_L"}, { FL_Shift_R, "FL_Shift_R"}, { FL_Control_L, "FL_Control_L"}, { FL_Control_R, "FL_Control_R"}, { FL_Caps_Lock, "FL_Caps_Lock"}, { FL_Alt_L, "FL_Alt_L"}, { FL_Alt_R, "FL_Alt_R"}, { FL_Meta_L, "FL_Meta_L"}, { FL_Meta_R, "FL_Meta_R"}, { FL_Menu, "FL_Menu"}, { FL_Help, "FL_Help"}, { FL_Num_Lock, "FL_Num_Lock"}, { FL_KP_Enter, "FL_KP_Enter"}, { FL_Alt_Gr, "FL_Alt_Gr"} }; int main(int argc, char** argv) { Fl::add_handler(handle); MyWindow *window = make_window(); window->show(argc,argv); while (Fl::wait()) { const char *str; // update all the buttons with the current key and shift state: for (int c = 0; c < window->children(); c++) { Fl_Widget* b = window->child(c); if (b->callback() == (Fl_Callback*)key_cb) { int i = (int)b->argument(); if (!i) i = b->label()[0]; Fl_Button *btn = ((Fl_Button*)b); int state = Fl::event_key(i); if (btn->value()!=state) btn->value(state); } else if (b->callback() == (Fl_Callback*)shift_cb) { int i = (int)b->argument(); Fl_Button *btn = ((Fl_Button*)b); int state = Fl::event_state(i); if (btn->value()!=state) btn->value(state); } } // figure out the keyname: char buffer[100]; const char *keyname = buffer; int k = Fl::event_key(); if (!k) keyname = "0"; else if (k < 128) { // ASCII snprintf(buffer, sizeof(buffer), "'%c'", k); } else if (k >= 0xa0 && k <= 0xff) { // ISO-8859-1 (international keyboards) char key[8]; int kl = fl_utf8encode((unsigned)k, key); key[kl] = '\0'; snprintf(buffer, sizeof(buffer), "'%s'", key); } else if (k > FL_F && k <= FL_F_Last) { snprintf(buffer, sizeof(buffer), "FL_F+%d", k - FL_F); } else if (k >= FL_KP && k <= FL_KP_Last) { snprintf(buffer, sizeof(buffer), "FL_KP+'%c'", k-FL_KP); } else if (k >= FL_Button && k <= FL_Button+7) { snprintf(buffer, sizeof(buffer), "FL_Button+%d", k-FL_Button); } else { snprintf(buffer, sizeof(buffer), "0x%04x", k); for (int i = 0; i < int(sizeof(key_table)/sizeof(*key_table)); i++) if (key_table[i].n == k) {keyname = key_table[i].text; break;} } if (strcmp(key_output->value(), keyname)) key_output->value(keyname); str = Fl::event_text(); if (strcmp(text_output->value(), str)) text_output->value(str); } return 0; }