3712 lines
123 KiB
C
Raw Normal View History

/*
* xvkbd - Virtual Keyboard for X Window System
* (Version 3.2, 2010-03-14)
*
* Copyright (C) 2000-2010 by Tom Sato <VEF00200@nifty.ne.jp>
* http://homepage3.nifty.com/tsato/xvkbd/
*
* 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; either version 2
* of the License, or any later version.
*
* 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.
*/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <ctype.h>
#include <string.h>
#include <signal.h>
#include <errno.h>
#include <time.h>
#include <fnmatch.h>
#include <limits.h>
#include <X11/Intrinsic.h>
#include <X11/StringDefs.h>
#include <X11/Shell.h>
#include <X11/keysym.h>
#include <X11/cursorfont.h>
#include <X11/Xproto.h> /* to get request code */
#include <X11/Xaw/Box.h>
#include <X11/Xaw/Form.h>
#include <X11/Xaw/Command.h>
#include <X11/Xaw/Repeater.h>
#include <X11/Xaw/Label.h>
#include <X11/Xaw/MenuButton.h>
#include <X11/Xaw/SimpleMenu.h>
#include <X11/Xaw/SmeBSB.h>
#include <X11/Xaw/SmeLine.h>
#include <X11/Xaw/AsciiText.h>
#include <X11/Xaw/Viewport.h>
#include <X11/Xaw/List.h>
#include <X11/Xaw/Toggle.h>
#include <X11/Xmu/WinUtil.h>
#include <X11/Xatom.h>
#ifdef USE_I18N
# include <X11/Xlocale.h>
#endif
#ifdef USE_XTEST
# include <X11/extensions/XTest.h>
#endif
#include "resources.h"
#define PROGRAM_NAME_WITH_VERSION "xvkbd (v3.2)"
/*
* Default keyboard layout is hardcoded here.
* Layout of the main keyboard can be redefined by resources.
*/
#define NUM_KEY_ROWS 25
#define NUM_KEY_COLS 25
static char *keys_normal[NUM_KEY_ROWS][NUM_KEY_COLS] = {
{ "F1", "F2", "F3", "F4", "F5", "F6", "F7", "F8", "F9", "F10", "F11", "F12", "BackSpace" },
{ "Escape", "1", "2", "3", "4", "5", "6", "7", "8", "9", "0", "-", "=", "\\", "`" },
{ "Tab", "q", "w", "e", "r", "t", "y", "u", "i", "o", "p", "[", "]", "Delete" },
{ "Control_L", "a", "s", "d", "f", "g", "h", "j", "k", "l", ";", "'", "Return" },
{ "Shift_L", "z", "x", "c", "v", "b", "n", "m", ",", ".", "/", "Multi_key", "Shift_R" },
{ "MainMenu", "Caps_Lock", "Alt_L", "Meta_L", "space", "Meta_R", "Alt_R",
"Left", "Right", "Up", "Down", "Focus" },
};
static char *keys_shift[NUM_KEY_ROWS][NUM_KEY_COLS] = {
{ "F1", "F2", "F3", "F4", "F5", "F6", "F7", "F8", "F9", "F10", "F11", "F12", "BackSpace" },
{ "Escape", "!", "@", "#", "$", "%", "^", "&", "*", "(", ")", "_", "+", "|", "~" },
{ "Tab", "Q", "W", "E", "R", "T", "Y", "U", "I", "O", "P", "{", "}", "Delete" },
{ "Control_L", "A", "S", "D", "F", "G", "H", "J", "K", "L", ":", "\"", "Return" },
{ "Shift_L", "Z", "X", "C", "V", "B", "N", "M", "<", ">", "?", "Multi_key", "Shift_R" },
{ "MainMenu", "Caps_Lock", "Alt_L", "Meta_L", "space", "Meta_R", "Alt_R",
"Left", "Right", "Up", "Down", "Focus" },
};
static char *keys_altgr[NUM_KEY_ROWS][NUM_KEY_COLS] = { { NULL } };
static char *keys_shift_altgr[NUM_KEY_ROWS][NUM_KEY_COLS] = { { NULL } };
static char *key_labels[NUM_KEY_ROWS][NUM_KEY_COLS] = {
{ "F1", "F2", "F3", "F4", "F5", "F6", "F7", "F8", "F9", "F10", "F11", "F12", "Backspace" },
{ "Esc", "!\n1", "@\n2", "#\n3", "$\n4", "%\n5", "^\n6",
"&\n7", "*\n8", "(\n9", ")\n0", "_\n-", "+\n=", "|\n\\", "~\n`" },
{ "Tab", "Q", "W", "E", "R", "T", "Y", "U", "I", "O", "P", "{\n[", "}\n]", "Del" },
{ "Control", "A", "S", "D", "F", "G", "H", "J", "K", "L", ":\n;", "\"\n'", "Return" },
{ "Shift", "Z", "X", "C", "V", "B", "N", "M", "<\n,", ">\n.", "?\n/", "Com\npose", "Shift" },
{ "MainMenu", "Caps\nLock", "Alt", "Meta", "", "Meta", "Alt",
"left", "right", "up", "down", "Focus" },
};
static char *normal_key_labels[NUM_KEY_ROWS][NUM_KEY_COLS] = {
{ "F1", "F2", "F3", "F4", "F5", "F6", "F7", "F8", "F9", "F10", "F11", "F12", "Backspace" },
{ "Esc", "1", "2", "3", "4", "5", "6", "7", "8", "9", "0", "-", "=", "\\", "`" },
{ "Tab", "q", "w", "e", "r", "t", "y", "u", "i", "o", "p", "[", "]", "Del" },
{ "Ctrl", "a", "s", "d", "f", "g", "h", "j", "k", "l", ";", "'", "Return" },
{ "Shift", "z", "x", "c", "v", "b", "n", "m", ",", ".", "/", "Comp", "Shift" },
{ "MainMenu", "Caps", "Alt", "Meta", "", "Meta", "Alt",
"left", "right", "up", "down", "Focus" },
};
static char *shift_key_labels[NUM_KEY_ROWS][NUM_KEY_COLS] = {
{ "F1", "F2", "F3", "F4", "F5", "F6", "F7", "F8", "F9", "F10", "F11", "F12", "Backspace" },
{ "Esc", "!", "@", "#", "$", "%", "^", "&", "*", "(", ")", "_", "+", "|", "~" },
{ "Tab", "Q", "W", "E", "R", "T", "Y", "U", "I", "O", "P", "{", "}", "Del" },
{ "Ctrl", "A", "S", "D", "F", "G", "H", "J", "K", "L", ":", "\"", "Return" },
{ "Shift", "Z", "X", "C", "V", "B", "N", "M", "<", ">", "?", "Comp", "Shift" },
{ "MainMenu", "Caps", "Alt", "Meta", "", "Meta", "Alt",
"left", "right", "up", "down", "Focus" },
};
static char *altgr_key_labels[NUM_KEY_ROWS][NUM_KEY_COLS] = { { NULL } };
static char *shift_altgr_key_labels[NUM_KEY_ROWS][NUM_KEY_COLS] = { { NULL } };
#define NUM_KEYPAD_ROWS NUM_KEY_ROWS
#define NUM_KEYPAD_COLS NUM_KEY_COLS
static char *keypad[NUM_KEYPAD_ROWS][NUM_KEYPAD_COLS] = {
{ "Num_Lock", "KP_Divide", "KP_Multiply", "Focus" },
{ "Home", "Up", "Page_Up", "KP_Add" },
{ "Left", "5", "Right", "KP_Subtract" },
{ "End", "Down", "Page_Down", "KP_Enter" },
{ "Insert", "Delete" },
};
static char *keypad_shift[NUM_KEYPAD_ROWS][NUM_KEYPAD_COLS] = {
{ "Num_Lock", "KP_Divide", "KP_Multiply", "Focus" },
{ "KP_7", "KP_8", "KP_9", "KP_Add" },
{ "KP_4", "KP_5", "KP_6", "KP_Subtract" },
{ "KP_1", "KP_2", "KP_3", "KP_Enter" },
{ "KP_0", "." },
};
static char *keypad_label[NUM_KEYPAD_ROWS][NUM_KEYPAD_COLS] = {
{ "Num\nLock", "/", "*", "Focus" },
{ "7\nHome", "8\nUp ", "9\nPgUp", "+" },
{ "4\nLeft", "5\n ", "6\nRight", "-" },
{ "1\nEnd ", "2\nDown", "3\nPgDn", "Enter" },
{ "0\nIns ", ".\nDel " },
};
#define NUM_SUN_FKEY_ROWS 6
#define NUM_SUN_FKEY_COLS 3
static char *sun_fkey[NUM_SUN_FKEY_ROWS][NUM_SUN_FKEY_COLS] = {
{ "L1", "L2" },
{ "L3", "L4" },
{ "L5", "L6" },
{ "L7", "L8" },
{ "L9", "L10" },
{ "Help" },
};
static char *sun_fkey_label[NUM_SUN_FKEY_ROWS][NUM_SUN_FKEY_COLS] = {
{ "Stop \nL1", "Again\nL2" },
{ "Props\nL3", "Undo \nL4" },
{ "Front\nL5", "Copy \nL6" },
{ "Open \nL7", "Paste\nL8" },
{ "Find \nL9", "Cut \nL10" },
{ "Help" },
};
/*
* Image for arrow keys
*/
#define up_width 7
#define up_height 13
static unsigned char up_bits[] = {
0x08, 0x1c, 0x1c, 0x3e, 0x2a, 0x49, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08,
0x08};
#define down_width 7
#define down_height 13
static unsigned char down_bits[] = {
0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x49, 0x2a, 0x3e, 0x1c, 0x1c,
0x08};
#define left_width 13
#define left_height 13
static unsigned char left_bits[] = {
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x20, 0x00, 0x18, 0x00, 0x0e, 0x00,
0xff, 0x1f, 0x0e, 0x00, 0x18, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00};
#define right_width 13
#define right_height 13
static unsigned char right_bits[] = {
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80, 0x00, 0x00, 0x03, 0x00, 0x0e,
0xff, 0x1f, 0x00, 0x0e, 0x00, 0x03, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00};
#define check_width 16
#define check_height 16
static unsigned char check_bits[] = {
0x00, 0x00, 0x00, 0x00, 0x00, 0x70, 0x00, 0x38, 0x00, 0x1e, 0x08, 0x0f,
0x8c, 0x07, 0xde, 0x03, 0xfe, 0x03, 0xfc, 0x01, 0xf8, 0x00, 0xf0, 0x00,
0x70, 0x00, 0x60, 0x00, 0x00, 0x00, 0x00, 0x00};
#define back_width 18
#define back_height 13
static unsigned char back_bits[] = {
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x60, 0x00, 0x00,
0x78, 0x00, 0x00, 0xfe, 0xff, 0x03, 0xff, 0xff, 0x03, 0xfe, 0xff, 0x03,
0x78, 0x00, 0x00, 0x60, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00};
/*
* Resources and options
*/
#define Offset(entry) XtOffset(struct appres_struct *, entry)
static XtResource application_resources[] = {
{ "description", "Description", XtRString, sizeof(char *),
Offset(description), XtRImmediate,
PROGRAM_NAME_WITH_VERSION " - virtual keyboard for X window system\n\n"
"Copyright (C) 2000-2010 by Tom Sato <VEF00200@nifty.ne.jp>\n"
"http://homepage3.nifty.com/tsato/xvkbd/\n\n"
"This program is free software with ABSOLUTELY NO WARRANTY,\n"
"distributed under the terms of the GNU General Public License.\n" },
{ "showManualCommand", "ShowManualCommand", XtRString, sizeof(char *),
Offset(show_manual_command), XtRImmediate, "xterm -e man xvkbd &" },
{ "windowGeometry", "Geometry", XtRString, sizeof(char *),
Offset(geometry), XtRImmediate, "" },
{ "inheritGeoemetry", "Inherit", XtRBoolean, sizeof(Boolean),
Offset(inherit_geometry), XtRImmediate, (XtPointer)TRUE },
{ "debug", "Debug", XtRBoolean, sizeof(Boolean),
Offset(debug), XtRImmediate, (XtPointer)FALSE },
{ "version", "Version", XtRBoolean, sizeof(Boolean),
Offset(version), XtRImmediate, (XtPointer)FALSE },
#ifdef USE_XTEST
{ "xtest", "XTest", XtRBoolean, sizeof(Boolean),
Offset(xtest), XtRImmediate, (XtPointer)TRUE },
#else
{ "xtest", "XTest", XtRBoolean, sizeof(Boolean),
Offset(xtest), XtRImmediate, (XtPointer)FALSE },
#endif
{ "noSync", "NoSync", XtRBoolean, sizeof(Boolean),
Offset(no_sync), XtRImmediate, (XtPointer)FALSE },
{ "alwaysOnTop", "AlwaysOnTop", XtRBoolean, sizeof(Boolean),
Offset(always_on_top), XtRImmediate, (XtPointer)FALSE },
{ "wmToolbar", "WmToolbar", XtRBoolean, sizeof(Boolean),
Offset(wm_toolbar), XtRImmediate, (XtPointer)FALSE },
{ "jumpPointer", "JumpPointer", XtRBoolean, sizeof(Boolean),
Offset(jump_pointer), XtRImmediate, (XtPointer)TRUE },
{ "jumpPointerAlways", "JumpPointer", XtRBoolean, sizeof(Boolean),
Offset(jump_pointer_always), XtRImmediate, (XtPointer)TRUE },
{ "jumpPointerBack", "JumpPointer", XtRBoolean, sizeof(Boolean),
Offset(jump_pointer_back), XtRImmediate, (XtPointer)TRUE },
{ "quickModifiers", "QuickModifiers", XtRBoolean, sizeof(Boolean),
Offset(quick_modifiers), XtRImmediate, (XtPointer)TRUE },
{ "altgrLock", "ModifiersLock", XtRBoolean, sizeof(Boolean),
Offset(altgr_lock), XtRImmediate, (XtPointer)FALSE },
{ "shiftLock", "ModifiersLock", XtRBoolean, sizeof(Boolean),
Offset(shift_lock), XtRImmediate, (XtPointer)FALSE },
{ "modifiersLock", "ModifiersLock", XtRBoolean, sizeof(Boolean),
Offset(modifiers_lock), XtRImmediate, (XtPointer)FALSE },
{ "numLockState", "NumLockState", XtRBoolean, sizeof(Boolean),
Offset(num_lock_state), XtRImmediate, (XtPointer)TRUE },
{ "autoRepeat", "AutoRepeat", XtRBoolean, sizeof(Boolean),
Offset(auto_repeat), XtRImmediate, (XtPointer)TRUE },
{ "modalKeytop", "ModalKeytop", XtRBoolean, sizeof(Boolean),
Offset(modal_keytop), XtRImmediate, (XtPointer)FALSE },
{ "minimizable", "Minimizable", XtRBoolean, sizeof(Boolean),
Offset(minimizable), XtRImmediate, (XtPointer)FALSE },
{ "secure", "Secure", XtRBoolean, sizeof(Boolean),
Offset(secure), XtRImmediate, (XtPointer)FALSE },
{ "no_root", "NoRoot", XtRBoolean, sizeof(Boolean),
Offset(no_root), XtRImmediate, (XtPointer)FALSE },
{ "wait_idle", "Text", XtRString, sizeof(char *),
Offset(wait_idle), XtRImmediate, "" },
{ "nonexitable", "Secure", XtRBoolean, sizeof(Boolean),
Offset(nonexitable), XtRImmediate, (XtPointer)FALSE },
{ "modalKeytop", "ModalKeytop", XtRBoolean, sizeof(Boolean),
Offset(modal_keytop), XtRImmediate, (XtPointer)FALSE },
{ "modalThreshold", "ModalThreshold", XtRInt, sizeof(int),
Offset(modal_threshold), XtRImmediate, (XtPointer)150 },
{ "keypad", "Keypad", XtRBoolean, sizeof(Boolean),
Offset(keypad), XtRImmediate, (XtPointer)TRUE },
{ "functionkey", "FunctionKey", XtRBoolean, sizeof(Boolean),
Offset(function_key), XtRImmediate, (XtPointer)TRUE },
{ "compact", "Compact", XtRBoolean, sizeof(Boolean),
Offset(compact), XtRImmediate, (XtPointer)FALSE },
{ "keypadOnly", "KeypadOnly", XtRBoolean, sizeof(Boolean),
Offset(keypad_only), XtRImmediate, (XtPointer)FALSE },
{ "keypadKeysym", "KeypadKeysym", XtRBoolean, sizeof(Boolean),
Offset(keypad_keysym), XtRImmediate, (XtPointer)FALSE },
{ "autoAddKeysym", "AutoAddKeysym", XtRBoolean, sizeof(Boolean),
Offset(auto_add_keysym), XtRImmediate, (XtPointer)TRUE },
{ "listWidgets", "Debug", XtRBoolean, sizeof(Boolean),
Offset(list_widgets), XtRImmediate, (XtPointer)FALSE },
{ "positiveModifiers", "PositiveModifiers", XtRString, sizeof(char *),
Offset(positive_modifiers), XtRImmediate, "" },
{ "text", "Text", XtRString, sizeof(char *),
Offset(text), XtRImmediate, "" },
{ "file", "File", XtRString, sizeof(char *),
Offset(file), XtRImmediate, "" },
{ "window", "Window", XtRString, sizeof(char *),
Offset(window), XtRImmediate, "" },
{ "instance", "Instance", XtRString, sizeof(char *),
Offset(instance), XtRImmediate, "" },
{ "widget", "Widget", XtRString, sizeof(char *),
Offset(widget), XtRImmediate, "" },
{ "generalFont", XtCFont, XtRFontStruct, sizeof(XFontStruct *),
Offset(general_font), XtRString, XtDefaultFont},
{ "letterFont", XtCFont, XtRFontStruct, sizeof(XFontStruct *),
Offset(letter_font), XtRString, XtDefaultFont},
{ "specialFont", XtCFont, XtRFontStruct, sizeof(XFontStruct *),
Offset(special_font), XtRString, XtDefaultFont},
{ "keypadFont", XtCFont, XtRFontStruct, sizeof(XFontStruct *),
Offset(keypad_font), XtRString, XtDefaultFont},
{ "generalBackground", XtCBackground, XtRPixel, sizeof(Pixel),
Offset(general_background), XtRString, "gray" },
{ "specialBackground", XtCBackground, XtRPixel, sizeof(Pixel),
Offset(special_background), XtRString, "gray" },
{ "specialForeground", XtCForeground, XtRPixel, sizeof(Pixel),
Offset(special_foreground), XtRString, "black" },
#ifdef USE_I18N
{ "specialFontSet", XtCFontSet, XtRFontSet, sizeof(XFontSet),
Offset(special_fontset), XtRString, XtDefaultFontSet},
#endif
{ "highlightBackground", XtCBackground, XtRPixel, sizeof(Pixel),
Offset(highlight_background), XtRString, "gray" },
{ "highlightForeground", XtCForeground, XtRPixel, sizeof(Pixel),
Offset(highlight_foreground), XtRString, "forestgreen" },
{ "focusBackground", XtCBackground, XtRPixel, sizeof(Pixel),
Offset(focus_background), XtRString, "gray" },
{ "remoteFocusBackground", XtCBackground, XtRPixel, sizeof(Pixel),
Offset(remote_focus_background), XtRString, "cyan" },
{ "balloonBackground", XtCBackground, XtRPixel, sizeof(Pixel),
Offset(balloon_background), XtRString, "LightYellow1" },
{ "launchBalloonBackground", XtCBackground, XtRPixel, sizeof(Pixel),
Offset(launch_balloon_background), XtRString, "SkyBlue1" },
{ "normalkeys", "NormalKeys", XtRString, sizeof(char *),
Offset(keys_normal), XtRImmediate, "" },
{ "shiftkeys", "ShiftKeys", XtRString, sizeof(char *),
Offset(keys_shift), XtRImmediate, "" },
{ "altgrkeys", "AltgrKeys", XtRString, sizeof(char *),
Offset(keys_altgr), XtRImmediate, "" },
{ "shiftaltgrkeys", "ShiftAltgrKeys", XtRString, sizeof(char *),
Offset(keys_shift_altgr), XtRImmediate, "" },
{ "keylabels", "KeyLabels", XtRString, sizeof(char *),
Offset(key_labels), XtRImmediate, "" },
{ "normalkeylabels", "NormalKeyLabels", XtRString, sizeof(char *),
Offset(normal_key_labels), XtRImmediate, "" },
{ "shiftkeylabels", "ShiftKeyLabels", XtRString, sizeof(char *),
Offset(shift_key_labels), XtRImmediate, "" },
{ "altgrkeylabels", "AltgrKeyLabels", XtRString, sizeof(char *),
Offset(altgr_key_labels), XtRImmediate, "" },
{ "shiftaltgrkeylabels", "ShiftAltgrKeyLabels", XtRString, sizeof(char *),
Offset(shift_altgr_key_labels), XtRImmediate, "" },
{ "normalkeypad", "NormalKeypad", XtRString, sizeof(char *),
Offset(keypad_normal), XtRImmediate, "" },
{ "shiftkeypad", "ShiftKeypad", XtRString, sizeof(char *),
Offset(keypad_shift), XtRImmediate, "" },
{ "keypad_labels", "KeypadLabels", XtRString, sizeof(char *),
Offset(keypad_labels), XtRImmediate, "" },
{ "deadkeys", "DeadKeys", XtRString, sizeof(char *),
Offset(deadkeys), XtRImmediate, "" },
{ "altgrKeycode", "AltgrKeycode", XtRInt, sizeof(int),
Offset(altgr_keycode), XtRImmediate, (XtPointer)0 },
{ "keyFile", "KeyFile", XtRString, sizeof(char *),
Offset(key_file), XtRImmediate, ".xvkbd" },
{ "dictFile", "DictFile", XtRString, sizeof(char *),
Offset(dict_file), XtRImmediate, "/usr/share/dict/words" },
{ "customizations", "Customizations", XtRString, sizeof(char *),
Offset(customizations), XtRImmediate, "default" },
{ "editableFunctionKeys", "EditableFunctionKeys", XtRInt, sizeof(int),
Offset(editable_function_keys), XtRImmediate, (XtPointer)12 },
{ "maxWidthRatio", "MaxRatio", XtRFloat, sizeof(float),
Offset(max_width_ratio), XtRString, "0.9" },
{ "maxHeightRatio", "MaxRatio", XtRFloat, sizeof(float),
Offset(max_height_ratio), XtRString, "0.5" },
{ "textDelay", "TextDelay", XtRInt, sizeof(int),
Offset(text_delay), XtRImmediate, (XtPointer)0 },
{ "keyClickPitch", "KeyClickPitch", XtRInt, sizeof(int),
Offset(key_click_pitch), XtRImmediate, (XtPointer)1000 },
{ "keyClickDuration", "KeyClickDuration", XtRInt, sizeof(int),
Offset(key_click_duration), XtRImmediate, (XtPointer)1 },
{ "autoClickDelay", "AutoClickDelay", XtRInt, sizeof(int),
Offset(autoclick_delay), XtRImmediate, (XtPointer)0 },
};
#undef Offset
static XrmOptionDescRec options[] = {
{ "-geometry", ".windowGeometry", XrmoptionSepArg, NULL },
{ "-windowgeometry", ".windowGeometry", XrmoptionSepArg, NULL },
{ "-debug", ".debug", XrmoptionNoArg, "True" },
#ifdef USE_XTEST
{ "-xtest", ".xtest", XrmoptionNoArg, "True" },
{ "-xsendevent", ".xtest", XrmoptionNoArg, "False" },
{ "-no-jump-pointer", ".jumpPointer", XrmoptionNoArg, "False" },
{ "-no-back-pointer", ".jumpPointerBack", XrmoptionNoArg, "False" },
#endif
{ "-no-sync", ".noSync", XrmoptionNoArg, "True" },
{ "-always-on-top", ".alwaysOnTop", XrmoptionNoArg, "True" }, /* EXPERIMENTAL */
{ "-quick", ".quickModifiers", XrmoptionNoArg, "True" },
{ "-modifiers", ".positiveModifiers", XrmoptionSepArg, NULL },
{ "-text", ".text", XrmoptionSepArg, NULL },
{ "-file", ".file", XrmoptionSepArg, NULL },
{ "-delay", ".textDelay", XrmoptionSepArg, NULL },
{ "-window", ".window", XrmoptionSepArg, NULL },
{ "-instance", ".instance", XrmoptionSepArg, NULL },
{ "-widget", ".widget", XrmoptionSepArg, NULL },
{ "-altgr-lock", ".altgrLock", XrmoptionNoArg, "True" },
{ "-no-altgr-lock", ".altgrLock", XrmoptionNoArg, "False" },
{ "-no-repeat", ".autoRepeat", XrmoptionNoArg, "False" },
{ "-norepeat", ".autoRepeat", XrmoptionNoArg, "False" },
{ "-no-keypad", ".keypad", XrmoptionNoArg, "False" },
{ "-nokeypad", ".keypad", XrmoptionNoArg, "False" },
{ "-no-functionkey", ".functionkey", XrmoptionNoArg, "False" },
{ "-nofunctionkey", ".functionkey", XrmoptionNoArg, "False" },
{ "-highlight", ".highlightForeground", XrmoptionSepArg, NULL },
{ "-compact", ".compact", XrmoptionNoArg, "True" },
{ "-keypad", ".keypadOnly", XrmoptionNoArg, "True" },
{ "-true-keypad", ".keypadKeysym", XrmoptionNoArg, "True" },
{ "-truekeypad", ".keypadKeysym", XrmoptionNoArg, "True" },
{ "-no-add-keysym", ".autoAddKeysym", XrmoptionNoArg, "False" },
{ "-altgr-keycode", ".altgrKeycode", XrmoptionSepArg, NULL },
{ "-list", ".listWidgets", XrmoptionNoArg, "True" },
{ "-modal", ".modalKeytop", XrmoptionNoArg, "True" },
{ "-minimizable", ".minimizable", XrmoptionNoArg, "True" },
{ "-secure", ".secure", XrmoptionNoArg, "True" },
{ "-no_root", ".no_root", XrmoptionNoArg, "True" },
{ "-wait_idle", ".wait_idle", XrmoptionSepArg, NULL },
{ "-nonexitable", ".nonexitable", XrmoptionNoArg, "True" },
{ "-xdm", ".Secure", XrmoptionNoArg, "True" },
{ "-dict", ".dictFile", XrmoptionSepArg, NULL },
{ "-keyfile", ".keyFile", XrmoptionSepArg, NULL },
{ "-customizations", ".customizations", XrmoptionSepArg, NULL },
{ "-version", ".version", XrmoptionNoArg, "True" },
{ "-help", ".version", XrmoptionNoArg, "True" },
};
/*
* Global variables
*/
static char dict_filename[PATH_MAX] = "";
static int argc1;
static char **argv1;
static XtAppContext app_con;
static Widget toplevel;
static Widget key_widgets[NUM_KEY_ROWS][NUM_KEY_COLS];
static Widget main_menu = None;
static Dimension toplevel_height = 1000;
static Display *dpy;
static Atom wm_delete_window = None;
static KeySym *keysym_table = NULL;
static int min_keycode, max_keycode;
static int keysym_per_keycode;
static Boolean error_detected;
static int alt_mask = 0;
static int meta_mask = 0;
static int altgr_mask = 0;
static KeySym altgr_keysym = NoSymbol;
static int shift_state = 0;
static int mouse_shift = 0;
static Display *target_dpy = NULL;
static Window toplevel_parent = None;
static Window focused_window = None;
static Window focused_subwindow = None;
static Pixmap xvkbd_pixmap = None;
static int AddKeysym(KeySym keysym, Boolean top); /* forward */
static void SendString(const unsigned char *str);
static void MakeKeyboard(Boolean remake);
static void MakeKeypad(Widget form, Widget from_vert, Widget from_horiz);
static void MakeSunFunctionKey(Widget form, Widget from_vert, Widget from_horiz);
static void MakeDeadkeyPanel(Widget form);
static void RefreshMainMenu(void);
static void PopupFunctionKeyEditor(void);
static void DeleteWindowProc(Widget w, XEvent *event, String *pars, Cardinal *n_pars);
/*
* Search for window which has specified instance name (WM_NAME)
* or class name (WM_CLASS).
*/
static Window FindWindow(Window top, char *name)
{
Window w;
Window *children, dummy;
unsigned int nchildren;
int i;
XClassHint hint;
char *win_name;
w = None;
if (appres.debug) fprintf(stderr, "FindWindow: id=0x%lX", (long)top);
if (XGetClassHint(target_dpy, top, &hint)) {
if (hint.res_name) {
if (appres.debug) fprintf(stderr, " instance=\"%s\"", hint.res_name);
if ((strlen(name) > 0 && fnmatch(name, hint.res_name, 0) == 0)
|| (strlen(appres.instance) > 0 && fnmatch(appres.instance, hint.res_name, 0) == 0)) w = top;
XFree(hint.res_name);
}
if (strlen(name) > 0 && hint.res_class) {
if (appres.debug) fprintf(stderr, " class=\"%s\"", hint.res_class);
if (strlen(name) > 0 && fnmatch(name, hint.res_class, 0) == 0) w = top;
XFree(hint.res_class);
}
}
if (XFetchName(target_dpy, top, &win_name)) { /* window title */
if (appres.debug) fprintf(stderr, " title=\"%s\"", win_name);
if (strlen(name) > 0 && fnmatch(name, win_name, 0) == 0) w = top;
XFree(win_name);
}
if (appres.debug) fprintf(stderr, "%s\n", (w == None) ? "" : " [matched]");
if (w == None &&
XQueryTree(target_dpy, top, &dummy, &dummy, &children, &nchildren)) {
for (i = 0; i < nchildren; i++) {
w = FindWindow(children[i], name);
if (w != None) break;
}
if (children) XFree((char *)children);
}
return(w);
}
/*
* This will be called to get window to set input focus,
* when user pressed the "Focus" button.
*/
static void GetFocusedWindow(void)
{
Cursor cursor;
XEvent event;
Window target_root, child;
int junk_i;
unsigned junk_u;
Window junk_w;
int scrn;
int cur_x, cur_y, last_x, last_y;
double x_ratio, y_ratio;
XFlush(target_dpy);
target_root = RootWindow(target_dpy, DefaultScreen(target_dpy));
cursor = XCreateFontCursor(dpy, (target_dpy == dpy) ? XC_crosshair : XC_dot);
if (XGrabPointer(dpy, RootWindow(dpy, DefaultScreen(dpy)), False, ButtonPressMask,
GrabModeSync, GrabModeAsync, None,
cursor, CurrentTime) == 0) {
if (appres.debug)
fprintf(stderr, "Grab pointer - waiting for button press\n");
last_x = -1;
last_y = -1;
x_ratio = ((double)WidthOfScreen(DefaultScreenOfDisplay(target_dpy))
/ WidthOfScreen(XtScreen(toplevel)));
y_ratio = ((double)HeightOfScreen(DefaultScreenOfDisplay(target_dpy))
/ HeightOfScreen(XtScreen(toplevel)));
do {
XAllowEvents(dpy, SyncPointer, CurrentTime);
if (target_dpy == dpy) {
XNextEvent(dpy, &event);
} else {
XCheckTypedEvent(dpy, ButtonPress, &event);
if (XQueryPointer(dpy, RootWindow(dpy, DefaultScreen(dpy)), &junk_w, &junk_w,
&cur_x, &cur_y, &junk_i, &junk_i, &junk_u)) {
cur_x = cur_x * x_ratio;
cur_y = cur_y * y_ratio;
}
if (cur_x != last_x || cur_y != last_y) {
if (appres.debug) fprintf(stderr, "Moving pointer to (%d, %d) on %s\n",
cur_x, cur_y, XDisplayString(target_dpy));
XWarpPointer(target_dpy, None, target_root, 0, 0, 0, 0, cur_x, cur_y);
XFlush(target_dpy);
last_x = cur_x;
last_y = cur_y;
XQueryPointer(target_dpy, target_root, &junk_w, &child,
&cur_x, &cur_y, &junk_i, &junk_i, &junk_u);
usleep(10000);
} else {
usleep(100000);
}
}
} while (event.type != ButtonPress);
XUngrabPointer(dpy, CurrentTime);
focused_window = None;
if (target_dpy == dpy) focused_window = event.xbutton.subwindow;
if (focused_window == None) {
XFlush(target_dpy);
for (scrn = 0; scrn < ScreenCount(target_dpy); scrn++) {
if (XQueryPointer(target_dpy, RootWindow(target_dpy, scrn), &junk_w, &child,
&junk_i, &junk_i, &junk_i, &junk_i, &junk_u)) {
if (appres.debug)
fprintf(stderr, "Window on the other display/screen (screen #%d of %s) focused\n",
scrn, XDisplayString(target_dpy));
target_root = RootWindow(target_dpy, scrn);
focused_window = child;
break;
}
}
}
if (focused_window == None) focused_window = target_root;
else focused_window = XmuClientWindow(target_dpy, focused_window);
if (appres.debug) fprintf(stderr, "Selected window is: 0x%lX on %s\n",
focused_window, XDisplayString(target_dpy));
if (target_dpy == dpy && XtWindow(toplevel) == focused_window) {
focused_window = None;
focused_subwindow = focused_window;
return;
}
focused_subwindow = focused_window;
do { /* search the child window */
XQueryPointer(target_dpy, focused_subwindow, &junk_w, &child,
&junk_i, &junk_i, &junk_i, &junk_i, &junk_u);
if (child != None) {
focused_subwindow = child;
if (appres.debug) fprintf(stderr, " going down: 0x%lX\n", focused_subwindow);
}
} while (child != None);
if (appres.list_widgets || strlen(appres.widget) != 0) {
child = FindWidget(toplevel, focused_window, appres.widget);
if (child != None) focused_subwindow = child;
}
} else {
fprintf(stderr, "%s: cannot grab pointer\n", PROGRAM_NAME);
}
}
/*
* Read keyboard mapping and modifier mapping.
* Keyboard mapping is used to know what keys are in shifted position.
* Modifier mapping is required because we should know Alt and Meta
* key are used as which modifier.
*/
static void Highlight(char *name, Boolean state);
static void ReadKeymap(void)
{
int i;
int keycode, inx, pos;
KeySym keysym;
XModifierKeymap *modifiers;
Widget w;
int last_altgr_mask;
XDisplayKeycodes(target_dpy, &min_keycode, &max_keycode);
if (keysym_table != NULL) XFree(keysym_table);
keysym_table = XGetKeyboardMapping(target_dpy,
min_keycode, max_keycode - min_keycode + 1,
&keysym_per_keycode);
for (keycode = min_keycode; keycode <= max_keycode; keycode++) {
/* if the first keysym is alphabet and the second keysym is NoSymbol,
it is equivalent to pair of lowercase and uppercase alphabet */
inx = (keycode - min_keycode) * keysym_per_keycode;
if (keysym_table[inx + 1] == NoSymbol
&& ((XK_A <= keysym_table[inx] && keysym_table[inx] <= XK_Z)
|| (XK_a <= keysym_table[inx] && keysym_table[inx] <= XK_z))) {
if (XK_A <= keysym_table[inx] && keysym_table[inx] <= XK_Z)
keysym_table[inx] = keysym_table[inx] - XK_A + XK_a;
keysym_table[inx + 1] = keysym_table[inx] - XK_a + XK_A;
}
}
last_altgr_mask = altgr_mask;
alt_mask = 0;
meta_mask = 0;
altgr_mask = 0;
altgr_keysym = NoSymbol;
modifiers = XGetModifierMapping(target_dpy);
for (i = 0; i < 8; i++) {
for (pos = 0; pos < modifiers->max_keypermod; pos++) {
keycode = modifiers->modifiermap[i * modifiers->max_keypermod + pos];
if (keycode < min_keycode || max_keycode < keycode) continue;
keysym = keysym_table[(keycode - min_keycode) * keysym_per_keycode];
if (keysym == XK_Alt_L || keysym == XK_Alt_R) {
alt_mask = 1 << i;
} else if (keysym == XK_Meta_L || keysym == XK_Meta_R) {
meta_mask = 1 << i;
} else if (keysym == XK_Mode_switch) {
if (appres.debug)
fprintf(stderr, "%s: found Mode_switch at %dth modifier\n", PROGRAM_NAME, i);
if (altgr_keysym == XK_ISO_Level3_Shift) {
if (appres.debug)
fprintf(stderr, "%s: both ISO_Level3_Shift and Mode_switch found - ISO_Level3_Shift prefered\n", PROGRAM_NAME);
} else {
altgr_mask = 0x0101 << i;
/* I don't know why, but 0x2000 was required for mod3 on my Linux box */
altgr_keysym = keysym;
}
} else if (keysym == XK_ISO_Level3_Shift) {
/* if no Mode_switch, try to use ISO_Level3_Shift instead */
/* however, it may not work as intended - I don't know why */
if (appres.debug)
fprintf(stderr, "%s: found ISO_Level3_Shift at %dth modifier\n", PROGRAM_NAME, i);
if (altgr_keysym == XK_Mode_switch) {
if (appres.debug)
fprintf(stderr, "%s: both ISO_Level3_Shift and Mode_switch found - ISO_Level3_Shift prefered\n", PROGRAM_NAME);
}
altgr_mask = 1 << i;
altgr_keysym = keysym;
}
}
}
XFreeModifiermap(modifiers);
if (altgr_keysym != XK_Mode_switch) {
fprintf(stderr, "%s: Mode_switch not available as a modifier\n", PROGRAM_NAME);
if (altgr_keysym == XK_ISO_Level3_Shift)
fprintf(stderr, "%s: although ISO_Level3_Shift is used instead, AltGr may not work correctly\n", PROGRAM_NAME);
else
fprintf(stderr, "%s: AltGr can't be used\n", PROGRAM_NAME);
}
w = XtNameToWidget(toplevel, "*Multi_key");
if (w != None) {
if (XKeysymToKeycode(target_dpy, XK_Multi_key) == NoSymbol) {
if (!appres.auto_add_keysym || AddKeysym(XK_Multi_key, FALSE) == NoSymbol)
XtSetSensitive(w, FALSE);
}
}
w = XtNameToWidget(toplevel, "*Mode_switch");
if (w != None) {
if (appres.xtest && 0 < appres.altgr_keycode) {
XtSetSensitive(w, TRUE);
if (appres.debug)
fprintf(stderr, "%s: keycode %d will be used for AltGr - it was specified with altgrKeycode\n",
PROGRAM_NAME, appres.altgr_keycode);
} else if (altgr_mask) {
XtSetSensitive(w, TRUE);
} else {
XtSetSensitive(w, FALSE);
if (shift_state & last_altgr_mask) {
shift_state &= ~last_altgr_mask;
Highlight("Mode_switch", FALSE);
}
}
}
}
/*
* This will called when X error is detected when attempting to
* send a event to a client window; this will normally caused
* when the client window is destroyed.
*/
static int MyErrorHandler(Display *my_dpy, XErrorEvent *event)
{
char msg[200];
error_detected = TRUE;
if (event->error_code == BadWindow) {
if (appres.debug)
fprintf(stderr, "%s: BadWindow - couldn't find target window 0x%lX (destroyed?)\n",
PROGRAM_NAME, (long)focused_window);
return 0;
}
XGetErrorText(my_dpy, event->error_code, msg, sizeof(msg) - 1);
fprintf(stderr, "X error trapped: %s, request-code=%d\n", msg, event->request_code);
return 0;
}
/*
* Send event to the focused window.
* If input focus is specified explicitly, select the window
* before send event to the window.
*/
static void SendEvent(XKeyEvent *event)
{
static Boolean first = TRUE;
if (!appres.no_sync) {
XSync(event->display, FALSE);
XSetErrorHandler(MyErrorHandler);
}
error_detected = FALSE;
if (focused_window != None) {
/* set input focus if input focus is set explicitly */
if (appres.debug)
fprintf(stderr, "Set input focus to window 0x%lX (0x%lX)\n",
(long)focused_window, (long)event->window);
XSetInputFocus(event->display, focused_window, RevertToParent, CurrentTime);
if (!appres.no_sync) XSync(event->display, FALSE);
}
if (!error_detected) {
if (appres.xtest) {
#ifdef USE_XTEST
if (appres.debug)
fprintf(stderr, "XTestFakeKeyEvent(0x%lx, %ld, %d)\n",
(long)event->display, (long)event->keycode, event->type == KeyPress);
if (appres.jump_pointer) {
Window root, child, w;
int root_x, root_y, x, y;
unsigned int mask;
int revert_to;
w = None;
if (first || strlen(appres.text) == 0 || appres.jump_pointer_back) {
first = FALSE;
w = focused_subwindow;
if (w == None && appres.jump_pointer_always)
XGetInputFocus(event->display, &w, &revert_to);
if (w != None) {
if (appres.debug)
fprintf(stderr, "SendEvent: jump pointer to window 0x%lx\n", (long)w);
XQueryPointer(event->display, w,
&root, &child, &root_x, &root_y, &x, &y, &mask);
XWarpPointer(event->display, None, w, 0, 0, 0, 0, 1, 1);
XFlush(event->display);
}
}
XTestFakeKeyEvent(event->display, event->keycode, event->type == KeyPress, 0);
XFlush(event->display);
if (w != None && appres.jump_pointer_back) {
XWarpPointer(event->display, None, root, 0, 0, 0, 0, root_x, root_y);
XFlush(event->display);
}
} else {
XTestFakeKeyEvent(event->display, event->keycode, event->type == KeyPress, 0);
XFlush(event->display);
}
#else
fprintf(stderr, "%s: this binary is compiled without XTEST support\n",
PROGRAM_NAME);
#endif
} else {
XSendEvent(event->display, event->window, TRUE, KeyPressMask, (XEvent *)event);
if (!appres.no_sync) XSync(event->display, FALSE);
if (error_detected
&& (focused_subwindow != None) && (focused_subwindow != event->window)) {
error_detected = FALSE;
event->window = focused_subwindow;
if (appres.debug)
fprintf(stderr, " retry: send event to window 0x%lX (0x%lX)\n",
(long)focused_window, (long)event->window);
XSendEvent(event->display, event->window, TRUE, KeyPressMask, (XEvent *)event);
if (!appres.no_sync) XSync(event->display, FALSE);
}
}
}
if (error_detected) {
/* reset focus because focused window is (probably) no longer exist */
XBell(dpy, 0);
focused_window = None;
focused_subwindow = None;
}
XSetErrorHandler(NULL);
}
/*
* Insert a specified keysym to unused position in the keymap table.
* This will be called to add required keysyms on-the-fly.
* if the second parameter is TRUE, the keysym will be added to the
* non-shifted position - this may be required for modifier keys
* (e.g. Mode_switch) and some special keys (e.g. F20).
*/
static int AddKeysym(KeySym keysym, Boolean top)
{
int keycode, pos, max_pos, inx, phase;
if (top) {
max_pos = 0;
} else {
max_pos = keysym_per_keycode - 1;
if (4 <= max_pos) max_pos = 3;
if (2 <= max_pos && altgr_keysym != XK_Mode_switch) max_pos = 1;
}
for (phase = 0; phase < 2; phase++) {
for (keycode = max_keycode; min_keycode <= keycode; keycode--) {
for (pos = max_pos; 0 <= pos; pos--) {
inx = (keycode - min_keycode) * keysym_per_keycode;
if ((phase != 0 || keysym_table[inx] == NoSymbol) && keysym_table[inx] < 0xFF00) {
/* In the first phase, to avoid modifing existing keys, */
/* add the keysym only to the keys which has no keysym in the first position. */
/* If no place fuond in the first phase, add the keysym for any keys except */
/* for modifier keys and other special keys */
if (keysym_table[inx + pos] == NoSymbol) {
if (appres.debug)
fprintf(stderr, "*** Adding keysym \"%s\" at keycode %d position %d/%d\n",
XKeysymToString(keysym), keycode, pos, keysym_per_keycode);
keysym_table[inx + pos] = keysym;
XChangeKeyboardMapping(target_dpy, keycode, keysym_per_keycode, &keysym_table[inx], 1);
XFlush(target_dpy);
return keycode;
}
}
}
}
}
fprintf(stderr, "%s: couldn't add \"%s\" to keymap\n",
PROGRAM_NAME, XKeysymToString(keysym));
XBell(dpy, 0);
return NoSymbol;
}
/*
* Add the specified key as a new modifier.
* This is used to use Mode_switch (AltGr) as a modifier.
*/
static void AddModifier(KeySym keysym)
{
XModifierKeymap *modifiers;
int keycode, i, pos;
keycode = XKeysymToKeycode(target_dpy, keysym);
if (keycode == NoSymbol) keycode = AddKeysym(keysym, TRUE);
modifiers = XGetModifierMapping(target_dpy);
for (i = 7; 3 < i; i--) {
if (modifiers->modifiermap[i * modifiers->max_keypermod] == NoSymbol
|| ((keysym_table[(modifiers->modifiermap[i * modifiers->max_keypermod]
- min_keycode) * keysym_per_keycode]) == XK_ISO_Level3_Shift
&& keysym == XK_Mode_switch)) {
for (pos = 0; pos < modifiers->max_keypermod; pos++) {
if (modifiers->modifiermap[i * modifiers->max_keypermod + pos] == NoSymbol) {
if (appres.debug)
fprintf(stderr, "Adding modifier \"%s\" as %dth modifier\n",
XKeysymToString(keysym), i);
modifiers->modifiermap[i * modifiers->max_keypermod + pos] = keycode;
XSetModifierMapping(target_dpy, modifiers);
return;
}
}
}
}
fprintf(stderr, "%s: couldn't add \"%s\" as modifier\n",
PROGRAM_NAME, XKeysymToString(keysym));
XBell(dpy, 0);
}
/*
* Send sequence of KeyPressed/KeyReleased events to the focused
* window to simulate keyboard. If modifiers (shift, control, etc)
* are set ON, many events will be sent.
*/
static void SendKeyPressedEvent(KeySym keysym, unsigned int shift)
{
Window cur_focus;
int revert_to;
XKeyEvent event;
int keycode;
Window root, *children;
unsigned int n_children;
int phase, inx;
Boolean found;
if (focused_subwindow != None)
cur_focus = focused_subwindow;
else
XGetInputFocus(target_dpy, &cur_focus, &revert_to);
if (appres.debug) {
char ch = '?';
if ((keysym & ~0x7f) == 0 && isprint(keysym)) ch = keysym;
fprintf(stderr, "SendKeyPressedEvent: focus=0x%lX, key=0x%lX (%c), shift=0x%lX\n",
(long)cur_focus, (long)keysym, ch, (long)shift);
}
if (XtWindow(toplevel) != None) {
if (toplevel_parent == None) {
XQueryTree(target_dpy, RootWindow(target_dpy, DefaultScreen(target_dpy)),
&root, &toplevel_parent, &children, &n_children);
XFree(children);
}
if (cur_focus == None || cur_focus == PointerRoot
|| cur_focus == XtWindow(toplevel) || cur_focus == toplevel_parent) {
/* notice user when no window focused or the xvkbd window is focused */
XBell(dpy, 0);
return;
}
}
found = FALSE;
keycode = 0;
if (keysym != NoSymbol) {
for (phase = 0; phase < 2; phase++) {
for (keycode = min_keycode; !found && (keycode <= max_keycode); keycode++) {
/* Determine keycode for the keysym: we use this instead
of XKeysymToKeycode() because we must know shift_state, too */
inx = (keycode - min_keycode) * keysym_per_keycode;
if (keysym_table[inx] == keysym) {
shift &= ~altgr_mask;
if (keysym_table[inx + 1] != NoSymbol) shift &= ~ShiftMask;
found = TRUE;
break;
} else if (keysym_table[inx + 1] == keysym) {
shift &= ~altgr_mask;
shift |= ShiftMask;
found = TRUE;
break;
}
}
if (!found && altgr_mask && 3 <= keysym_per_keycode) {
for (keycode = min_keycode; !found && (keycode <= max_keycode); keycode++) {
inx = (keycode - min_keycode) * keysym_per_keycode;
if (keysym_table[inx + 2] == keysym) {
shift &= ~ShiftMask;
shift |= altgr_mask;
found = TRUE;
break;
} else if (4 <= keysym_per_keycode && keysym_table[inx + 3] == keysym) {
shift |= ShiftMask | altgr_mask;
found = TRUE;
break;
}
}
}
if (found || !appres.auto_add_keysym) break;
if (0xF000 <= keysym) {
/* for special keys such as function keys,
first try to add it in the non-shifted position of the keymap */
if (AddKeysym(keysym, TRUE) == NoSymbol) AddKeysym(keysym, FALSE);
} else {
AddKeysym(keysym, FALSE);
}
}
if (appres.debug) {
if (found)
fprintf(stderr, "SendKeyPressedEvent: keysym=0x%lx, keycode=%ld, shift=0x%lX\n",
(long)keysym, (long)keycode, (long)shift);
else
fprintf(stderr, "SendKeyPressedEvent: keysym=0x%lx - keycode not found\n",
(long)keysym);
}
}
event.display = target_dpy;
event.window = cur_focus;
event.root = RootWindow(event.display, DefaultScreen(event.display));
event.subwindow = None;
event.time = CurrentTime;
event.x = 1;
event.y = 1;
event.x_root = 1;
event.y_root = 1;
event.same_screen = TRUE;
#ifdef USE_XTEST
if (appres.xtest) {
Window root, child;
int root_x, root_y, x, y;
unsigned int mask;
XQueryPointer(target_dpy, event.root, &root, &child, &root_x, &root_y, &x, &y, &mask);
event.type = KeyRelease;
event.state = 0;
if (mask & ControlMask) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Control_L);
SendEvent(&event);
}
if (mask & alt_mask) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Alt_L);
SendEvent(&event);
}
if (mask & meta_mask) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Meta_L);
SendEvent(&event);
}
if (mask & altgr_mask) {
if (0 < appres.altgr_keycode)
event.keycode = appres.altgr_keycode;
else
event.keycode = XKeysymToKeycode(target_dpy, altgr_keysym);
SendEvent(&event);
}
if (mask & ShiftMask) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Shift_L);
SendEvent(&event);
}
if (mask & LockMask) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Caps_Lock);
SendEvent(&event);
}
}
#endif
event.type = KeyPress;
event.state = 0;
if (shift & ControlMask) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Control_L);
SendEvent(&event);
event.state |= ControlMask;
}
if (shift & alt_mask) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Alt_L);
SendEvent(&event);
event.state |= alt_mask;
}
if (shift & meta_mask) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Meta_L);
SendEvent(&event);
event.state |= meta_mask;
}
if (shift & altgr_mask) {
if (0 < appres.altgr_keycode)
event.keycode = appres.altgr_keycode;
else
event.keycode = XKeysymToKeycode(target_dpy, altgr_keysym);
SendEvent(&event);
event.state |= altgr_mask;
}
if (shift & ShiftMask) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Shift_L);
SendEvent(&event);
event.state |= ShiftMask;
}
if (keysym != NoSymbol) { /* send event for the key itself */
event.keycode = found ? keycode : XKeysymToKeycode(target_dpy, keysym);
if (event.keycode == NoSymbol) {
if ((keysym & ~0x7f) == 0 && isprint(keysym))
fprintf(stderr, "%s: no such key: %c\n",
PROGRAM_NAME, (char)keysym);
else if (XKeysymToString(keysym) != NULL)
fprintf(stderr, "%s: no such key: keysym=%s (0x%lX)\n",
PROGRAM_NAME, XKeysymToString(keysym), (long)keysym);
else
fprintf(stderr, "%s: no such key: keysym=0x%lX\n",
PROGRAM_NAME, (long)keysym);
XBell(dpy, 0);
} else {
SendEvent(&event);
event.type = KeyRelease;
SendEvent(&event);
}
}
event.type = KeyRelease;
if (shift & ShiftMask) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Shift_L);
SendEvent(&event);
event.state &= ~ShiftMask;
}
if (shift & altgr_mask) {
if (0 < appres.altgr_keycode)
event.keycode = appres.altgr_keycode;
else
event.keycode = XKeysymToKeycode(target_dpy, altgr_keysym);
SendEvent(&event);
event.state &= ~altgr_mask;
}
if (shift & meta_mask) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Meta_L);
SendEvent(&event);
event.state &= ~meta_mask;
}
if (shift & alt_mask) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Alt_L);
SendEvent(&event);
event.state &= ~alt_mask;
}
if (shift & ControlMask) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Control_L);
SendEvent(&event);
event.state &= ~ControlMask;
}
if (appres.no_sync) XFlush(dpy);
}
/*
* Word completion - list of words which match the prefix entered
* via xvkbd can be listed, and choosing one of them will send the
* suffix to the clients.
* Words for completion will be read from dictionary file specified
* with xvkbd.dictFile resource, such as /usr/dict/words.
*/
static Widget completion_panel = None;
static Widget completion_entry = None;
static Widget completion_list = None;
static Widget props_dict_entry = None;
static char completion_text[100] = "";
#define HASH_SIZE 100
#define Hash(str) ((toupper(str[0]) * 26 + toupper(str[1])) % HASH_SIZE)
static struct WORDLIST {
struct WORDLIST *next;
char *word;
} completion_words[HASH_SIZE];
static int n_completion_words = 0;
#define MAX_WORDS 200
static String word_list[MAX_WORDS + 1];
static int n_word_list = 0;
static void SetDefaultDictionary(void)
{
strncpy(dict_filename, appres.dict_file, sizeof(dict_filename));
XtVaSetValues(props_dict_entry, XtNstring, dict_filename, NULL);
}
static void ReadCompletionDictionary(void)
{
static Boolean first = TRUE;
static char cur_dict_filename[PATH_MAX] = "";
FILE *fp;
struct WORDLIST *node_ptr;
char str[50];
int i;
struct WORDLIST *p;
if (strcmp(cur_dict_filename, dict_filename) == 0) return;
strcpy(cur_dict_filename, dict_filename);
if (!first) {
int cnt = 0;
for (i = 0; i < HASH_SIZE; i++) {
while (completion_words[i].next != NULL) {
p = completion_words[i].next;
completion_words[i].next = p->next;
free(p);
cnt++;
}
}
if (appres.debug)
fprintf(stderr, "ReadCompletionDictionary: %d words freed\n", cnt);
}
first = FALSE;
for (i = 0; i < HASH_SIZE; i++) {
completion_words[i].next = NULL;
completion_words[i].word = NULL;
}
n_completion_words = 0;
fp = fopen(dict_filename, "r");
if (fp == NULL) {
fprintf(stderr, "%s: can't read dictionary file %s: %s\n",
PROGRAM_NAME, dict_filename, strerror(errno));
} else {
while (fgets(str, sizeof(str) - 1, fp)) {
if (3 < strlen(str)) {
str[strlen(str) - 1] = '\0';
node_ptr = &completion_words[Hash(str)];
while (node_ptr->word != NULL) node_ptr = node_ptr->next;
node_ptr->word = XtNewString(str);
node_ptr->next = malloc(sizeof(struct WORDLIST));
node_ptr->next->next = NULL;
node_ptr->next->word = NULL;
n_completion_words++;
}
}
fclose(fp);
if (appres.debug)
fprintf(stderr, "ReadCompletionDictionary: %d words allocated\n", n_completion_words);
}
}
static void AddToCompletionText(KeySym keysym)
{
int len;
struct WORDLIST *node_ptr;
if (completion_entry != None) {
if (n_completion_words == 0) {
XtVaSetValues(completion_entry, XtNlabel, "Couldn't read dictionary file", NULL);
return;
}
len = strlen(completion_text);
if (keysym == XK_BackSpace || keysym == XK_Delete) {
if (0 < len) completion_text[len - 1] = '\0';
} else if (keysym != NoSymbol
&& !ispunct((char)keysym) && !isspace((char)keysym)) {
if (len < sizeof(completion_text) - 2) {
completion_text[len + 1] = '\0';
completion_text[len] = keysym;
}
} else {
completion_text[0] = '\0';
}
XtVaSetValues(completion_entry, XtNlabel, completion_text, NULL);
n_word_list = 0;
if (2 <= strlen(completion_text)) {
node_ptr = &completion_words[Hash(completion_text)];
while (node_ptr->word != NULL && n_word_list < MAX_WORDS) {
if (strlen(completion_text) + 1 < strlen(node_ptr->word) &&
strncasecmp(node_ptr->word, completion_text, strlen(completion_text)) == 0) {
word_list[n_word_list] = node_ptr->word;
n_word_list = n_word_list + 1;
}
node_ptr = node_ptr->next;
}
}
word_list[n_word_list] = NULL;
XawListChange(completion_list, word_list, 0, 0, TRUE);
}
}
static void CompletionWordSelected(Widget w, XtPointer client_data, XtPointer call_data)
{
Boolean capitalize;
unsigned char ch;
int n, i;
n = ((XawListReturnStruct *)call_data)->list_index;
if (0 <= n && n < n_word_list) {
capitalize = TRUE;
for (i = 0; completion_text[i] != '\0'; i++) {
if (islower(completion_text[i])) capitalize = FALSE;
}
for (i = strlen(completion_text); word_list[n][i] != '\0'; i++) {
ch = word_list[n][i];
if (capitalize) ch = toupper(ch);
SendKeyPressedEvent(ch, 0);
}
}
AddToCompletionText(NoSymbol);
}
static void PopupCompletionPanel(void)
{
Widget form, label, view;
char msg[100];
if (completion_panel == None) {
completion_panel = XtVaCreatePopupShell("completion_panel", transientShellWidgetClass,
toplevel, NULL);
form = XtVaCreateManagedWidget("form", formWidgetClass, completion_panel, NULL);
label = XtVaCreateManagedWidget("label", labelWidgetClass, form, NULL);
completion_entry = XtVaCreateManagedWidget("entry", labelWidgetClass, form,
XtNfromHoriz, label, NULL);
view = XtVaCreateManagedWidget("view", viewportWidgetClass, form,
XtNfromVert, label, NULL);
completion_list = XtVaCreateManagedWidget("list", listWidgetClass, view, NULL);
XtAddCallback(completion_list, XtNcallback, CompletionWordSelected, NULL);
XtRealizeWidget(completion_panel);
XSetWMProtocols(dpy, XtWindow(completion_panel), &wm_delete_window, 1);
XtPopup(completion_panel, XtGrabNone);
AddToCompletionText(NoSymbol);
XFlush(dpy);
} else {
XtPopup(completion_panel, XtGrabNone);
}
ReadCompletionDictionary();
sprintf(msg, "%d words in the dictionary", n_completion_words);
XtVaSetValues(completion_entry, XtNlabel, msg, NULL);
completion_text[0] = '\0';
n_word_list = 0;
word_list[n_word_list] = NULL;
XawListChange(completion_list, word_list, 0, 0, TRUE);
}
/*
* Send given string to the focused window as if the string
* is typed from a keyboard.
*/
static void KeyPressed(Widget w, char *key, char *data);
static void SendString(const unsigned char *str)
{
const unsigned char *cp, *cp2; /* I remember "unsigned" might be required for some systems */
char key[50];
int len;
int val;
Window target_root, child, junk_w;
int junk_i;
unsigned junk_u;
int cur_x, cur_y;
shift_state = 0;
for (cp = str; *cp != '\0'; cp++) {
if (0 < appres.text_delay)
usleep(appres.text_delay * 1000);
if (appres.wait_idle && strlen(appres.wait_idle) > 0) {
int pid = atoi(appres.wait_idle);
int ret;
do {
char cmd[80];
snprintf(cmd, sizeof(cmd), "/proc/%d/status", pid);
FILE *f = fopen(cmd, "r");
if (f == NULL) {
fprintf(stderr, "Process not found: %d\n", pid);
exit(-1);
}
fclose(f);
snprintf(cmd, sizeof(cmd), "grep 'State.*running' /proc/%d/status", pid);
ret = system(cmd);
if (ret == 0)
usleep(50);
} while (ret == 0);
}
if (*cp == '\\') {
cp++;
switch (*cp) {
case '\0':
fprintf(stderr, "%s: missing character after \"\\\"\n",
PROGRAM_NAME);
return;
case '[': /* we can write any keysym as "\[the-keysym]" here */
cp2 = strchr(cp, ']');
if (cp2 == NULL) {
fprintf(stderr, "%s: no closing \"]\" after \"\\[\"\n",
PROGRAM_NAME);
} else {
len = cp2 - cp - 1;
if (sizeof(key) <= len) len = sizeof(key) - 1;
strncpy(key, cp + 1, len);
key[len] = '\0';
KeyPressed(None, key, NULL);
cp = cp2;
}
break;
case 'S': shift_state |= ShiftMask; break;
case 'C': shift_state |= ControlMask; break;
case 'A': shift_state |= alt_mask; break;
case 'M': shift_state |= meta_mask; break;
case 'b': SendKeyPressedEvent(XK_BackSpace, shift_state); shift_state = 0; break;
case 't': SendKeyPressedEvent(XK_Tab, shift_state); shift_state = 0; break;
case 'n': SendKeyPressedEvent(XK_Linefeed, shift_state); shift_state = 0; break;
case 'r': SendKeyPressedEvent(XK_Return, shift_state); shift_state = 0; break;
case 'e': SendKeyPressedEvent(XK_Escape, shift_state); shift_state = 0; break;
case 'd': SendKeyPressedEvent(XK_Delete, shift_state); shift_state = 0; break;
case 'D': /* delay */
cp++;
if ('1' <= *cp && *cp <= '9') {
usleep((*cp - '0') * 100000);
} else {
fprintf(stderr, "%s: no digit after \"\\m\"\n",
PROGRAM_NAME);
}
break;
case 'm': /* simulate click mouse button */
cp++;
if ('1' <= *cp && *cp <= '9') {
if (appres.debug) fprintf(stderr, "XTestFakeButtonEvent(%d)\n", *cp - '0');
XTestFakeButtonEvent(target_dpy, *cp - '0', True, CurrentTime);
XTestFakeButtonEvent(target_dpy, *cp - '0', False, CurrentTime);
XFlush(dpy);
} else {
fprintf(stderr, "%s: no digit after \"\\m\"\n",
PROGRAM_NAME);
}
break;
case 'x':
case 'y': /* move mouse pointer */
sscanf(cp + 1, "%d", &val);
target_root = RootWindow(target_dpy, DefaultScreen(target_dpy));
XQueryPointer(target_dpy, target_root, &junk_w, &child,
&cur_x, &cur_y, &junk_i, &junk_i, &junk_u);
if (*cp == 'x') {
if (isdigit(*(cp + 1))) cur_x = val;
else cur_x += val;
} else {
if (isdigit(*(cp + 1))) cur_y = val;
else cur_y += val;
}
XWarpPointer(target_dpy, None, target_root, 0, 0, 0, 0, cur_x, cur_y);
XFlush(dpy);
cp++;
while (isdigit(*(cp + 1)) || *(cp + 1) == '+' || *(cp + 1) == '-') cp++;
break;
default:
SendKeyPressedEvent(*cp, shift_state);
shift_state = 0;
break;
}
} else {
SendKeyPressedEvent(*cp, shift_state);
shift_state = 0;
}
}
}
/*
* Send content of the file as if the it is typed from a keyboard.
*/
static void SendFileContent(const char *file)
{
FILE *fp;
int ch;
fp = stdin;
if (strcmp(file, "-") != 0) fp = fopen(file, "r");
if (fp == NULL) {
fprintf(stderr, "%s: can't read the file: %s\n", PROGRAM_NAME, file);
exit(1);
}
while ((ch = fgetc(fp)) != EOF) {
if (0 < appres.text_delay) usleep(appres.text_delay * 1000);
if (ch == '\n') { /* newline - send Return instead */
SendKeyPressedEvent(XK_Return, 0);
} else if (ch == '\033') { /* ESC */
SendKeyPressedEvent(XK_Escape, 0);
} else if (ch == '\177') { /* DEL */
SendKeyPressedEvent(XK_Delete, 0);
} else if (1 <= ch && ch <= 26) { /* Ctrl-x */
SendKeyPressedEvent('a' + ch - 1, ControlMask);
} else { /* normal characters */
SendKeyPressedEvent(ch, 0);
}
}
if (strcmp(file, "-") != 0) fclose(fp);
}
/*
* Highlight/unhighligh spcified modifier key on the screen.
*/
static void Highlight(char *name, Boolean state)
{
char name1[50];
Widget w;
sprintf(name1, "*%s", name);
w = XtNameToWidget(toplevel, name1);
if (w != None) {
if (strstr(name, "Focus") != NULL) {
if (target_dpy == dpy)
XtVaSetValues(w, XtNbackground, appres.focus_background, NULL);
else
XtVaSetValues(w, XtNbackground, appres.remote_focus_background, NULL);
if (state)
XtVaSetValues(w, XtNforeground, appres.highlight_foreground, NULL);
else
XtVaSetValues(w, XtNforeground, appres.special_foreground, NULL);
} else {
if (state)
XtVaSetValues(w, XtNbackground, appres.highlight_background,
XtNforeground, appres.highlight_foreground, NULL);
else
XtVaSetValues(w, XtNbackground, appres.special_background,
XtNforeground, appres.special_foreground, NULL);
}
}
}
/*
* Highlight/unhighligh keys on the screen to reflect the state.
*/
static void RefreshShiftState(Boolean force)
{
static Boolean first = TRUE;
static int last_shift_state = 0;
static int last_mouse_shift = 0;
static int last_num_lock_state = FALSE;
static Display *last_target_dpy = NULL;
static long last_focus = 0;
int cur_shift;
int changed;
int first_row, row, col;
Boolean shifted;
char *label;
int mask;
cur_shift = shift_state | mouse_shift;
changed = cur_shift ^ (last_shift_state | last_mouse_shift);
if (first || force) changed = 0xffff;
if (changed & ShiftMask) {
Highlight("Shift_L", cur_shift & ShiftMask);
Highlight("Shift_R", cur_shift & ShiftMask);
}
if (changed & ControlMask) {
Highlight("Control_L", cur_shift & ControlMask);
Highlight("Control_R", cur_shift & ControlMask);
}
if (changed & alt_mask) {
Highlight("Alt_L", cur_shift & alt_mask);
Highlight("Alt_R", cur_shift & alt_mask);
}
if (changed & meta_mask) {
Highlight("Meta_L", cur_shift & meta_mask);
Highlight("Meta_R", cur_shift & meta_mask);
}
if (changed & LockMask) {
Highlight("Caps_Lock", cur_shift & LockMask);
}
if (changed & altgr_mask) {
Highlight("Mode_switch", cur_shift & altgr_mask);
}
if (last_num_lock_state != appres.num_lock_state) {
Highlight("Num_Lock", appres.num_lock_state);
Highlight("keypad_panel*Num_Lock", appres.num_lock_state);
}
if (last_target_dpy != target_dpy || last_focus != focused_window) {
Highlight("Focus", focused_window != 0);
Highlight("keypad*Focus", focused_window != 0);
Highlight("keypad_panel*Focus", focused_window != 0);
last_target_dpy = target_dpy;
last_focus = focused_window;
}
mask = ShiftMask | LockMask | altgr_mask;
changed = (shift_state & mask) ^ (last_shift_state & mask);
if (first || force) changed = TRUE;
if (changed && !appres.keypad_only
&& (appres.modal_keytop || toplevel_height < appres.modal_threshold)) {
first_row = appres.function_key ? 0 : 1;
for (row = first_row; row < NUM_KEY_ROWS; row++) {
for (col = 0; col < NUM_KEY_COLS; col++) {
shifted = (shift_state & ShiftMask);
if (key_widgets[row][col] != None) {
if ((shift_state & altgr_mask) && altgr_key_labels[row][col] != NULL) {
if (shifted && shift_altgr_key_labels[row][col] != NULL)
label = shift_altgr_key_labels[row][col];
else
label = altgr_key_labels[row][col];
} else {
if ((shift_state & LockMask)
&& isalpha(keys_normal[row][col][0]) && keys_normal[row][col][1] == '\0')
shifted = !shifted;
if (shifted && shift_key_labels[row][col] != NULL)
label = shift_key_labels[row][col];
else
label = normal_key_labels[row][col];
}
if (label == NULL) {
fprintf(stderr, "%s: no label for key %d,%d\n", PROGRAM_NAME, row, col);
label = "";
}
if (strcmp(label, "space") == 0) label = "";
XtVaSetValues(key_widgets[row][col], XtNlabel, label, NULL);
}
}
}
}
last_shift_state = shift_state;
last_mouse_shift = mouse_shift;
last_num_lock_state = appres.num_lock_state;
first = FALSE;
#ifdef USE_XTEST
if (appres.xtest && strlen(appres.positive_modifiers) != 0) {
/* modifiers specified in positiveModifiers resouce will be hold down
so that it can be used with, for example, mouse operations */
Window root, child;
int root_x, root_y, x, y;
unsigned int mask;
XKeyEvent event;
event.display = target_dpy;
event.window = RootWindow(event.display, DefaultScreen(event.display));
event.root = event.window;
event.subwindow = None;
event.time = CurrentTime;
event.x = 1;
event.y = 1;
event.x_root = 1;
event.y_root = 1;
event.same_screen = TRUE;
event.state = 0;
XQueryPointer(target_dpy, event.root, &root, &child, &root_x, &root_y, &x, &y, &mask);
if (strstr(appres.positive_modifiers, "shift") != NULL
&& (shift_state & ShiftMask) != (mask & ShiftMask)) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Shift_L);
event.type = (shift_state & ShiftMask) ? KeyPress : KeyRelease;
SendEvent(&event);
}
if (strstr(appres.positive_modifiers, "control") != NULL
&& (shift_state & ControlMask) != (mask & ControlMask)) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Control_L);
event.type = (shift_state & ControlMask) ? KeyPress : KeyRelease;
SendEvent(&event);
}
if (strstr(appres.positive_modifiers, "alt") != NULL
&& (shift_state & alt_mask) != (mask & alt_mask)) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Alt_L);
event.type = (shift_state & alt_mask) ? KeyPress : KeyRelease;
SendEvent(&event);
}
if (strstr(appres.positive_modifiers, "meta") != NULL
&& (shift_state & meta_mask) != (mask & meta_mask)) {
event.keycode = XKeysymToKeycode(target_dpy, XK_Meta_L);
event.type = (shift_state & meta_mask) ? KeyPress : KeyRelease;
SendEvent(&event);
}
}
#endif
}
/*
* This function will be called when mouse button is pressed on a key
* on the screen. Most operation will be performed in KeyPressed()
* which will be called as callback for the Command widgets, and we
* only need to check which mouse button is pressed here.
*/
static unsigned int n_key_repeat;
static void ButtonDownAction(Widget w, XEvent *event, String *pars, Cardinal *n_pars)
{
n_key_repeat = 0;
switch (event->xbutton.button) {
case Button2:
mouse_shift |= ControlMask;
break;
case Button3:
case Button4:
mouse_shift |= ShiftMask;
break;
}
RefreshShiftState(FALSE);
}
/*
* This function will be called when mouse button is released on a key
* on the screen, after callback is called.
*/
static void ButtonUpAction(Widget w, XEvent *event, String *pars, Cardinal *n_pars)
{
if (appres.quick_modifiers) {
if (n_key_repeat == 1) XtCallCallbacks(w, XtNcallback, NULL);
}
mouse_shift = 0;
RefreshShiftState(FALSE);
}
/*
* Get the geometry of the base window.
*/
static char *GetWindowGeometry(Widget w)
{
static char geom[50];
Position x0, y0;
Window root;
int x1, y1;
unsigned int wd, ht, bd, dp;
XtVaGetValues(w, XtNx, &x0, XtNy, &y0, NULL);
XGetGeometry(dpy, XtWindow(w), &root, &x1, &y1, &wd, &ht, &bd, &dp);
sprintf(geom, "%dx%d+%d+%d", wd, ht, (int)(x0 - x1), (int)(y0 - y1));
return geom;
}
/*
* Set window manager hint.
* ("Extended Window Manager Hints", http://standards.freedesktop.org/wm-spec/)
*/
static void SetWindowManagerHint(Boolean initial)
{
if (initial) {
if (appres.wm_toolbar) {
Atom net_wm_window_type = XInternAtom(dpy, "_NET_WM_WINDOW_TYPE", False);
Atom net_wm_window_type_toolbar = XInternAtom(dpy, "_NET_WM_WINDOW_TYPE_TOOLBAR", False);
XChangeProperty(dpy, XtWindow(toplevel),
net_wm_window_type, XA_ATOM, 32,
PropModeReplace,
(unsigned char *) &net_wm_window_type_toolbar, 1);
if (appres.debug)
fprintf(stderr, "SetWindowManagerHint: set _NET_WM_WINDOW_TYPE_TOOLBAR\n");
}
}
if (!initial || appres.always_on_top) {
const int net_wm_state_remove = 0;
const int net_wm_state_add = 1;
Atom net_wm_state = XInternAtom(dpy, "_NET_WM_STATE", False);
Atom net_wm_state_above = XInternAtom(dpy, "_NET_WM_STATE_ABOVE", False);
XClientMessageEvent ev;
ev.type = ClientMessage;
ev.display = dpy;
ev.window = XtWindow(toplevel);
ev.message_type = net_wm_state;
ev.format = 32;
ev.data.l[0] = appres.always_on_top ? net_wm_state_add : net_wm_state_remove;
ev.data.l[1] = net_wm_state_above;
ev.data.l[2] = 0;
XSendEvent(dpy, RootWindow(dpy, DefaultScreen(dpy)),
FALSE, SubstructureNotifyMask | SubstructureRedirectMask,
(XEvent *)&ev);
if (appres.debug)
fprintf(stderr, "SetWindowManagerHint: _NET_WM_STATE_ABOVE = %d\n", ev.data.l[0]);
}
}
/*
* Restart the program to (possibly) change the keyboard layout,
* by loading the app-default file for the selected "customization".
*/
static void LayoutSelected(Widget w, char *key, char *data)
{
static char *env_lang, *env_xenv;
char name[50];
char customization[30] = "", lang[30] = "C";
char *xenv = NULL;
int i;
if (key != NULL) {
if (strcmp(key, "default") != 0) {
sscanf(key, "%29[^/]/%29s", customization, lang);
sprintf(name, "XVkbd-%s", customization);
xenv = XtResolvePathname(dpy, "app-defaults", name, NULL, NULL, NULL, 0, NULL);
if (xenv == NULL) {
fprintf(stderr, "%s: app-default file \"%s\" not installed\n",
PROGRAM_NAME, name);
}
}
env_lang = malloc(strlen("LC_ALL=") + strlen(lang) + 1);
sprintf(env_lang, "LC_ALL=%s", lang);
putenv(env_lang);
if (xenv != NULL) {
env_xenv = malloc(strlen("XENVIRONMENT=") + strlen(xenv) + 1);
sprintf(env_xenv, "XENVIRONMENT=%s", xenv);
putenv(env_xenv);
} else if (getenv("XENVIRONMENT") != NULL) {
putenv("XENVIRONMENT=");
}
}
for (i = 1; i < argc1; i++) {
if (strncmp(argv1[i], "-geom", strlen("-geom")) == 0) {
if (appres.inherit_geometry) {
argv1[i + 1] = GetWindowGeometry(toplevel);
} else if (i + 2 == argc1) {
argv1[i] = NULL;
argc1 = i;
}
break;
}
}
if (i == argc1 && appres.inherit_geometry) {
argv1[argc1++] = "-geometry";
argv1[argc1++] = GetWindowGeometry(toplevel);
argv1[argc1] = NULL;
}
if (appres.debug) {
fprintf(stderr, "XENVIRONMENT=%s, LC_ALL=%s\n", (xenv != NULL) ? xenv : "", lang);
fprintf(stderr, "Exec:");
for (i = 0; i < argc1; i++) fprintf(stderr, " %s", argv1[i]);
fprintf(stderr, "\n");
}
execvp(argv1[0], argv1);
}
/*
* Popup a window to select the (possibly) keyboard layout.
* The "XVkbd.customizations" resource will define the list,
* such as "default,german,swissgerman,french,latin1,jisx6004/ja".
* For example, "german" here will make this program to load
* "XVkbd-german" app-default file. Locale for each configuration
* can be specified by putting the locale name after "/".
*/
static void PopupLayoutPanel(void)
{
static Widget layout_panel = None;
char *customizations;
char *cp, *cp2;
Widget box, button;
if (layout_panel == None) {
layout_panel = XtVaCreatePopupShell("layout_panel", transientShellWidgetClass,
toplevel, NULL);
box = XtVaCreateManagedWidget("box", boxWidgetClass, layout_panel, NULL);
customizations = XtNewString(appres.customizations);
cp = strtok(customizations, " \t,");
while (cp != NULL) {
cp2 = strchr(cp, '/');
if (cp2 != NULL) *cp2 = '\0'; /* temporary remove '/' */
button = XtVaCreateManagedWidget(cp, commandWidgetClass, box, NULL);
if (cp2 != NULL) *cp2 = '/';
XtAddCallback(button, XtNcallback, (XtCallbackProc)LayoutSelected, XtNewString(cp));
cp = strtok(NULL, " \t,");
}
XtRealizeWidget(layout_panel);
XSetWMProtocols(dpy, XtWindow(layout_panel), &wm_delete_window, 1);
XtFree(customizations);
}
XtPopup(layout_panel, XtGrabNone);
}
/*
* Property panel
*/
static void SaveFunctionKey(void); /* forward */
static Widget props_panel = None;
static Widget autoclick_buttons = None;
static Widget click_buttons = None;
static Boolean props_panel_active = FALSE;
static void PropsItemToggled(Widget w, char *key, char *data)
{
Boolean last_wm_toolbar = appres.wm_toolbar;
if (!props_panel_active) return;
#ifdef USE_XTEST
XtVaGetValues(XtNameToWidget(props_panel, "*use_xtest"),
XtNstate, &appres.xtest, NULL);
#endif
XtVaGetValues(XtNameToWidget(props_panel, "*quick_modifiers"),
XtNstate, &appres.quick_modifiers, NULL);
XtVaGetValues(XtNameToWidget(props_panel, "*shift_lock"),
XtNstate, &appres.shift_lock, NULL);
XtVaGetValues(XtNameToWidget(props_panel, "*altgr_lock"),
XtNstate, &appres.altgr_lock, NULL);
XtVaGetValues(XtNameToWidget(props_panel, "*modifiers_lock"),
XtNstate, &appres.modifiers_lock, NULL);
XtVaGetValues(XtNameToWidget(props_panel, "*always_on_top"),
XtNstate, &appres.always_on_top, NULL);
XtVaGetValues(XtNameToWidget(props_panel, "*wm_toolbar"),
XtNstate, &appres.wm_toolbar, NULL);
XtVaGetValues(XtNameToWidget(props_panel, "*jump_pointer"),
XtNstate, &appres.jump_pointer, NULL);
appres.key_click_duration = (int)XawToggleGetCurrent(click_buttons);
appres.autoclick_delay = (int)XawToggleGetCurrent(autoclick_buttons);
SaveFunctionKey();
SetWindowManagerHint(FALSE);
if (appres.wm_toolbar != last_wm_toolbar) LayoutSelected(None, NULL, NULL);
}
static void PropsSetState(void)
{
#ifdef USE_XTEST
XtVaSetValues(XtNameToWidget(props_panel, "*use_xtest"),
XtNstate, appres.xtest, NULL);
#endif
XtVaSetValues(XtNameToWidget(props_panel, "*quick_modifiers"),
XtNstate, appres.quick_modifiers, NULL);
XtVaSetValues(XtNameToWidget(props_panel, "*shift_lock"),
XtNstate, appres.shift_lock, NULL);
if (XtNameToWidget(toplevel, "*Mode_switch") == None) {
XtSetSensitive(XtNameToWidget(props_panel, "*altgr_lock"), FALSE);
XtVaSetValues(XtNameToWidget(props_panel, "*altgr_lock"),
XtNstate, FALSE, NULL);
} else {
XtVaSetValues(XtNameToWidget(props_panel, "*altgr_lock"),
XtNstate, appres.altgr_lock, NULL);
}
XtVaSetValues(XtNameToWidget(props_panel, "*modifiers_lock"),
XtNstate, appres.modifiers_lock, NULL);
XtVaSetValues(XtNameToWidget(props_panel, "*always_on_top"),
XtNstate, appres.always_on_top, NULL);
XtVaSetValues(XtNameToWidget(props_panel, "*wm_toolbar"),
XtNstate, appres.wm_toolbar, NULL);
XtVaSetValues(XtNameToWidget(props_panel, "*jump_pointer"),
XtNstate, appres.jump_pointer, NULL);
XawToggleSetCurrent(click_buttons, (XtPointer)appres.key_click_duration);
XawToggleSetCurrent(autoclick_buttons, (XtPointer)appres.autoclick_delay);
}
static void ClosePropsPanel(void)
{
XtPopdown(props_panel);
XFlush(dpy);
SaveFunctionKey();
if (completion_panel != None) XtPopdown(completion_panel);
}
static void PopupPropsPanel(void)
{
static char *props_items[] = {
"quick_modifiers",
"shift_lock", "altgr_lock", "modifiers_lock",
"always_on_top",
"wm_toolbar",
#ifdef USE_XTEST
"use_xtest",
#endif
"jump_pointer",
};
if (props_panel == None) {
Widget label, button;
Widget form, w;
int i;
int val;
props_panel = XtVaCreatePopupShell("props_panel", transientShellWidgetClass,
toplevel, NULL);
form = XtVaCreateManagedWidget("form", formWidgetClass, props_panel, NULL);
w = None;
for (i = 0; i < XtNumber(props_items); i++) {
w = XtVaCreateManagedWidget(props_items[i], toggleWidgetClass,
form, XtNfromVert, w, NULL);
XtAddCallback(w, XtNcallback, (XtCallbackProc)PropsItemToggled,
(XtPointer)props_items[i]);
}
label = XtVaCreateManagedWidget("click", labelWidgetClass,
form, XtNfromVert, w, NULL);
button = XtVaCreateManagedWidget("OFF", toggleWidgetClass,
form, XtNfromVert, w, XtNfromHoriz, label,
XtNwidth, 0, XtNhorizDistance, 0, NULL);
XtVaSetValues(button, XtNradioGroup, button, XtNradioData, (XtPointer)0, NULL);
XtAddCallback(button, XtNcallback, (XtCallbackProc)PropsItemToggled,
(XtPointer)0);
click_buttons = button;
for (val = 1; val <= 50; val *= 2) {
char s1[10];
sprintf(s1, "%dms", val);
button = XtVaCreateManagedWidget(s1, toggleWidgetClass,
form, XtNfromVert, w, XtNfromHoriz, button,
XtNradioData, (XtPointer)val,
XtNradioGroup, click_buttons,
XtNwidth, 0, XtNhorizDistance, 0, NULL);
XtAddCallback(button, XtNcallback, (XtCallbackProc)PropsItemToggled,
NULL);
}
w = label;
label = XtVaCreateManagedWidget("autoclick", labelWidgetClass,
form, XtNfromVert, w, NULL);
button = XtVaCreateManagedWidget("OFF", toggleWidgetClass,
form, XtNfromVert, w, XtNfromHoriz, label,
XtNwidth, 0, XtNhorizDistance, 0, NULL);
XtVaSetValues(button, XtNradioGroup, button, XtNradioData, (XtPointer)0, NULL);
XtAddCallback(button, XtNcallback, (XtCallbackProc)PropsItemToggled,
(XtPointer)0);
autoclick_buttons = button;
for (val = 500; val <= 1000; val += 100) {
char s1[10];
sprintf(s1, "%dms", val);
button = XtVaCreateManagedWidget(s1, toggleWidgetClass,
form, XtNfromVert, w, XtNfromHoriz, button,
XtNradioData, (XtPointer)val,
XtNradioGroup, autoclick_buttons,
XtNwidth, 0, XtNhorizDistance, 0, NULL);
XtAddCallback(button, XtNcallback, (XtCallbackProc)PropsItemToggled,
(XtPointer)val);
}
w = label;
label = XtVaCreateManagedWidget("dict_entry_label", labelWidgetClass,
form, XtNfromVert, w, NULL);
props_dict_entry = XtVaCreateManagedWidget("dict_entry", asciiTextWidgetClass, form,
XtNfromVert, w, XtNfromHoriz, label,
XtNuseStringInPlace, True,
XtNstring, dict_filename,
XtNeditType, XawtextEdit,
XtNlength, sizeof(dict_filename) - 1,
NULL);
button = XtVaCreateManagedWidget("dict_default_button", commandWidgetClass,
form, XtNfromVert, w, XtNfromHoriz, props_dict_entry,
NULL);
XtAddCallback(button, XtNcallback, (XtCallbackProc)SetDefaultDictionary, NULL);
w = XtVaCreateManagedWidget("dismiss", commandWidgetClass, form,
XtNfromVert, label, NULL);
XtAddCallback(w, XtNcallback, (XtCallbackProc)ClosePropsPanel, NULL);
XtRealizeWidget(props_panel);
XSetWMProtocols(dpy, XtWindow(props_panel), &wm_delete_window, 1);
}
XtPopup(props_panel, XtGrabNone);
PropsSetState();
props_panel_active = TRUE;
if (completion_panel != None) XtPopdown(completion_panel);
}
/*
* Callback for main menu (activated from "xvkbd" logo).
*/
static Widget about_panel = None;
static Widget keypad_panel = None;
static Widget sun_fkey_panel = None;
static Widget deadkey_panel = None;
static Widget display_panel = None;
static Widget display_status = None;
#define DISPLAY_NAME_LENGTH 50
static void OpenRemoteDisplay(Widget w, char *display_name, char *data)
{
static char name[DISPLAY_NAME_LENGTH + 10];
char *cp;
focused_window = None;
focused_subwindow = None;
if (target_dpy != NULL && target_dpy != dpy) XCloseDisplay(target_dpy);
strncpy(name, (display_name == NULL) ? "" : display_name, sizeof(name));
for (cp = name; isascii(*cp) && isprint(*cp); cp++) ;
*cp = '\0';
if (strlen(name) == 0) {
target_dpy = dpy;
XtVaSetValues(display_status, XtNlabel, "Disconnected - local display selected", NULL);
XtPopdown(display_panel);
} else {
if (strchr(name, ':') == NULL) strcat(name, ":0");
target_dpy = XOpenDisplay(name);
if (target_dpy == NULL) {
XtVaSetValues(display_status, XtNlabel, "Couldn't connect to the display", NULL);
target_dpy = dpy;
XBell(dpy, 0);
} else {
XtVaSetValues(display_status, XtNlabel, "Connected", NULL);
XtPopdown(display_panel);
}
}
ReadKeymap();
if (!altgr_mask && appres.auto_add_keysym) AddModifier(XK_Mode_switch);
RefreshMainMenu();
RefreshShiftState(FALSE);
}
static void MenuSelected(Widget w, char *key)
{
Widget form;
if (strcmp(key, "man") == 0) {
if (!appres.secure) system(appres.show_manual_command);
} else if (strcmp(key, "about") == 0) {
if (about_panel == None) {
about_panel = XtVaCreatePopupShell("about_panel", transientShellWidgetClass,
toplevel, NULL);
XtVaCreateManagedWidget("message", labelWidgetClass, about_panel,
XtNlabel, appres.description, NULL);
XtRealizeWidget(about_panel);
XSetWMProtocols(dpy, XtWindow(about_panel), &wm_delete_window, 1);
}
XtPopup(about_panel, XtGrabNone);
} else if (strcmp(key, "keypad") == 0) {
if (keypad_panel == None) {
keypad_panel = XtVaCreatePopupShell("keypad_panel", transientShellWidgetClass,
toplevel, NULL);
form = XtVaCreateManagedWidget("form", formWidgetClass, keypad_panel, NULL);
MakeKeypad(form, None, None);
XtRealizeWidget(keypad_panel);
XSetWMProtocols(dpy, XtWindow(keypad_panel), &wm_delete_window, 1);
}
XtPopup(keypad_panel, XtGrabNone);
} else if (strcmp(key, "sun_fkey") == 0) {
if (sun_fkey_panel == None) {
sun_fkey_panel = XtVaCreatePopupShell("sun_fkey_panel", transientShellWidgetClass,
toplevel, NULL);
form = XtVaCreateManagedWidget("form", formWidgetClass, sun_fkey_panel, NULL);
MakeSunFunctionKey(form, None, None);
XtRealizeWidget(sun_fkey_panel);
XSetWMProtocols(dpy, XtWindow(sun_fkey_panel), &wm_delete_window, 1);
}
XtPopup(sun_fkey_panel, XtGrabNone);
} else if (strcmp(key, "deadkey") == 0) {
if (deadkey_panel == None) {
deadkey_panel = XtVaCreatePopupShell("deadkey_panel", transientShellWidgetClass,
toplevel, NULL);
form = XtVaCreateManagedWidget("form", formWidgetClass, deadkey_panel, NULL);
MakeDeadkeyPanel(form);
XtRealizeWidget(deadkey_panel);
XSetWMProtocols(dpy, XtWindow(deadkey_panel), &wm_delete_window, 1);
}
XtPopup(deadkey_panel, XtGrabNone);
} else if (strcmp(key, "completion") == 0) {
PopupCompletionPanel();
} else if (strcmp(key, "select_layout") == 0) {
PopupLayoutPanel();
} else if (strcmp(key, "edit_fkey") == 0) {
PopupFunctionKeyEditor();
} else if (strcmp(key, "show_keypad") == 0
|| strcmp(key, "show_functionkey") == 0) {
if (strcmp(key, "show_keypad") == 0) appres.keypad = !appres.keypad;
else appres.function_key = !appres.function_key;
MakeKeyboard(TRUE);
} else if (strcmp(key, "props") == 0) {
PopupPropsPanel();
} else if (strcmp(key, "open_display") == 0) {
if (display_panel == None) {
Widget label, entry, button;
static char display_name[DISPLAY_NAME_LENGTH] = "";
display_panel = XtVaCreatePopupShell("display_panel", transientShellWidgetClass,
toplevel, NULL);
form = XtVaCreateManagedWidget("form", formWidgetClass, display_panel, NULL);
label = XtVaCreateManagedWidget("label", labelWidgetClass, form, NULL);
entry = XtVaCreateManagedWidget("entry", asciiTextWidgetClass, form,
XtNfromHoriz, label,
XtNuseStringInPlace, True,
XtNeditType, XawtextEdit,
XtNstring, display_name,
XtNlength, sizeof(display_name) - 1,
NULL);
button = XtVaCreateManagedWidget("ok", commandWidgetClass, form,
XtNfromHoriz, entry, NULL);
XtAddCallback(button, XtNcallback, (XtCallbackProc)OpenRemoteDisplay, (XtPointer)display_name);
display_status = XtVaCreateManagedWidget("status", labelWidgetClass, form,
XtNfromVert, label,
XtNlabel, "", NULL);
XtRealizeWidget(display_panel);
XSetWMProtocols(dpy, XtWindow(display_panel), &wm_delete_window, 1);
XtSetKeyboardFocus(display_panel, entry);
}
XtPopup(display_panel, XtGrabNone);
} else if (strcmp(key, "close_display") == 0) {
OpenRemoteDisplay(None, NULL, NULL);
} else if (strcmp(key, "quit") == 0) {
DeleteWindowProc(None, NULL, NULL, NULL);
}
}
static void ClosePopupPanel(Widget w)
{
if (w == keypad_panel) {
XtDestroyWidget(w);
keypad_panel = None;
} else if (w == props_panel) {
ClosePropsPanel();
} else {
XtPopdown(w);
}
}
/*
* Iconify/uniconify the xvkbd window even if window manager is not
* available.
*/
static void IconifyWindow(Widget w, Boolean iconify)
{
static Widget iconified_window = None;
static Widget uniconify_button = None;
static Position x0, y0;
static int x1, y1;
static unsigned int wd, ht, bd, dp;
if (iconify) {
Window root;
int i;
XUnmapWindow(dpy, XtWindow(toplevel));
if (iconified_window == None) {
Widget box;
iconified_window = XtVaCreatePopupShell("iconified_window", transientShellWidgetClass,
toplevel, XtNoverrideRedirect, TRUE, NULL);
box = XtVaCreateManagedWidget("form", boxWidgetClass, iconified_window, NULL);
uniconify_button = XtVaCreateManagedWidget("uniconify_button", commandWidgetClass, box,
XtNbitmap, xvkbd_pixmap,
XtNhorizDistance, 10, XtNvertDistance, 0,
NULL);
XtAddCallback(uniconify_button, XtNcallback, (XtCallbackProc)IconifyWindow, FALSE);
XtRealizeWidget(iconified_window);
XSetWMProtocols(dpy, XtWindow(iconified_window), &wm_delete_window, 1);
}
XtVaGetValues(toplevel, XtNx, &x0, XtNy, &y0, NULL);
XGetGeometry(dpy, XtWindow(toplevel), &root, &x1, &y1, &wd, &ht, &bd, &dp);
XMoveResizeWindow(dpy, XtWindow(iconified_window), x0 + bd, y0 + bd, wd, ht);
XtPopup(iconified_window, XtGrabNone);
for (i = 9; 0 < i; i--) {
Dimension btn_wd, btn_ht;
Dimension wd1, ht1;
wd1 = wd * i / 10;
ht1 = ht * i / 10;
XtVaGetValues(uniconify_button, XtNwidth, &btn_wd, XtNheight, &btn_ht, NULL);
if (i == 1 || wd1 < btn_wd) wd1 = btn_wd;
if (i == 1 || ht1 < btn_ht) ht1 = btn_ht;
XMoveResizeWindow(dpy, XtWindow(iconified_window), x0 + bd, y0 + (ht - ht1) + bd, wd1, ht1);
XFlush(dpy);
usleep(10000);
}
} else {
if (iconified_window != None) XtPopdown(iconified_window);
XMapWindow(dpy, XtWindow(toplevel));
}
}
static void SignalUser1(void)
{
XWindowAttributes attr;
XGetWindowAttributes(dpy, XtWindow(toplevel), &attr);
IconifyWindow(None, attr.map_state != IsUnmapped);
XSync(dpy, FALSE);
}
/*
* This will be called when user pressed a key on the screen.
*/
static const char *FindFunctionKeyValue(const char *key, Boolean shiftable);
static void ShowBalloon(Widget w, XEvent *event, String *pars, Cardinal *n_pars);
static void KeyClick(void);
static void StopAutoclick(void);
static void KeyPressed(Widget w, char *key, char *data)
{
int row, col;
int cur_shift;
char *key1;
KeySym keysym;
Boolean shifted;
const char *value;
Boolean found;
if (appres.debug) fprintf(stderr, "KeyPressed: key=%s, widget=%lx\n", key, (long)w);
value = FindFunctionKeyValue(key, TRUE);
if (value != NULL) {
if (appres.debug) fprintf(stderr, "Assigned string: %s\n", value);
if (value[0] == '!') {
if (appres.debug) fprintf(stderr, "Launching: %s\n", value + 1);
if (!appres.secure) system(value + 1);
} else {
if (value[0] == '\\') value = value + 1;
if (appres.debug) fprintf(stderr, "Sending: %s\n", value);
SendString(value);
}
ShowBalloon(w, NULL, NULL, NULL);
return;
}
if (strncmp(key, "Shift", strlen("Shift")) == 0) {
if (shift_state & ShiftMask) SendKeyPressedEvent(NoSymbol, shift_state);
shift_state ^= ShiftMask;
} else if (strncmp(key, "Control", strlen("Control")) == 0) {
if (shift_state & ControlMask) SendKeyPressedEvent(NoSymbol, shift_state);
shift_state ^= ControlMask;
} else if (strncmp(key, "Alt", strlen("Alt")) == 0) {
if (shift_state & alt_mask) SendKeyPressedEvent(NoSymbol, shift_state);
shift_state ^= alt_mask;
} else if (strncmp(key, "Meta", strlen("Meta")) == 0) {
if (shift_state & meta_mask) SendKeyPressedEvent(NoSymbol, shift_state);
shift_state ^= meta_mask;
} else if (strcmp(key, "Caps_Lock") == 0) {
if (shift_state & LockMask) SendKeyPressedEvent(NoSymbol, shift_state);
shift_state ^= LockMask;
} else if (strcmp(key, "Mode_switch") == 0) {
if (shift_state & altgr_mask) SendKeyPressedEvent(NoSymbol, shift_state);
shift_state ^= altgr_mask;
} else if (strcmp(key, "Num_Lock") == 0) {
appres.num_lock_state = !appres.num_lock_state;
} else if (strcmp(key, "Focus") == 0) {
cur_shift = shift_state | mouse_shift;
if (cur_shift & ShiftMask) {
focused_window = None;
focused_subwindow = None;
} else {
GetFocusedWindow();
}
} else {
if (appres.quick_modifiers && mouse_shift == 0 && w != None) {
Window junk_w;
int junk_i;
unsigned junk_u;
int cur_x, cur_y;
Dimension btn_wd, btn_ht;
n_key_repeat = n_key_repeat + 1;
if (n_key_repeat == 1) return;
XtVaGetValues(w, XtNwidth, &btn_wd, XtNheight, &btn_ht, NULL);
XQueryPointer(dpy, XtWindow(w), &junk_w, &junk_w,
&junk_i, &junk_i, &cur_x, &cur_y, &junk_u);
mouse_shift = 0;
if (cur_x < 0 && btn_ht < cur_y) {
mouse_shift |= alt_mask; /* left-down */
} else {
if (cur_y < 0) mouse_shift |= ShiftMask; /* up */
else if (btn_ht < cur_y) mouse_shift |= meta_mask; /* down */
if (cur_x < 0) mouse_shift |= ControlMask; /* left */
else if (btn_wd < cur_x) mouse_shift |= altgr_mask; /* right */
}
}
cur_shift = shift_state | mouse_shift;
shifted = (cur_shift & ShiftMask);
key1 = key;
if (w != None) {
if (sscanf(key, "pad%d,%d", &row, &col) == 2) {
if (appres.num_lock_state) shifted = !shifted;
key1 = shifted ? keypad_shift[row][col]: keypad[row][col];
} else {
found = FALSE;
if (sscanf(key, "%d,%d", &row, &col) == 2) {
found = TRUE;
} else if (w != None) {
int first_row = appres.function_key ? 0 : 1;
for (row = first_row; row < NUM_KEY_ROWS; row++) {
for (col = 0; col < NUM_KEY_COLS; col++) {
if (key_widgets[row][col] == w) {
found = TRUE;
break;
}
}
if (col < NUM_KEY_COLS) break;
}
}
if (found) {
if ((cur_shift & LockMask)
&& isalpha(keys_normal[row][col][0]) && keys_normal[row][col][1] == '\0')
shifted = !shifted;
if ((cur_shift & altgr_mask) && keys_altgr[row][col] != NULL) {
if (shifted && keys_shift_altgr[row][col] != NULL) {
key1 = keys_shift_altgr[row][col];
if (strcmp(keys_altgr[row][col], keys_shift_altgr[row][col]) != 0)
cur_shift &= ~ShiftMask;
} else {
key1 = keys_altgr[row][col];
}
} else {
if (shifted && keys_shift[row][col] != NULL) {
key1 = keys_shift[row][col];
if (strcmp(keys_normal[row][col], keys_shift[row][col]) != 0)
cur_shift &= ~ShiftMask;
} else {
key1 = keys_normal[row][col];
}
}
} /* if (found) ... */
} /* if (sscanf(key, "pad%d,%d", ... */
} /* if (w != None) ... */
if (strlen(key1) == 1) {
SendKeyPressedEvent((KeySym)*key1 & 0xff, cur_shift);
AddToCompletionText((KeySym)*key1);
} else {
if (islower(key1[0]) && key1[1] == ':') {
switch (key1[0]) {
case 's': cur_shift |= ShiftMask; break;
case 'c': cur_shift |= ControlMask; break;
case 'a': cur_shift |= alt_mask; break;
case 'm': cur_shift |= meta_mask; break;
default: fprintf(stderr, "%s: unknown modidier: %s\n",
PROGRAM_NAME, key1); break;
}
key1 = key1 + 2;
}
keysym = XStringToKeysym(key1);
if ((!appres.keypad_keysym && strncmp(key1, "KP_", 3) == 0)
|| XKeysymToKeycode(target_dpy, keysym) == NoSymbol) {
switch ((unsigned)keysym) {
case XK_KP_Equal: keysym = XK_equal; break;
case XK_KP_Divide: keysym = XK_slash; break;
case XK_KP_Multiply: keysym = XK_asterisk; break;
case XK_KP_Add: keysym = XK_plus; break;
case XK_KP_Subtract: keysym = XK_minus; break;
case XK_KP_Enter: keysym = XK_Return; break;
case XK_KP_1: keysym = XK_1; break;
case XK_KP_2: keysym = XK_2; break;
case XK_KP_3: keysym = XK_3; break;
case XK_KP_4: keysym = XK_4; break;
case XK_KP_5: keysym = XK_5; break;
case XK_KP_6: keysym = XK_6; break;
case XK_KP_7: keysym = XK_7; break;
case XK_KP_8: keysym = XK_8; break;
case XK_KP_9: keysym = XK_9; break;
case XK_Shift_L: keysym = XK_Shift_R; break;
case XK_Shift_R: keysym = XK_Shift_L; break;
case XK_Control_L: keysym = XK_Control_R; break;
case XK_Control_R: keysym = XK_Control_L; break;
case XK_Alt_L: keysym = XK_Alt_R; break;
case XK_Alt_R: keysym = XK_Alt_L; break;
case XK_Meta_L: keysym = XK_Meta_R; break;
case XK_Meta_R: keysym = XK_Meta_L; break;
default:
if (keysym == NoSymbol || !appres.auto_add_keysym)
fprintf(stderr, "%s: no such key: %s\n",
PROGRAM_NAME, key1); break;
}
}
SendKeyPressedEvent(keysym, cur_shift);
AddToCompletionText(keysym);
if ((cur_shift & ControlMask) && (cur_shift & alt_mask)) {
if (strstr(XServerVendor(dpy), "XFree86") != NULL) {
if (strcmp(key1, "KP_Add") == 0) {
if (!appres.secure) system("xvidtune -next");
} else if (strcmp(key1, "KP_Subtract") == 0) {
if (!appres.secure) system("xvidtune -prev");
}
}
}
}
if (!appres.shift_lock)
shift_state &= ~ShiftMask;
if (!appres.modifiers_lock)
shift_state &= ~(ControlMask | alt_mask | meta_mask);
if (!appres.altgr_lock)
shift_state &= ~altgr_mask;
}
RefreshShiftState(FALSE);
if (w != None) {
KeyClick();
/* StopAutoclick(); */
}
}
/*
* Redefine keyboard layout.
* "spec" is a sequence of words separated with spaces, and it is
* usally specified in app-defaults file, as:
*
* xvkbd.AltGrKeys: \
* F1 F2 F3 F4 F5 F6 F7 F8 F9 F10 F11 F12 BackSpace \n\
* Escape \271 \262 \263 \243 \254 \251 { [ ] } \\ ' ^ ' \n\
* ...
*
* White spaces separate the keys, and " \n" (note that white space
* before the \n) separate the rows of keys.
*/
static void RedefineKeys(char *array[NUM_KEY_ROWS][NUM_KEY_COLS], const char *spec)
{
char *s = XtNewString(spec);
char *cp;
int row, col;
int key_rows = NUM_KEY_ROWS;
int key_cols = NUM_KEY_COLS;
for (row = 0; row < key_rows; row++) {
for (col = 0; col < key_cols; col++) array[row][col] = NULL;
}
row = 0;
col = 0;
cp = strtok(s, " ");
while (cp != NULL) {
if (*cp == '\n') {
row = row + 1;
col = 0;
cp = cp + 1;
}
if (*cp != '\0') {
if (key_rows <= row) {
fprintf(stderr, "%s: too many key rows: \"%s\" ignored\n",
PROGRAM_NAME, cp);
} else if (key_cols <= col) {
fprintf(stderr, "%s: too many keys in a row: \"%s\" ignored\n",
PROGRAM_NAME, cp);
} else {
array[row][col] = XtNewString(cp);
col = col + 1;
}
}
cp = strtok(NULL, " ");
}
XtFree(s);
}
/*
* Create keyboard on the screen.
*/
static Widget MakeKey(Widget parent, const char *name, const char *label, Pixel color)
{
static Pixmap up_pixmap = None;
static Pixmap down_pixmap = None;
static Pixmap left_pixmap = None;
static Pixmap right_pixmap = None;
static Pixmap back_pixmap = None;
Widget w;
Window scr = RootWindow(dpy, DefaultScreen(dpy));
char str[50];
int len;
if (!appres.auto_repeat
|| strncmp(name, "Shift", strlen("Shift")) == 0
|| strncmp(name, "Control", strlen("Control")) == 0
|| strncmp(name, "Alt", strlen("Alt")) == 0
|| strncmp(name, "Meta", strlen("Meta")) == 0
|| strcmp(name, "Caps_Lock") == 0
|| strcmp(name, "Mode_switch") == 0
|| strcmp(name, "Num_Lock") == 0
|| strcmp(name, "Focus") == 0) {
w = XtVaCreateManagedWidget(name, commandWidgetClass, parent,
XtNbackground, color, NULL);
} else {
w = XtVaCreateManagedWidget(name, repeaterWidgetClass, parent,
XtNbackground, color, NULL);
}
XtAddCallback(w, XtNcallback, (XtCallbackProc)KeyPressed, (XtPointer)name);
if (label != NULL) {
strncpy(str, label, sizeof(str) - 1);
if (strcmp(str, "space") == 0) strcpy(str, "");
len = strlen(str);
if (3 <= len) {
if (str[1] == '_') str[1] = ' ';
if (str[len - 2] == '_') str[len - 2] = ' ';
}
XtVaSetValues(w, XtNlabel, str, NULL);
if (strcmp(label, "up") == 0) {
if (up_pixmap == None)
up_pixmap = XCreateBitmapFromData(dpy, scr,
(char *)up_bits, up_width, up_height);
XtVaSetValues(w, XtNbitmap, up_pixmap, NULL);
} else if (strcmp(label, "down") == 0) {
if (down_pixmap == None)
down_pixmap = XCreateBitmapFromData(dpy, scr,
(char *)down_bits, down_width, down_height);
XtVaSetValues(w, XtNbitmap, down_pixmap, NULL);
} else if (strcmp(label, "left") == 0) {
if (left_pixmap == None)
left_pixmap = XCreateBitmapFromData(dpy, scr,
(char *)left_bits, left_width, left_height);
XtVaSetValues(w, XtNbitmap, left_pixmap, NULL);
} else if (strcmp(label, "right") == 0) {
if (right_pixmap == None)
right_pixmap = XCreateBitmapFromData(dpy, scr,
(char *)right_bits, right_width, right_height);
XtVaSetValues(w, XtNbitmap, right_pixmap, NULL);
} else if (strcmp(label, "back") == 0) {
if (back_pixmap == None)
back_pixmap = XCreateBitmapFromData(dpy, scr,
(char *)back_bits, back_width, back_height);
XtVaSetValues(w, XtNbitmap, back_pixmap, NULL);
}
}
return w;
}
static void MakeKeypad(Widget form, Widget from_vert, Widget from_horiz)
{
Widget key, left, upper;
Pixel color;
XFontStruct *font;
int row, col;
Widget keypad_box;
Widget keypad_row[NUM_KEYPAD_ROWS];
char name[50];
keypad_box = XtVaCreateManagedWidget("keypad", formWidgetClass, form, NULL);
if (from_horiz != None)
XtVaSetValues(keypad_box, XtNfromHoriz, from_horiz, NULL);
else
XtVaSetValues(keypad_box, XtNhorizDistance, 0, NULL);
if (from_vert != None)
XtVaSetValues(keypad_box, XtNfromVert, from_vert, NULL);
else
XtVaSetValues(keypad_box, XtNvertDistance, 0, NULL);
upper = None;
for (row = 0; row < NUM_KEYPAD_ROWS; row++) {
left = None;
for (col = 0; keypad[row][col] != NULL; col++) {
font = appres.keypad_font;
if (strlen(keypad_label[row][col]) == 1) font = appres.letter_font;
color = appres.special_background;
if (strcmp(keypad[row][col], "Focus") == 0)
color = appres.focus_background;
else if (strcmp(keypad_shift[row][col], ".") == 0
|| (strncmp(keypad_shift[row][col], "KP_", 3) == 0
&& isdigit(keypad_shift[row][col][3])))
color = appres.general_background;
strcpy(name, keypad[row][col]);
if (strcmp(name, "Focus") != 0 && strcmp(name, "Num_Lock") != 0)
sprintf(name, "pad%d,%d", row, col);
key = MakeKey(keypad_box, XtNewString(name),
keypad_label[row][col], color);
XtVaSetValues(key, XtNfont, font, NULL);
if (row != 0)
XtVaSetValues(key, XtNfromVert, keypad_row[row - 1], NULL);
if (left != None)
XtVaSetValues(key, XtNfromHoriz, left, NULL);
if (col == 0)
keypad_row[row] = key;
left = key;
}
}
}
static void MakeSunFunctionKey(Widget form, Widget from_vert, Widget from_horiz)
{
Widget key, left, upper;
int row, col;
Widget fkey_box;
Widget fkey_row[NUM_SUN_FKEY_ROWS];
fkey_box = XtVaCreateManagedWidget("fkey", formWidgetClass, form, NULL);
if (from_horiz != None)
XtVaSetValues(fkey_box, XtNfromHoriz, from_horiz, NULL);
else
XtVaSetValues(fkey_box, XtNhorizDistance, 0, NULL);
if (from_vert != None)
XtVaSetValues(fkey_box, XtNfromVert, from_vert, NULL);
else
XtVaSetValues(fkey_box, XtNvertDistance, 0, NULL);
upper = None;
for (row = 0; row < NUM_SUN_FKEY_ROWS; row++) {
left = None;
for (col = 0; sun_fkey[row][col] != NULL; col++) {
key = MakeKey(fkey_box, sun_fkey[row][col],
sun_fkey_label[row][col], appres.special_background);
XtVaSetValues(key, XtNfont, appres.keypad_font, NULL);
if (row != 0)
XtVaSetValues(key, XtNfromVert, fkey_row[row - 1], NULL);
if (left != None)
XtVaSetValues(key, XtNfromHoriz, left, NULL);
if (col == 0)
fkey_row[row] = key;
left = key;
}
}
}
static void MakeDeadkeyPanel(Widget form)
{
Widget deadkey_box, left, key;
char *deadkeys, *cp, *cp2;
deadkeys = XtNewString(appres.deadkeys);
deadkey_box = XtVaCreateManagedWidget("deadkey", formWidgetClass, form, NULL);
left = None;
cp = strtok(deadkeys, " \t,");
while (cp != NULL) {
cp2 = XtNewString(cp);
key = MakeKey(deadkey_box, cp2, NULL, appres.general_background);
if (left != None) XtVaSetValues(key, XtNfromHoriz, left, NULL);
left = key;
cp = strtok(NULL, " \t,");
}
XtFree(deadkeys);
}
static void RefreshMainMenu(void)
{
static Pixmap check_pixmap = None;
if (check_pixmap == None) {
check_pixmap = XCreateBitmapFromData(dpy, RootWindow(dpy, DefaultScreen(dpy)),
(char *)check_bits, check_width, check_height);
}
XtVaSetValues(XtNameToWidget(main_menu, "*show_keypad"),
XtNrightBitmap, appres.keypad ? check_pixmap : None, NULL);
XtVaSetValues(XtNameToWidget(main_menu, "*show_functionkey"),
XtNrightBitmap, appres.function_key ? check_pixmap : None, NULL);
XtSetSensitive(XtNameToWidget(main_menu, "*edit_fkey"), appres.function_key);
XtSetSensitive(XtNameToWidget(main_menu, "*close_display"), target_dpy != dpy);
}
static void MakeKeyboard(Boolean remake)
{
static char *main_menu_items[] = {
"about", "man", "keypad", "sun_fkey", "deadkey", "completion", "",
"select_layout",
"edit_fkey",
"show_keypad",
"show_functionkey",
"props",
"",
"open_display", "close_display", "",
"quit" };
Widget form, key, left;
Pixel color;
XFontStruct *font;
Dimension wd, max_wd;
int row, col, first_row;
char name[50], *label;
Widget key_box[NUM_KEY_ROWS];
Widget menu_entry;
int i;
#include "xvkbd.xbm"
#include "iconify.xbm"
if (remake) {
appres.geometry = GetWindowGeometry(toplevel);
XtUnrealizeWidget(toplevel);
XtDestroyWidget(XtNameToWidget(toplevel, "form"));
}
form = XtVaCreateManagedWidget("form", formWidgetClass, toplevel, NULL);
key_box[0] = None;
key_box[1] = None;
first_row = appres.function_key ? 0 : 1;
if (!appres.keypad_only) {
for (row = first_row; row < NUM_KEY_ROWS; row++) {
if (keys_normal[row][0] == NULL) continue;
sprintf(name, "row%d", row);
key_box[row] = XtVaCreateManagedWidget(name, formWidgetClass, form, NULL);
key_box[row + 1] = None;
if (row != first_row)
XtVaSetValues(key_box[row], XtNfromVert, key_box[row - 1], NULL);
else if (!appres.function_key)
XtVaSetValues(key_box[row], XtNvertDistance, 0, NULL);
left = None;
for (col = 0; keys_normal[row][col] != NULL; col++) {
strcpy(name, keys_normal[row][col]);
if (strcmp(name, "MainMenu") == 0) {
Widget iconify_button = None;
if (appres.minimizable) {
Pixmap iconify_pixmap = XCreateBitmapFromData(dpy, RootWindow(dpy, DefaultScreen(dpy)),
(char *)iconify_bits, iconify_width, iconify_height);
iconify_button = XtVaCreateManagedWidget("Iconify", commandWidgetClass, key_box[row],
XtNbitmap, iconify_pixmap, NULL);
XtAddCallback(iconify_button, XtNcallback, (XtCallbackProc)IconifyWindow, (void *)TRUE);
}
xvkbd_pixmap = XCreateBitmapFromData(dpy, RootWindow(dpy, DefaultScreen(dpy)),
(char *)xvkbd_bits, xvkbd_width, xvkbd_height);
key = XtVaCreateManagedWidget("MainMenu", menuButtonWidgetClass, key_box[row],
XtNbitmap, xvkbd_pixmap, XtNfromHoriz, iconify_button, NULL);
main_menu = XtVaCreatePopupShell("menu", simpleMenuWidgetClass, key, NULL);
for (i = 0; i < XtNumber(main_menu_items); i++) {
if (strlen(main_menu_items[i]) == 0) {
XtVaCreateManagedWidget("separator", smeLineObjectClass, main_menu, NULL);
} else {
menu_entry = XtVaCreateManagedWidget(main_menu_items[i], smeBSBObjectClass,
main_menu, NULL);
XtAddCallback(menu_entry, XtNcallback, (XtCallbackProc)MenuSelected,
(XtPointer)main_menu_items[i]);
}
}
} else {
label = appres.modal_keytop ? normal_key_labels[row][col] : key_labels[row][col];
if (isascii(name[0]) && isupper(name[0])) {
if (strcmp(name, "Focus") == 0) {
color = appres.focus_background;
font = appres.keypad_font;
} else {
color = appres.special_background;
if (label != NULL && strchr(label, '\n') != NULL) font = appres.keypad_font;
else font = appres.special_font;
}
} else {
color = appres.general_background;
font = appres.general_font;
if (isalpha(name[0])) font = appres.letter_font;
if (strcmp(name, "space") != 0) sprintf(name, "%d,%d", row, col);
}
key = MakeKey(key_box[row], XtNewString(name), label, color);
XtVaGetValues(key, XtNwidth, &wd, NULL);
if (wd <= 1) {
/* keys can be removed by setting its width to 1 */
XtDestroyWidget(key);
key = None;
} else {
XtVaSetValues(key, XtNfont, font, NULL);
#ifdef USE_I18N
if (font == appres.special_font || font == appres.keypad_font)
XtVaSetValues(key, XtNfontSet, appres.special_fontset, NULL);
#endif
}
}
if (key != None) {
if (left != None) XtVaSetValues(key, XtNfromHoriz, left, NULL);
left = key;
}
key_widgets[row][col] = key;
}
}
}
if (appres.keypad) MakeKeypad(form, key_box[0], key_box[1]);
if (!appres.keypad_only && appres.function_key && appres.keypad) {
XtVaCreateManagedWidget("banner", labelWidgetClass, form,
XtNfromHoriz, key_box[1],
XtNlabel, PROGRAM_NAME_WITH_VERSION, NULL);
}
XtRealizeWidget(toplevel);
SetWindowManagerHint(TRUE);
if (!remake && strlen(appres.geometry) == 0) {
Window root;
int x1, y1;
unsigned int wd, ht, bd, dp;
int max_wd, max_ht;
XGetGeometry(dpy, XtWindow(toplevel), &root, &x1, &y1, &wd, &ht, &bd, &dp);
max_wd = XtScreen(toplevel)->width * appres.max_width_ratio;
max_ht = XtScreen(toplevel)->height * appres.max_height_ratio;
if (appres.debug)
fprintf(stderr, "window size: %dx%d, max size: %dx%d\n", wd, ht, max_wd, max_ht);
if (max_wd < wd || max_ht < ht) {
if (max_wd < wd) wd = max_wd;
if (max_ht < ht) ht = max_ht;
if (appres.debug)
fprintf(stderr, "setting window size: %dx%d\n", wd, ht);
XResizeWindow(dpy, XtWindow(toplevel), wd, ht);
}
}
if (!appres.debug && key_box[first_row] != None) {
if (appres.keypad) {
XtVaGetValues(key_box[1], XtNwidth, &max_wd, NULL);
} else {
max_wd = 0;
for (row = first_row; row < NUM_KEY_ROWS && key_box[row] != None; row++) {
XtVaGetValues(key_box[row], XtNwidth, &wd, NULL);
if (max_wd < wd) max_wd = wd;
}
}
for (row = first_row; row < NUM_KEY_ROWS && key_box[row] != None; row++) {
XtVaSetValues(key_box[row], XtNwidth, max_wd, NULL);
}
}
if (0 < strlen(appres.geometry)) {
if (appres.wm_toolbar) {
if (appres.debug)
fprintf(stderr, "window fgeometry ignored; _NET_WM_WINDOW_TYPE_TOOLBAR set on\n");
} else {
if (appres.debug)
fprintf(stderr, "setting window geometry: %s\n", appres.geometry);
XtVaSetValues(toplevel, XtNgeometry, appres.geometry, NULL);
XtUnrealizeWidget(toplevel);
XtRealizeWidget(toplevel);
}
}
ReadKeymap();
if (main_menu != None) RefreshMainMenu();
RefreshShiftState(FALSE);
XtMapWidget(toplevel);
if (wm_delete_window == None)
wm_delete_window = XInternAtom(dpy, "WM_DELETE_WINDOW", FALSE);
XSetWMProtocols(dpy, XtWindow(toplevel), &wm_delete_window, 1);
XtVaGetValues(toplevel, XtNheight, &toplevel_height, NULL);
}
/*
* WM_DELETE_WINDOW has been sent - terminate the program.
*/
static void DeleteWindowProc(Widget w, XEvent *event,
String *pars, Cardinal *n_pars)
{
if (appres.nonexitable) {
XBell(dpy, 0);
} else {
shift_state = 0;
RefreshShiftState(TRUE);
XtDestroyApplicationContext(XtWidgetToApplicationContext(toplevel));
exit(0);
}
}
/*
* Callback for ConfigureNotify event, which will be invoked when
* the toplevel window is resized.
* We may need to switch the keytop labels when window becomes
* smaller than appres.modal_threshold, and vice versa.
*/
static void WindowResized(Widget w, XEvent *event,
String *pars, Cardinal *n_pars)
{
Dimension ht;
XtVaGetValues(toplevel, XtNheight, &ht, NULL);
if (appres.modal_threshold <= ht) {
if (toplevel_height < appres.modal_threshold) MakeKeyboard(TRUE);
} else {
toplevel_height = ht;
}
RefreshShiftState(TRUE);
}
/*
* Load list of text to be assigned to function keys.
* Each line contains name of the key (with optional modifier)
* and the text to be assigned to the key, as:
*
* F1 text for F1
* s:F2 text for Shift-F2
*/
#ifndef PATH_MAX
# define PATH_MAX 300
#endif
static char fkey_filename[PATH_MAX] = "";
static struct fkey_struct {
struct fkey_struct *next;
char *value;
} *fkey_list = NULL;
static void ReadFuncionKeys(void)
{
FILE *fp;
char str[200], key[200];
struct fkey_struct *sp = NULL, *new_node;
char len;
int val;
const char *home;
/* If KeyFile is not started with "/", consider the filename is relative to $HOME */
/* and put value of the $HOME environment variable before the KeyFile. */
/* To avoid possible buffer overflow, $HOME will not be added when resulting filename */
/* is too long. */
home = getenv("HOME");
if (appres.key_file[0] != '/' && home != NULL
&& strlen(home) + strlen(appres.key_file) + 1 < sizeof(fkey_filename))
sprintf(fkey_filename, "%s/%s", home, appres.key_file);
else
strncpy(fkey_filename, appres.key_file, sizeof(fkey_filename));
strncpy(dict_filename, appres.dict_file, sizeof(dict_filename));
fp = fopen(fkey_filename, "r");
if (fp == NULL) return;
while (fgets(str, sizeof(str) - 1, fp)) {
if (str[0] == '#') {
sscanf(&str[1], "%s %d", key, &val);
if (strcmp(key, "quick_modifiers") == 0)
appres.quick_modifiers = val;
else if (strcmp(key, "shift_lock") == 0)
appres.shift_lock = val;
else if (strcmp(key, "altgr_lock") == 0)
appres.altgr_lock = val;
else if (strcmp(key, "modifiers_lock") == 0)
appres.modifiers_lock = val;
else if (strcmp(key, "key_click") == 0)
appres.key_click_duration = val;
else if (strcmp(key, "autoclick") == 0)
appres.autoclick_delay = val;
else if (strcmp(key, "always_on_top") == 0)
appres.always_on_top = val;
else if (strcmp(key, "wm_toolbar") == 0)
appres.wm_toolbar = val;
else if (strcmp(key, "jump_pointer") == 0)
appres.jump_pointer = val;
else if (strcmp(key, "dict_file") == 0) {
sscanf(&str[1], "%*s %s", &key);
strncpy(dict_filename, key, sizeof(dict_filename));
}
} else if (isalpha(str[0])) {
len = strlen(str);
if (str[len - 1] == '\n') str[len - 1] = '\0';
new_node = malloc(sizeof(struct fkey_struct));
if (fkey_list == NULL) fkey_list = new_node;
else sp->next = new_node;
sp = new_node;
sp->next = NULL;
sp->value = XtNewString(str);
}
}
fclose(fp);
}
/*
* Edit string assigned for function keys.
* Modifiers (Shift, Ctrl, etc.) can't be handled here.
*/
static Widget edit_fkey_panel = None;
static Widget fkey_menu_button = None;
static Widget fkey_value_menu_button = None;
static Widget fkey_value_entry = None;
static char fkey_value[100] = "";
static char cur_fkey[20] = "";
static char *cur_fkey_value_mode = "";
static void FKeyValueMenuSelected(Widget w, char *key)
{
char *key1, *cp;
if (key[0] == 'c') {
cur_fkey_value_mode = "command";
key1 = "*command";
} else {
cur_fkey_value_mode = "string";
key1 = "*string";
}
XtVaGetValues(XtNameToWidget(fkey_value_menu_button, key1), XtNlabel, &cp, NULL);
XtVaSetValues(fkey_value_menu_button, XtNlabel, cp, NULL);
}
static void FKeyMenuSelected(Widget w, char *key)
{
struct fkey_struct *sp, *sp2;
int len;
const char *value, *prefix;
char key2[20];
if (key == NULL)
strcpy(key2, "");
else if (strncmp(key, "Shift-", strlen("Shift-")) == 0)
sprintf(key2, "s:%s", &key[strlen("Shift-")]);
else
strcpy(key2, key);
if (strcmp(cur_fkey, key2) != 0) {
if (strlen(cur_fkey) != 0) {
len = strlen(cur_fkey);
sp2 = NULL;
for (sp = fkey_list; sp != NULL; sp = sp->next) {
if (strncmp(sp->value, cur_fkey, len) == 0 && isspace(sp->value[len]))
break;
sp2 = sp;
}
if (strlen(fkey_value) != 0) { /* assign new string for the function key */
if (sp == NULL) { /* it was not defined before now */
sp = malloc(sizeof(struct fkey_struct));
if (fkey_list == NULL) fkey_list = sp;
else sp2->next = sp;
sp->next = NULL;
sp->value = NULL;
}
sp->value = realloc(sp->value, len + strlen(fkey_value) + 5);
prefix = "";
if (cur_fkey_value_mode[0] == 'c') prefix = "!";
else if (fkey_value[0] == '!') prefix = "\\";
sprintf(sp->value, "%s %s%s", cur_fkey, prefix, fkey_value);
} else { /* empty string - remove the entry for the function key */
if (sp != NULL) {
if (sp2 != NULL) sp2->next = sp->next;
else fkey_list = sp->next;
free(sp->value);
free(sp);
}
}
}
if (key != NULL) {
XtVaSetValues(fkey_menu_button, XtNlabel, key, NULL);
value = FindFunctionKeyValue(key2, FALSE);
if (value == NULL) value = "";
FKeyValueMenuSelected(None, (value[0] == '!') ? "command" : "string");
if (value[0] == '!' || value[0] == '\\') value = value + 1;
strncpy(fkey_value, value, sizeof(fkey_value) - 1);
XtVaSetValues(fkey_value_entry, XtNstring, fkey_value, NULL);
strcpy(cur_fkey, key2);
}
}
}
static void CloseFunctionKeyPanel(void)
{
XtPopdown(edit_fkey_panel);
}
static void SaveFunctionKey(void)
{
struct fkey_struct *sp;
FILE *fp;
if (appres.debug) fprintf(stderr, "SaveFunctionKey\n");
if (edit_fkey_panel != None) FKeyMenuSelected(None, NULL);
fp = fopen(fkey_filename, "w");
if (fp == NULL) {
fprintf(stderr, "%s: can't open \"%s\": %s\n",
PROGRAM_NAME, fkey_filename, strerror(errno));
return;
}
fprintf(fp, "#quick_modifiers %d\n", appres.quick_modifiers);
fprintf(fp, "#shift_lock %d\n", appres.shift_lock);
fprintf(fp, "#altgr_lock %d\n", appres.altgr_lock);
fprintf(fp, "#modifiers_lock %d\n", appres.modifiers_lock);
fprintf(fp, "#key_click %d\n", appres.key_click_duration);
fprintf(fp, "#autoclick %d\n", appres.autoclick_delay);
fprintf(fp, "#always_on_top %d\n", appres.always_on_top);
fprintf(fp, "#wm_toolbar %d\n", appres.wm_toolbar);
fprintf(fp, "#jump_pointer %d\n", appres.jump_pointer);
fprintf(fp, "#dict_file %s\n", dict_filename);
for (sp = fkey_list; sp != NULL; sp = sp->next) {
fprintf(fp, "%s\n", sp->value);
}
fclose(fp);
if (edit_fkey_panel != None) CloseFunctionKeyPanel();
}
static void PopupFunctionKeyEditor(void)
{
Widget form, form2, menu, menu_entry, button;
char label[20];
char *key;
int i, j;
if (edit_fkey_panel == None) {
edit_fkey_panel = XtVaCreatePopupShell("edit_fkey_panel", transientShellWidgetClass,
toplevel, NULL);
form = XtVaCreateManagedWidget("form", formWidgetClass, edit_fkey_panel, NULL);
form2 = XtVaCreateManagedWidget("form2", formWidgetClass, form, NULL);
XtVaCreateManagedWidget("fkey_label", labelWidgetClass, form2, NULL);
fkey_menu_button = XtVaCreateManagedWidget("fkey_menu", menuButtonWidgetClass,
form2, NULL);
menu = XtVaCreatePopupShell("menu", simpleMenuWidgetClass, fkey_menu_button, NULL);
for (j = 0; j <= 1; j++) {
for (i = 1; i <= appres.editable_function_keys; i++) {
if (j == 0)
sprintf(label, "F%d", i);
else
sprintf(label, "Shift-F%d", i);
key = XtNewString(label);
menu_entry = XtVaCreateManagedWidget(key, smeBSBObjectClass, menu, NULL);
XtAddCallback(menu_entry, XtNcallback, (XtCallbackProc)FKeyMenuSelected,
(XtPointer)key);
}
}
fkey_value_menu_button = XtVaCreateManagedWidget("fkey_value_menu", menuButtonWidgetClass,
form2, NULL);
menu = XtVaCreatePopupShell("menu", simpleMenuWidgetClass, fkey_value_menu_button, NULL);
menu_entry = XtVaCreateManagedWidget("string", smeBSBObjectClass, menu, NULL);
XtAddCallback(menu_entry, XtNcallback, (XtCallbackProc)FKeyValueMenuSelected,
(XtPointer)"string");
menu_entry = XtVaCreateManagedWidget("command", smeBSBObjectClass, menu, NULL);
XtAddCallback(menu_entry, XtNcallback, (XtCallbackProc)FKeyValueMenuSelected,
(XtPointer)"command");
XtVaCreateManagedWidget("fkey_value_sep", labelWidgetClass, form2, NULL);
fkey_value_entry = XtVaCreateManagedWidget("fkey_value", asciiTextWidgetClass, form2,
XtNuseStringInPlace, True,
XtNeditType, XawtextEdit,
XtNstring, fkey_value,
XtNlength, sizeof(fkey_value) - 1,
NULL);
button = XtVaCreateManagedWidget("save_button", commandWidgetClass, form, NULL);
XtAddCallback(button, XtNcallback, (XtCallbackProc)SaveFunctionKey, NULL);
button = XtVaCreateManagedWidget("close_button", commandWidgetClass, form, NULL);
XtAddCallback(button, XtNcallback, (XtCallbackProc)CloseFunctionKeyPanel, NULL);
XtRealizeWidget(edit_fkey_panel);
XSetWMProtocols(dpy, XtWindow(edit_fkey_panel), &wm_delete_window, 1);
XtSetKeyboardFocus(edit_fkey_panel, fkey_value_entry);
FKeyMenuSelected(None, "F1");
}
XtPopup(edit_fkey_panel, XtGrabNone);
}
/*
* If text is assigned to the specified function key,
* return the text. Otherwise, return NULL.
*/
static const char *FindFunctionKeyValue(const char *key, Boolean shiftable)
{
char label[50];
char prefix;
struct fkey_struct *sp;
int len;
prefix = '\0';
if (shiftable) {
if (shift_state & meta_mask) prefix = 'm';
else if (shift_state & alt_mask) prefix = 'a';
else if (shift_state & ControlMask) prefix = 'c';
else if (shift_state & ShiftMask) prefix = 's';
}
if (prefix == '\0') sprintf(label, "%s", key);
else sprintf(label, "%c:%s", prefix, key);
len = strlen(label);
for (sp = fkey_list; sp != NULL; sp = sp->next) {
if (strncmp(sp->value, label, len) == 0 && isspace(sp->value[len]))
return &(sp->value[len + 1]);
}
return NULL;
}
/*
* Key click
*/
void KeyClick(void)
{
XKeyboardState ks;
XKeyboardControl kc;
if (0 < appres.key_click_duration) {
XGetKeyboardControl(dpy, &ks);
kc.bell_duration = ks.bell_duration;
kc.bell_pitch = appres.key_click_pitch;
kc.bell_duration = appres.key_click_duration;
XChangeKeyboardControl(dpy, KBBellPitch | KBBellDuration, &kc);
XBell(dpy, 0);
XSync(dpy, FALSE);
kc.bell_pitch = ks.bell_pitch;
kc.bell_duration = ks.bell_duration;
XChangeKeyboardControl(dpy, KBBellPitch | KBBellDuration, &kc);
XSync(dpy, FALSE);
}
}
/*
* Display balloon message for the function keys,
* if text is assigned to the key.
*/
static Boolean balloon_panel_open = FALSE;
static Widget balloon_panel = None;
static XtIntervalId autoclick_id = (XtIntervalId)0;
static void StopAutoclick(void)
{
if (autoclick_id != (XtIntervalId)0) {
if (appres.debug) fprintf(stderr, "StopAutoclick: %lx\n", (long)autoclick_id);
XtRemoveTimeOut(autoclick_id);
autoclick_id = (XtIntervalId)0;
}
}
static void Autoclick(void)
{
StopAutoclick();
XTestFakeButtonEvent(target_dpy, 1, True, CurrentTime);
XTestFakeButtonEvent(target_dpy, 1, False, CurrentTime);
}
static void ShowBalloon(Widget w, XEvent *event, String *pars, Cardinal *n_pars)
{
static Widget message;
Position x, y;
Dimension ht;
const char *value;
if (strcmp(XtName(w), "MainMenu") == 0) {
value = "Main menu";
} else {
if (0 < appres.autoclick_delay) {
autoclick_id = XtAppAddTimeOut(app_con, (long)appres.autoclick_delay,
(XtTimerCallbackProc)Autoclick, (XtPointer)w);
if (appres.debug) fprintf(stderr, "ShowBalloon: auto click triggerd: %lx, %d\n",
(long)autoclick_id, appres.autoclick_delay);
}
value = FindFunctionKeyValue(XtName(w), TRUE);
if (value == NULL) return;
}
if (balloon_panel == None) {
balloon_panel = XtVaCreatePopupShell("balloon_panel", transientShellWidgetClass, toplevel,
XtNoverrideRedirect, TRUE, NULL);
message = XtVaCreateManagedWidget("message", labelWidgetClass, balloon_panel, NULL);
}
XtVaGetValues(w, XtNheight, &ht, NULL);
XtTranslateCoords(w, 6, ht + 2, &x, &y);
XtVaSetValues(balloon_panel, XtNx, x, XtNy, y, NULL);
if (value[0] == '!') {
if (appres.secure) return;
XtVaSetValues(message, XtNlabel, value + 1,
XtNbackground, appres.launch_balloon_background, NULL);
} else {
if (value[0] == '\\') value = value + 1;
XtVaSetValues(message, XtNlabel, value,
XtNbackground, appres.balloon_background, NULL);
}
XtPopup(balloon_panel, XtGrabNone);
balloon_panel_open = TRUE;
}
static void CloseBalloon(Widget w, XEvent *event, String *pars, Cardinal *n_pars)
{
StopAutoclick();
if (balloon_panel_open) {
XtPopdown(balloon_panel);
balloon_panel_open = FALSE;
}
}
/*
* Set icon image.
*/
static void SetIconBitmap(Widget w)
{
#include "xvkbd_icon.xbm"
#include "xvkbd_iconmask.xbm"
Pixmap icon_pixmap, icon_mask;
icon_pixmap = XCreateBitmapFromData(XtDisplay(w), XtWindow(w),
(char *)xvkbd_icon_bits,
xvkbd_icon_width, xvkbd_icon_height);;
icon_mask = XCreateBitmapFromData(XtDisplay(w), XtWindow(w),
(char *)xvkbd_iconmask_bits,
xvkbd_iconmask_width, xvkbd_iconmask_height);
XtVaSetValues(w, XtNiconPixmap, icon_pixmap, XtNiconMask, icon_mask, NULL);
}
/*
* Callback for VisibilityChanged event, which will be invoked
* when xvkbd window is hidden by other window. ** EXPERIMENTAL **
*/
static void VisibilityChanged(Widget w, XEvent *event,
String *pars, Cardinal *n_pars)
{
static cnt = 0;
static time_t t1 = 0;
time_t t2;
if (!appres.always_on_top) return;
if (balloon_panel_open) return;
if (main_menu != None && XtWindow(main_menu) != None) {
XWindowAttributes attr;
XGetWindowAttributes(dpy, XtWindow(main_menu), &attr);
if (attr.map_state != IsUnmapped) return;
}
t2 = time(NULL);
if (t1 != t2) cnt = 0;
t1 = t2;
cnt = cnt + 1;
if (appres.debug)
fprintf(stderr, "%s: visibility of the window changed (cnt = %d)\n", PROGRAM_NAME, cnt);
if (cnt < 5)
XRaiseWindow(XtDisplay(toplevel), XtWindow(toplevel));
}
/*
* The main program.
*/
int main(int argc, char *argv[])
{
static XtActionsRec actions[] = {
{ "DeleteWindowProc", DeleteWindowProc },
{ "WindowResized", WindowResized },
{ "VisibilityChanged", VisibilityChanged },
{ "ReadKeymap", (XtActionProc)ReadKeymap },
{ "ButtonDownAction", ButtonDownAction },
{ "ButtonUpAction", ButtonUpAction },
{ "ShowBalloon", ShowBalloon },
{ "CloseBalloon", CloseBalloon },
{ "ClosePopupPanel", (XtActionProc)ClosePopupPanel },
};
static String fallback_resources[] = {
#include "XVkbd-common.h"
NULL,
};
Boolean open_keypad_panel = FALSE;
char ch;
Window child;
int op, ev, err;
argc1 = argc;
argv1 = malloc(sizeof(char *) * (argc1 + 5));
memcpy(argv1, argv, sizeof(char *) * argc1);
argv1[argc1] = NULL;
#ifdef USE_I18N
XtSetLanguageProc(NULL, NULL, NULL);
#endif
toplevel = XtVaAppInitialize(NULL, "XVkbd",
options, XtNumber(options),
&argc, argv, fallback_resources, NULL);
dpy = XtDisplay(toplevel);
app_con = XtWidgetToApplicationContext(toplevel);
XtAppAddActions(app_con, actions, XtNumber(actions));
target_dpy = dpy;
if (1 < argc) {
fprintf(stderr, "%s: illegal option: %s\n\n", PROGRAM_NAME, argv[1]);
}
XtGetApplicationResources(toplevel, &appres,
application_resources, XtNumber(application_resources),
NULL, 0);
if (appres.version) {
fprintf(stdout, "%s\n", appres.description);
exit(1);
}
if (appres.compact) {
appres.keypad = FALSE;
appres.function_key = FALSE;
}
if (appres.keypad_only && !appres.keypad) {
appres.keypad_only = FALSE;
open_keypad_panel = TRUE;
}
if (appres.no_sync) {
XSync(dpy, FALSE);
XSetErrorHandler(MyErrorHandler);
}
if (0 < strlen(appres.window) || 0 < strlen(appres.instance)) {
if (strcmp(appres.window, "root") == 0) {
focused_window = RootWindow(dpy, DefaultScreen(dpy));
} else if (sscanf(appres.window, "0x%lX%c", &focused_window, &ch) != 1) {
if (sscanf(appres.window, "%ld%c", &focused_window, &ch) != 1) {
focused_window = FindWindow(RootWindow(dpy, DefaultScreen(dpy)),
appres.window);
if (focused_window == None) {
fprintf(stderr, "%s: no such window: window=%s and class=%s\n", PROGRAM_NAME, appres.window, appres.instance);
if (appres.no_root)
exit(-1);
}
}
}
}
focused_subwindow = focused_window;
ReadKeymap();
if (!altgr_mask && appres.auto_add_keysym) AddModifier(XK_Mode_switch);
if (strlen(appres.text) != 0 || strlen(appres.file) != 0) {
appres.keypad_keysym = TRUE;
if (focused_window != None &&
(appres.list_widgets || strlen(appres.widget) != 0)) {
XtVaSetValues(toplevel, XtNwidth, 1, XtNheight, 1, NULL);
XtRealizeWidget(toplevel);
child = FindWidget(toplevel, focused_window, appres.widget);
if (child != None) focused_subwindow = child;
}
if (strlen(appres.text) != 0)
SendString(appres.text);
else
SendFileContent(appres.file);
exit(0);
} else {
ReadFuncionKeys();
if (0 < strlen(appres.keys_normal))
RedefineKeys(keys_normal, appres.keys_normal);
if (0 < strlen(appres.keys_shift))
RedefineKeys(keys_shift, appres.keys_shift);
if (0 < strlen(appres.keys_altgr))
RedefineKeys(keys_altgr, appres.keys_altgr);
if (0 < strlen(appres.keys_shift_altgr))
RedefineKeys(keys_shift_altgr, appres.keys_shift_altgr);
if (0 < strlen(appres.key_labels))
RedefineKeys(key_labels, appres.key_labels);
if (0 < strlen(appres.normal_key_labels))
RedefineKeys(normal_key_labels, appres.normal_key_labels);
if (0 < strlen(appres.shift_key_labels))
RedefineKeys(shift_key_labels, appres.shift_key_labels);
if (0 < strlen(appres.altgr_key_labels))
RedefineKeys(altgr_key_labels, appres.altgr_key_labels);
if (0 < strlen(appres.shift_altgr_key_labels))
RedefineKeys(shift_altgr_key_labels, appres.shift_altgr_key_labels);
if (0 < strlen(appres.keypad_normal)) {
RedefineKeys(keypad, appres.keypad_normal);
RedefineKeys(keypad_shift, appres.keypad_normal);
RedefineKeys(keypad_label, appres.keypad_normal);
}
if (0 < strlen(appres.keypad_shift))
RedefineKeys(keypad_shift, appres.keypad_shift);
if (0 < strlen(appres.keypad_labels))
RedefineKeys(keypad_label, appres.keypad_labels);
MakeKeyboard(FALSE);
if (focused_window != None &&
(appres.list_widgets || strlen(appres.widget) != 0)) {
child = FindWidget(toplevel, focused_window, appres.widget);
if (child != None) focused_subwindow = child;
}
if (main_menu != None) {
if (strlen(dict_filename) == 0)
XtSetSensitive(XtNameToWidget(main_menu, "*completion"), FALSE);
if (strlen(appres.customizations) == 0)
XtSetSensitive(XtNameToWidget(main_menu, "*select_layout"), FALSE);
if (appres.nonexitable)
XtSetSensitive(XtNameToWidget(main_menu, "*quit"), FALSE);
if (appres.secure) {
XtSetSensitive(XtNameToWidget(main_menu, "*man"), FALSE);
XtSetSensitive(XtNameToWidget(main_menu, "*open_display"), FALSE);
}
}
#ifdef USE_XTEST
if (!XQueryExtension(dpy, "XTEST", &op, &ev, &err)) {
if (appres.xtest) {
fprintf(stderr, "%s: XTEST extension is not supported by the X server\n",
PROGRAM_NAME);
fprintf(stderr, "%s: XSendEvent will be used instead\n",
PROGRAM_NAME);
appres.xtest = FALSE;
}
if (main_menu != None) {
XtSetSensitive(XtNameToWidget(main_menu, "*use_xtest"), FALSE);
RefreshMainMenu();
}
}
#endif
if (!appres.debug) {
#ifdef SYSV
signal(SIGINT, SIG_IGN);
signal(SIGQUIT, SIG_IGN);
#else
struct sigaction sigact;
sigact.sa_handler = SIG_IGN;
sigemptyset(&sigact.sa_mask);
sigact.sa_flags = 0;
sigaction(SIGINT, &sigact, NULL);
sigaction(SIGQUIT, &sigact, NULL);
#endif
}
{
#ifdef SYSV
signal(SIGUSR1, SignalUser1);
#else
struct sigaction sigact;
sigact.sa_handler = SignalUser1;
sigemptyset(&sigact.sa_mask);
sigact.sa_flags = 0;
sigaction(SIGUSR1, &sigact, NULL);
#endif
}
SetIconBitmap(toplevel);
if (open_keypad_panel) MenuSelected(None, "keypad");
XtAppMainLoop(app_con);
}
exit(0);
}
/*
* Replace setlocale() in the standard library here, because
* it may not support some locales used for localized keyboards.
*/
#if defined(USE_I18N) && !defined(HAVE_SETLOCALE)
char *setlocale(int category, const char *locale)
{
static char old_locale[100] = "C";
static char cur_locale[100] = "C";
const char *s;
if (locale == NULL) {
return cur_locale;
} else if (category == LC_ALL) {
strcpy(old_locale, cur_locale);
if (locale[0] == '\0') {
s = getenv("LC_ALL");
if (s == NULL) s = "C"; /* LC_ALL not defined */
} else {
s = locale;
}
strncpy(cur_locale, s, sizeof(cur_locale) - 1);
return old_locale;
} else {
return cur_locale;
}
}
#endif /* HAVE_SETLOCALE */