From 9f48da37571cf921c858b01b9d76e7d1798625fe Mon Sep 17 00:00:00 2001 From: Cecilio Pardo Date: Mon, 2 Dec 2024 17:30:42 +0100 Subject: [PATCH] Send events for key presses and key releases. Detect double/triple taps. * doc/lispref/commands.texi: Added information about new event. * src/gtkutil.c (xg_create_frame_widgets): Modified to handle key release events. (xg_maybe_send_low_level_key_event): New function that sends key events. (xg_widget_key_press_event_cb): Modified to send low level key events. (xg_widget_key_release_event_cb): New function to send low level key events on key release. * src/keyboard.c (kbd_buffer_get_event): Modified to handle low level keyboard events. (make_lispy_event): Modified to handle low level keyboard events. (kbd_low_level_key_is_enabled): New function to decide if a particular key should generate events, looking at configuration. (syms_of_keyboard): Added symbols and varialbles. (keys_of_keyboard): Modified to map low level key events to 'special-event-map'. * src/keyboard.h: Modified with function prototype. * src/pgtkterm.c (pgtk_maybe_send_low_level_key_event): New function that sends key events. (key_press_event): Modified to handle low level keyboard events. (key_release_event): Modified to handle low level keyboard events. * src/termhooks.h: Modified to define constant. * src/w32fns.c (w32_wnd_proc): Modified to generate low level key events. * src/w32term.c (w32_read_socket): Modified to generate low level key events. * src/w32term.h: Modified to define constants. * src/xterm.c (x_get_modifier_for_keycode): New function to decide which (if any) modifier corresponds to a given key. (x_maybe_send_low_level_key_event): New function that sends key events. (handle_one_xevent): Modified to handle low level key events. (syms_of_xterm): Modified to define symbol. * src/xterm.h: Modified with function prototype * lisp/low-level-key.el (New file). (low-level-key): Struct with event data. (llk-bindings): User bindings for low level key events. (llk-tap-timeout): User option. (llk-keysyms): List of available keysyms. (llk--define-xk): Macro for defining keysyms. (llk--define-keysyms): Build llk-keysyms. (llk-bind): Function to create a binding. (llk--event-history-for-tap): Event history for tap detection. (llk--detect-n-tap): Function to detect taps. (describe-low-level-key): Command to get information about a key. (llk--describe): Show help buffer with information about an event. (llk-handle): Handler for key events. (llk--generate-event): Generate input event for key. (llk--generate-events): Generate input events for key. --- doc/lispref/commands.texi | 25 +++ lisp/low-level-key.el | 373 ++++++++++++++++++++++++++++++++++++++ src/gtkutil.c | 81 +++++++++ src/keyboard.c | 60 ++++++ src/keyboard.h | 1 + src/pgtkterm.c | 55 ++++++ src/termhooks.h | 1 + src/w32fns.c | 11 ++ src/w32term.c | 49 +++++ src/w32term.h | 3 +- src/xterm.c | 142 +++++++++++++++ src/xterm.h | 2 + 12 files changed, 802 insertions(+), 1 deletion(-) create mode 100644 lisp/low-level-key.el diff --git a/doc/lispref/commands.texi b/doc/lispref/commands.texi index 7cc32a7fdb3..23c3312dbcb 100644 --- a/doc/lispref/commands.texi +++ b/doc/lispref/commands.texi @@ -2658,6 +2658,31 @@ Misc Events respectively. They are usually interpreted as being relative to the size of the object beneath the gesture: image, window, etc. +@cindex @code{low-level-key} event +@item (low-level-key @var{is-key-press} @var{key} @var{modifier} @var{time} @var{frame}) +This event is sent on the physical press or release of keys, only on +systems where it is supported, currently X, MS-Windows and PGTK, and +only if the variable @code{enable-low-level-key-events} has a +non-@code{nil} value. See its documentation for the values to use, that +can activate the events for all or some keys. + +@var{is-key-press} is @code{t} for a key press, @code{nil} for a key +release. @var{time} is the event's time in milliseconds, as reported by +the underlying platform, and should only be used to measure time +intervals between events of this same kind. @var{frame} is the frame +receiving the event. @var{modifier} is @code{nil} if the key is not a +modifier key, @code{t} if it is, but it is unknown which one, or one of +@code{shift}, @code{control}, @code{meta}, @code{alt}, @code{super}, +@code{hyper}. + +@var{key}, an integer, identifies the key pressed or released. This +number is platform dependent. + +This is a special event (@pxref{Special Events}), ignored by +default. Loading @file{low-level-key.el} sets a handler +that can generate normal input events for key press, release and multi +tap. See the function @code{llk-bind}. + @cindex @code{preedit-text} event @item (preedit-text @var{arg}) This event is sent when a system input method tells Emacs to display diff --git a/lisp/low-level-key.el b/lisp/low-level-key.el new file mode 100644 index 00000000000..8a8d912977d --- /dev/null +++ b/lisp/low-level-key.el @@ -0,0 +1,373 @@ +;;; low-level-key.el --- Handling of key press/release events -*- lexical-binding: t -*- + +;; Copyright (C) 2025 Free Software Foundation, Inc. + +;; This file is part of GNU Emacs. + +;; GNU Emacs 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 3 of the License, or +;; (at your option) any later version. + +;; GNU Emacs 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. + +;; You should have received a copy of the GNU General Public License +;; along with GNU Emacs. If not, see . + +;;; Commentary: + +;; Emacs can send low-level key events, that correspond to key presses +;; and releases. These events are by default disabled. + +;; When enabled by setting `enable-low-level-key-events' to a non-nil +;; value, emacs will begin sending them, and they will be ignored as +;; low-level-key is bound to `ignore' on `special-event-map'. + +;; This file sets a handler for them (`llk-handle') which generates +;; input events for key presses, key releases and double and triple +;; taps. These events can be bound to commands on normal keymaps. + +;; Because generating these events for all keys would interfere with +;; normal keyboard input, they must be activated individually by calling +;; the function `llk-bind'. + +;; The low-level-key event payload is described by the 'low-level-key' +;; struct. Use 'cl-describe-type' to get more information about it. +;; +;; After loading this file and setting a non-nil value for +;; 'enable-low-level-key-events', events begin to be handled by +;; 'llk-handle', which tries to detect n-taps, and creates input +;; events. See 'llk-bind'. + +;; Code: + +(require 'cl-lib) + +(defvar llk-tap-timeout (let ((time (mouse-double-click-time))) + (if (> time 0) time 800)) + "Time in milliseconds between key presses/releases to consider a double tap. +For triple taps, the time is twice this value.") + +(cl-defstruct (low-level-key (:type list)) + "Structure for low level key events. +Received as low-level-key on `special-event-map'." + (event-type nil :type symbol + :documentation "Type of event: low-level-key") + (is-key-press nil :type boolean + :documentation "t if the key has been pressed, nil if it has been released.") + (key nil :type integer + :documentation "The keysym number.") + (modifier nil :type symbol + :documentation "Modifier associated with this key. It is nil if the key is +not a modifier. It can be one of the following symbols: shift, control, meta, +super, hyper, alt. It can also be t if the key is a modifier but it can't be +identified, as in the PGTK backend.") + (time nil :type integer + :documentation "Timestamp in milliseconds of the event.") + (frame nil :type frame + :documentation "Frame where the event happened.")) + +(defvar llk-bindings nil + "List of bindings for low level key events (press/release/tap). + +Use the `llk-bind' function to add bindings. See its documentation for +a description of the binding information.") + +(defvar llk-keysyms nil + "List of keysyms and their names. +Each element has the form (CODE . KEYSYM), where code is a NUMBER and +KEYSYM a symbol, such as `xk-shift-l'") + +;; TODO docstring. +(defun llk-bind (key &rest events) + "Activates low level event generation for a key. + +KEY can be a number or a symbol. The symbols `shift', `control', +`meta', `super', `hyper', `alt' activate events for the corresponding +modifier keys. A number activates events for the corresponding KeySym. + +EVENTS are symbols that activate one event. Possible values are `press', +`release', `double' and `triple'. + +See `llk-keysyms' for a list of known values for KEY and their names. +For each of those, there is a corresponding variable. It is better to +use the variables to specify keys, as numerical values are +platform-dependent. The names are parallel to those for KeySyms on X, +as defined in `xkeysymdef.h'. For example, `XK_Shift_L' (the left shift +key), corresponds to `xk-shift-l'. + +The `xkeysymdef.h' file defines different KeySyms for capital and small +versions of latin letters. For this event, only the capital version is +used, with the variables `xk-a', `xk-b', etc. + +Low level key events must be enabled with the variable +`enable-low-level-key-events'. + +Once a key is activated with this function, input events will be +generated for them, and can be bound to commands using normal keymaps. + +For example, activating the double tap for the left shift key: + + (llk-bind xk-shift-l \\='double) + +will generate the event `double-xk-shift-l', than can be bound to a +command with: + + (keymap-global-set [double-xk-shift-l] COMMAND) + +Prefixes for events are `press-key-', `release-key-', `double-' and +`triple-'. + +If you use a KeySym number that is not on `llk-keysyms', the events will +use its numerical value." + (setq llk-bindings + (cl-delete-if (lambda (x) (eq (car x) key)) llk-bindings)) + (push (append (list key) events) llk-bindings)) + +;; We store the last events (key/modifier is-press timestamp) here to +;; test for multitap. This is not the use the low-level-key struct. +(defvar llk--event-history-for-tap nil + "Internal variable for detecting taps.") + +(defun llk--detect-n-tap (n timeout) + "Internal function to detect n-tap keys." + ;; Only care about last 2xN events + (ntake (* 2 n) llk--event-history-for-tap) + ;; If we have: + ;; - Exactly 2 * n events. + ;; - down, up, down, up, ... + ;; - not two much time between first and last + (and (eq (* 2 n) (length llk--event-history-for-tap)) + (cl-every #'eq + (ntake (* 2 n) + (list nil t nil t nil t nil t + nil t nil t nil t nil t)) + (mapcar 'cl-second llk--event-history-for-tap)) + (< (- (cl-third (cl-first llk--event-history-for-tap)) + (cl-third (car (last llk--event-history-for-tap)))) + timeout) + (progn (setq llk--event-history-for-tap nil) t))) + +(defun llk--generate-event (key event-type) + (when (numberp key) + (let ((sym (cdr (assoc key llk-keysyms)))) + (when sym + (setq key sym)))) + (push (intern (format "%s-%s" event-type key)) + unread-command-events)) + +(defun llk--generate-events (key is-press binding timestamp) + (if is-press + (when (member 'press binding) + (llk--generate-event key 'press-key)) + (when (member 'release binding) + (llk--generate-event key 'release-key))) + + (let ((double (member 'double binding)) + (triple (member 'triple binding))) + ;; a non-tap key clears the event history. + (if (or double triple) + (progn + ;; Clear the event history if it has events from another key. + (unless (equal (car (car llk--event-history-for-tap)) key) + (setq llk--event-history-for-tap nil)) + (push (list key is-press timestamp) llk--event-history-for-tap) + (and double + (llk--detect-n-tap 2 llk-tap-timeout) + (llk--generate-event key 'double-key)) + (and triple + (llk--detect-n-tap 3 (* 2 llk-tap-timeout)) + (llk--generate-event key 'triple-key))) + ;; A non-tap key clears the event history. + (setq llk--event-history-for-tap nil)))) + +(defun llk-handle () + (interactive) + (let* ((key (low-level-key-key last-input-event)) + (modifier (low-level-key-modifier last-input-event)) + (timestamp (low-level-key-time last-input-event)) + (is-press (low-level-key-is-key-press last-input-event)) + (binding (assoc key llk-bindings)) + (binding-modifier (assoc modifier llk-bindings))) + (if binding + (llk--generate-events key is-press binding timestamp) + (when binding-modifier + (llk--generate-events modifier is-press binding-modifier timestamp))))) + +(defmacro llk--define-xk (name x-keysym w32-keysym) + "Internal macro to define keysyms." + `(let ((ksym (pcase (window-system) + ((or 'pgtk 'x) ,x-keysym) + ('w32 ,w32-keysym)))) + (defconst ,name ksym "Constant for a keysym value.") + (push (cons ksym ',name) llk-keysyms))) + +(defun llk--define-keysyms () + "Initialize the keysym list, `llk-keysyms'." + (setq llk-keysyms nil) + + ;; tty keys + (llk--define-xk xk-backspace #xff08 #x08) ;; XK_BackSpace VK_BACK + (llk--define-xk xk-tab #xff09 #x09) ;; XK_Tab VK_TAB + (llk--define-xk xk-clear #xff0b #x0C) ;; XK_Clear VK_CLEAR + (llk--define-xk xk-return #xff0d #x0D) ;; XK_Return VK_RETURN + (llk--define-xk xk-pause #xff13 #x13) ;; XK_Pause VK_PAUSE + (llk--define-xk xk-scroll-lock #xff14 #x91) ;; XK_Scroll_Lock VK_SCROLL + (llk--define-xk xk-escape #xff1B #x1B) ;; XK_Escape VK_ESCAPE + (llk--define-xk xk-delete #xffff #x2E) ;; XK_Delete VK_DELETE + + ;; Cursor control and motion + (llk--define-xk xk-home #xff50 #x24) ;; XK_Home VK_HOME + (llk--define-xk xk-left #xff51 #x25) ;; XK_Left VK_LEFT + (llk--define-xk xk-up #xff52 #x26) ;; XK_Up VK_UP + (llk--define-xk xk-right #xff53 #x27) ;; XK_Right VK_RIGHT + (llk--define-xk xk-down #xff54 #x28) ;; XK_Down VK_DOWN + (llk--define-xk xk-page-up #xff55 #x21) ;; XK_Page_Up VK_PRIOR + (llk--define-xk xk-page-down #xff56 #x22) ;; XK_Page_Down VK_NEXT + (llk--define-xk xk-end #xff57 #x23) ;; XK_End VK_END + (llk--define-xk xk-begin #xff58 #x24) ;; XK_Begin VK_HOME + + ;; Special Windows keyboard keys + (llk--define-xk xk-win-l #xFF5B #x5B) ;; XK_Win_L VK_LWIN + (llk--define-xk xk-win-r #xFF5C #x5C) ;; XK_Win_R VK_RWIN + (llk--define-xk xk-app #xFF5D #x5D) ;; XK_App VK_APPS + + ;; Misc functions + (llk--define-xk xk-select #xff60 #x29) ;; XK_Select VK_SELECT + (llk--define-xk xk-print #xff61 #x2A) ;; XK_Print VK_PRINT + (llk--define-xk xk-insert #xff64 #x2D) ;; XK_Insert VK_INSERT + (llk--define-xk xk-num-lock #xff7f #x90) ;; XK_Num_Lock VK_NUMLOCK + + ;; Keypad + ;; TODO: Check values for MS-Windows + (llk--define-xk xk-kp-enter #xff8d nil) ;; XK_KP_Enter ??? + (llk--define-xk xk-kp-multiply #xffaa nil) ;; XK_KP_Multiply ??? + (llk--define-xk xk-kp-add #xffab nil) ;; XK_KP_Add ??? + (llk--define-xk xk-kp-subtract #xffad nil) ;; XK_KP_Subtract ??? + (llk--define-xk xk-kp-decimal #xffae nil) ;; XK_KP_Decimal ??? + (llk--define-xk xk-kp-divide #xffaf nil) ;; XK_KP_Divide ??? + (llk--define-xk xk-kp-0 #xffb0 #x60) ;; XK_KP_0 VK_NUMPAD0 + (llk--define-xk xk-kp-1 #xffb1 #x61) ;; XK_KP_1 VK_NUMPAD1 + (llk--define-xk xk-kp-2 #xffb2 #x62) ;; XK_KP_2 VK_NUMPAD2 + (llk--define-xk xk-kp-3 #xffb3 #x63) ;; XK_KP_3 VK_NUMPAD3 + (llk--define-xk xk-kp-4 #xffb4 #x64) ;; XK_KP_4 VK_NUMPAD4 + (llk--define-xk xk-kp-5 #xffb5 #x65) ;; XK_KP_5 VK_NUMPAD5 + (llk--define-xk xk-kp-6 #xffb6 #x66) ;; XK_KP_6 VK_NUMPAD6 + (llk--define-xk xk-kp-7 #xffb7 #x67) ;; XK_KP_7 VK_NUMPAD7 + (llk--define-xk xk-kp-8 #xffb8 #x68) ;; XK_KP_8 VK_NUMPAD8 + (llk--define-xk xk-kp-9 #xffb9 #x69) ;; XK_KP_9 VK_NUMPAD9 + + ;; Function keys + (llk--define-xk xk-f1 #xffbe #x70) ;; XK_F1 VK_F1 + (llk--define-xk xk-f2 #xffbf #x71) ;; XK_F2 VK_F2 + (llk--define-xk xk-f3 #xffc0 #x72) ;; XK_F3 VK_F3 + (llk--define-xk xk-f4 #xffc1 #x73) ;; XK_F4 VK_F4 + (llk--define-xk xk-f5 #xffc2 #x74) ;; XK_F5 VK_F5 + (llk--define-xk xk-f6 #xffc3 #x75) ;; XK_F6 VK_F6 + (llk--define-xk xk-f7 #xffc4 #x76) ;; XK_F7 VK_F7 + (llk--define-xk xk-f8 #xffc5 #x77) ;; XK_F8 VK_F8 + (llk--define-xk xk-f9 #xffc6 #x78) ;; XK_F9 VK_F9 + (llk--define-xk xk-f10 #xffc7 #x79) ;; XK_F10 VK_F10 + (llk--define-xk xk-f11 #xffc8 #x7A) ;; XK_F11 VK_F11 + (llk--define-xk xk-f12 #xffc9 #x7B) ;; XK_F12 VK_F12 + (llk--define-xk xk-f13 #xffca #x7C) ;; XK_F13 VK_F13 + (llk--define-xk xk-f14 #xffcb #x7D) ;; XK_F14 VK_F14 + (llk--define-xk xk-f15 #xffcc #x7E) ;; XK_F15 VK_F15 + (llk--define-xk xk-f16 #xffcd #x7F) ;; XK_F16 VK_F16 + (llk--define-xk xk-f17 #xffce #x80) ;; XK_F17 VK_F17 + (llk--define-xk xk-f18 #xffcf #x81) ;; XK_F18 VK_F18 + (llk--define-xk xk-f19 #xffd0 #x82) ;; XK_F19 VK_F19 + (llk--define-xk xk-f20 #xffd1 #x83) ;; XK_F20 VK_F20 + (llk--define-xk xk-f21 #xffd2 #x84) ;; XK_F21 VK_F21 + (llk--define-xk xk-f22 #xffd3 #x85) ;; XK_F22 VK_F22 + (llk--define-xk xk-f23 #xffd4 #x86) ;; XK_F23 VK_F23 + (llk--define-xk xk-f24 #xffd5 #x87) ;; XK_F24 VK_F24 + + ;; Modifier keys + (llk--define-xk xk-shift-l #xffe1 #xA0) ;; XK_Shift_L VK_LSHIFT + (llk--define-xk xk-shift-r #xffe2 #xA1) ;; XK_Shift_R VK_RSHIFT + (llk--define-xk xk-control-l #xffe3 #xA2) ;; XK_Control_L VK_LCONTROL + (llk--define-xk xk-control-r #xffe4 #xA3) ;; XK_Control_R VK_RCONTROL + (llk--define-xk xk-caps-lock #xffe5 #x14) ;; XK_Caps_Lock VK_CAPITAL + (llk--define-xk xk-meta-l #xffe7 nil) ;; XK_Meta_L + (llk--define-xk xk-meta-r #xffee nil) ;; XK_Meta_R + (llk--define-xk xk-alt-l #xffe9 #xA4) ;; XK_Alt_L VK_LMENU + (llk--define-xk xk-alt-r #xffea #xA5) ;; XK_Alt_R VK_RMENU + (llk--define-xk xk-super-l #xffeb nil) ;; XK_Super_L + (llk--define-xk xk-super-r #xffec nil) ;; XK_Super_R + (llk--define-xk xk-hyper-l #xffed nil) ;; XK_Hyper_L + (llk--define-xk xk-hyper-r #xffee nil) ;; XK_Hyper_R + + ;; Latin 1 + ;; For numbers and letters, MS-Windows does not define constant names. + ;; X11 defines distinct keysyms for lowercase and uppercase + ;; letters. We use only the uppercase ones. Events with lowercase + ;; letters are converted to uppercase. + (llk--define-xk xk-space #x0020 #x20) ;; XK_space VK_SPACE + (llk--define-xk xk-0 #x0030 #x30) ;; XK_0 + (llk--define-xk xk-1 #x0031 #x31) ;; XK_1 + (llk--define-xk xk-2 #x0032 #x32) ;; XK_2 + (llk--define-xk xk-3 #x0033 #x33) ;; XK_3 + (llk--define-xk xk-4 #x0034 #x34) ;; XK_4 + (llk--define-xk xk-5 #x0035 #x35) ;; XK_5 + (llk--define-xk xk-6 #x0036 #x36) ;; XK_6 + (llk--define-xk xk-7 #x0037 #x37) ;; XK_7 + (llk--define-xk xk-8 #x0038 #x38) ;; XK_8 + (llk--define-xk xk-9 #x0039 #x39) ;; XK_9 + (llk--define-xk xk-a #x0041 #x41) ;; XK_A + (llk--define-xk xk-b #x0042 #x42) ;; XK_B + (llk--define-xk xk-c #x0043 #x43) ;; XK_C + (llk--define-xk xk-d #x0044 #x44) ;; XK_D + (llk--define-xk xk-e #x0045 #x45) ;; XK_E + (llk--define-xk xk-f #x0046 #x46) ;; XK_F + (llk--define-xk xk-g #x0047 #x47) ;; XK_G + (llk--define-xk xk-h #x0048 #x48) ;; XK_H + (llk--define-xk xk-i #x0049 #x49) ;; XK_I + (llk--define-xk xk-j #x004A #x4A) ;; XK_J + (llk--define-xk xk-k #x004B #x4B) ;; XK_K + (llk--define-xk xk-l #x004C #x4C) ;; XK_L + (llk--define-xk xk-m #x004D #x4D) ;; XK_M + (llk--define-xk xk-n #x004E #x4E) ;; XK_N + (llk--define-xk xk-o #x004F #x4F) ;; XK_O + (llk--define-xk xk-p #x0050 #x50) ;; XK_P + (llk--define-xk xk-q #x0051 #x51) ;; XK_Q + (llk--define-xk xk-r #x0052 #x52) ;; XK_R + (llk--define-xk xk-s #x0053 #x53) ;; XK_S + (llk--define-xk xk-t #x0054 #x54) ;; XK_T + (llk--define-xk xk-u #x0055 #x55) ;; XK_U + (llk--define-xk xk-v #x0056 #x56) ;; XK_V + (llk--define-xk xk-w #x0057 #x57) ;; XK_W + (llk--define-xk xk-x #x0058 #x58) ;; XK_X + (llk--define-xk xk-y #x0059 #x59) ;; XK_Y + (llk--define-xk xk-z #x005A #x5A));; XK_Z + +(defun describe-low-level-key () + "Wait for key press and describe the low-level key event it generates." + (interactive) + (define-key special-event-map [low-level-key] 'llk--describe)) + +(defun llk--describe () + "Internal function for `special-event-map' to describe low level key events." + (interactive) + (when (low-level-key-is-key-press last-input-event) + (define-key special-event-map [low-level-key] 'llk-handle) + (with-help-window (help-buffer) + (insert "\n") + (let* ((xk (low-level-key-key last-input-event)) + (sym (assoc xk llk-keysyms))) + (insert (format "Keysym number: %d (#x%X),\n" xk xk)) + (if sym + (insert (format "which corresponds to named key %s.\n\n" (cdr sym))) + (insert "which does not correspond to any known named key.\n\n")) + (if (low-level-key-modifier last-input-event) + (insert (format "This key corresponds to the %s modifier.\n\n" + (low-level-key-modifier last-input-event))) + (insert "This key does not correspond to a modifier.\n\n")) + (insert "See the value of the `llk-keysyms' variable for a list of known keys.\n"))))) + +(llk--define-keysyms) +(define-key special-event-map [low-level-key] 'llk-handle) +(setq llk-bindings nil) diff --git a/src/gtkutil.c b/src/gtkutil.c index 0e9dd4dfe11..67c2c426b9e 100644 --- a/src/gtkutil.c +++ b/src/gtkutil.c @@ -98,6 +98,7 @@ G_DEFINE_TYPE (EmacsMenuBar, emacs_menu_bar, GTK_TYPE_MENU_BAR) static void xg_im_context_preedit_changed (GtkIMContext *, gpointer); static void xg_im_context_preedit_end (GtkIMContext *, gpointer); static bool xg_widget_key_press_event_cb (GtkWidget *, GdkEvent *, gpointer); +static bool xg_widget_key_release_event_cb (GtkWidget *, GdkEvent *, gpointer); #endif #if GTK_CHECK_VERSION (3, 10, 0) @@ -1749,6 +1750,12 @@ xg_create_frame_widgets (struct frame *f) g_signal_connect (G_OBJECT (wfixed), "key-press-event", G_CALLBACK (xg_widget_key_press_event_cb), NULL); + + g_signal_connect (G_OBJECT (wfixed), "key-release-event", + G_CALLBACK (xg_widget_key_release_event_cb), + NULL); + + #endif { @@ -6376,6 +6383,53 @@ xg_im_context_preedit_end (GtkIMContext *imc, gpointer user_data) kbd_buffer_store_event (&inev); } +#ifndef HAVE_XINPUT2 +static void +xg_maybe_send_low_level_key_event (struct frame *f, + GdkEvent *xev) +{ + GdkEventKey xkey = xev->key; + bool is_press; + Lisp_Object key, modifier; + union buffered_input_event inev; + + if (NILP (Venable_low_level_key_events)) + return; + + switch (xev->type) + { + case GDK_KEY_PRESS: + is_press = true; + break; + case GDK_KEY_RELEASE: + is_press = false; + break; + default: + return; + } + + modifier = x_get_modifier_for_keycode (FRAME_OUTPUT_DATA (f)->display_info, + xev->key.hardware_keycode); + + int keysym = xkey.keyval; + + if (keysym >= GDK_KEY_a && keysym <= GDK_KEY_z) + keysym -= GDK_KEY_a - GDK_KEY_A; + + if (!kbd_low_level_key_is_enabled (keysym, modifier)) + return; + + key = make_fixnum (keysym); + + EVENT_INIT (inev.ie); + XSETFRAME (inev.ie.frame_or_window, f); + inev.ie.kind = LOW_LEVEL_KEY_EVENT; + inev.ie.timestamp = xkey.time; + inev.ie.arg = list3 (is_press ? Qt : Qnil, key, modifier); + kbd_buffer_store_buffered_event (&inev, &xg_pending_quit_event); +} +#endif + static bool xg_widget_key_press_event_cb (GtkWidget *widget, GdkEvent *event, gpointer user_data) @@ -6404,6 +6458,10 @@ xg_widget_key_press_event_cb (GtkWidget *widget, GdkEvent *event, if (!f) return true; +#ifndef HAVE_XINPUT2 + xg_maybe_send_low_level_key_event (f, event); +#endif + if (popup_activated ()) return true; @@ -6557,6 +6615,29 @@ xg_widget_key_press_event_cb (GtkWidget *widget, GdkEvent *event, return true; } +static bool +xg_widget_key_release_event_cb (GtkWidget *widget, GdkEvent *event, + gpointer user_data) +{ +#ifndef HAVE_XINPUT2 + Lisp_Object tail, tem; + struct frame *f = NULL; + + FOR_EACH_FRAME (tail, tem) + { + if (FRAME_X_P (XFRAME (tem)) + && (FRAME_GTK_WIDGET (XFRAME (tem)) == widget)) + { + f = XFRAME (tem); + break; + } + } + if (f) + xg_maybe_send_low_level_key_event (f, event); +#endif + return true; +} + bool xg_filter_key (struct frame *frame, XEvent *xkey) { diff --git a/src/keyboard.c b/src/keyboard.c index f36243dd442..3b994a032e0 100644 --- a/src/keyboard.c +++ b/src/keyboard.c @@ -4273,6 +4273,7 @@ kbd_buffer_get_event (KBOARD **kbp, case CONFIG_CHANGED_EVENT: case FOCUS_OUT_EVENT: case SELECT_WINDOW_EVENT: + case LOW_LEVEL_KEY_EVENT: { obj = make_lispy_event (&event->ie); kbd_fetch_ptr = next_kbd_event (event); @@ -7117,12 +7118,47 @@ make_lispy_event (struct input_event *event) case PREEDIT_TEXT_EVENT: return list2 (Qpreedit_text, event->arg); + case LOW_LEVEL_KEY_EVENT: + return listn (6, Qlow_level_key, + XCAR (event->arg), /* Press or release. */ + XCAR (XCDR (event->arg)), /* The key symbol. */ + XCAR (XCDR (XCDR (event->arg))), /* The modifier. */ + make_fixnum (event->timestamp), + event->frame_or_window); + /* The 'kind' field of the event is something we don't recognize. */ default: emacs_abort (); } } +bool +kbd_low_level_key_is_enabled (int keysym, Lisp_Object modifier) +{ + if (Venable_low_level_key_events == Qt) + return true; + + if (Venable_low_level_key_events == Qnil) + return false; + + if (FIXNUMP (Venable_low_level_key_events)) + return keysym == XFIXNUM (Venable_low_level_key_events); + + if (Venable_low_level_key_events == Qmodifiers) + return modifier != Qnil; + + for (Lisp_Object e = Venable_low_level_key_events; CONSP (e); e = XCDR (e)) + { + Lisp_Object c = XCAR (e); + if (FIXNUMP (c) && XFIXNUM (c) == keysym) + return true; + if (c == Qmodifiers && modifier != Qnil) + return true; + } + + return false; +} + static Lisp_Object make_lispy_movement (struct frame *frame, Lisp_Object bar_window, enum scroll_bar_part part, Lisp_Object x, Lisp_Object y, Time t) @@ -12930,6 +12966,28 @@ syms_of_keyboard (void) DEFSYM (Qfile_notify, "file-notify"); #endif /* USE_FILE_NOTIFY */ + + DEFSYM (Qmodifiers, "modifiers"); + + DEFVAR_LISP ("enable-low-level-key-events", Venable_low_level_key_events, + doc: /* If non-nil, reception of low-level key events is enabled. + +The value configures the set of keys that are handled: + +If t, send events for all keys. + +If a number, send events for the corresponding keysym. This numbers are +platform dependente. See `llk-keysyms'. + +If a symbol, a predefined set of keys is selected. The only currently +valid symbol is 'modifiers. + +If a list of numbers and/or symbols, the corresponding keysyms and sets +are selected. */); + Venable_low_level_key_events = Qnil; + + DEFSYM (Qlow_level_key, "low-level-key"); + DEFSYM (Qtouch_end, "touch-end"); /* Menu and tool bar item parts. */ @@ -14017,6 +14075,8 @@ keys_of_keyboard (void) "handle-focus-out"); initial_define_lispy_key (Vspecial_event_map, "move-frame", "handle-move-frame"); + initial_define_lispy_key (Vspecial_event_map, "low-level-key", + "ignore"); } /* Mark the pointers in the kboard objects. diff --git a/src/keyboard.h b/src/keyboard.h index c1bb966d485..f0285f14a28 100644 --- a/src/keyboard.h +++ b/src/keyboard.h @@ -511,6 +511,7 @@ kbd_buffer_store_event_hold (struct input_event *event, extern Lisp_Object menu_item_eval_property (Lisp_Object); extern bool kbd_buffer_events_waiting (void); extern void add_user_signal (int, const char *); +extern bool kbd_low_level_key_is_enabled (int, Lisp_Object); extern int tty_read_avail_input (struct terminal *, struct input_event *); extern struct timespec timer_check (void); diff --git a/src/pgtkterm.c b/src/pgtkterm.c index 413cbd86c0d..3708c4ea7fc 100644 --- a/src/pgtkterm.c +++ b/src/pgtkterm.c @@ -5201,6 +5201,56 @@ pgtk_enqueue_preedit (struct frame *f, Lisp_Object preedit) evq_enqueue (&inev); } +static void +pgtk_maybe_send_low_level_key_event (struct frame *f, GdkEvent *event) +{ + GdkEventKey xkey = event->key; + bool is_press; + Lisp_Object key, modifier; + union buffered_input_event inev; + + if (NILP (Venable_low_level_key_events)) + return; + + switch (event->type) + { + case GDK_KEY_PRESS: + is_press = true; + break; + case GDK_KEY_RELEASE: + is_press = false; + break; + default: + return; + } + + /* We don't support modifier identification on PGTK. We only can tell + if the key corresponds to a modifier or not, which is used for + filtering enabled keys with kbd_low_level_key_is_enabled. */ + modifier = event->key.is_modifier ? Qt : Qnil; + + int keysym = xkey.keyval; + + if (keysym >= GDK_KEY_a && keysym <= GDK_KEY_z) + keysym -= GDK_KEY_a - GDK_KEY_A; + if (!kbd_low_level_key_is_enabled (keysym, modifier)) + return; + + if (!f) + f = pgtk_any_window_to_frame (event->key.window); + if (!f) + return; + + key = make_fixnum (keysym); + + EVENT_INIT (inev.ie); + XSETFRAME (inev.ie.frame_or_window, f); + inev.ie.kind = LOW_LEVEL_KEY_EVENT; + inev.ie.timestamp = event->key.time; + inev.ie.arg = list3 (is_press ? Qt : Qnil, key, modifier); + evq_enqueue (&inev); +} + static gboolean key_press_event (GtkWidget *widget, GdkEvent *event, gpointer *user_data) { @@ -5211,6 +5261,9 @@ key_press_event (GtkWidget *widget, GdkEvent *event, gpointer *user_data) struct pgtk_display_info *dpyinfo; f = pgtk_any_window_to_frame (gtk_widget_get_window (widget)); + + pgtk_maybe_send_low_level_key_event (f, event); + EVENT_INIT (inev.ie); hlinfo = MOUSE_HL_INFO (f); nbytes = 0; @@ -5454,6 +5507,8 @@ key_release_event (GtkWidget *widget, GdkDisplay *display; struct pgtk_display_info *dpyinfo; + pgtk_maybe_send_low_level_key_event (NULL, event); + display = gtk_widget_get_display (widget); dpyinfo = pgtk_display_info_for_display (display); diff --git a/src/termhooks.h b/src/termhooks.h index b32804a57b3..f75c119a9da 100644 --- a/src/termhooks.h +++ b/src/termhooks.h @@ -347,6 +347,7 @@ #define EMACS_TERMHOOKS_H /* In a NOTIFICATION_EVENT, .arg is a lambda to evaluate. */ , NOTIFICATION_EVENT #endif /* HAVE_ANDROID */ + , LOW_LEVEL_KEY_EVENT }; /* Bit width of an enum event_kind tag at the start of structs and unions. */ diff --git a/src/w32fns.c b/src/w32fns.c index c7963d2c616..49516d3bc7f 100644 --- a/src/w32fns.c +++ b/src/w32fns.c @@ -4669,6 +4669,11 @@ w32_wnd_proc (HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) case WM_KEYUP: case WM_SYSKEYUP: record_keyup (wParam, lParam); + if (!NILP (Venable_low_level_key_events)) + { + signal_user_input (); + my_post_msg (&wmsg, hwnd, WM_EMACS_LOW_LEVEL_KEY, wParam, lParam); + } goto dflt; case WM_KEYDOWN: @@ -4695,6 +4700,12 @@ w32_wnd_proc (HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) if (w32_use_fallback_wm_chars_method) wParam = map_keypad_keys (wParam, (lParam & 0x1000000L) != 0); + if (!NILP (Venable_low_level_key_events)) + { + signal_user_input (); + my_post_msg (&wmsg, hwnd, WM_EMACS_LOW_LEVEL_KEY, wParam, lParam); + } + windows_translate = 0; switch (wParam) diff --git a/src/w32term.c b/src/w32term.c index c81779b8517..ffc5b6ba522 100644 --- a/src/w32term.c +++ b/src/w32term.c @@ -5270,6 +5270,55 @@ w32_read_socket (struct terminal *terminal, } break; + case WM_EMACS_LOW_LEVEL_KEY: + WORD key_flags = HIWORD (msg.msg.lParam); + BOOL is_wm_keyup = key_flags & KF_UP; + + if (is_wm_keyup || (key_flags & KF_REPEAT) == 0) /* WM_KEYDOWN, not repeating. */ + { + WORD scan_code = LOBYTE (key_flags); + if (key_flags & KF_EXTENDED) + scan_code = MAKEWORD (scan_code, 0xE0); + + UINT translated = MapVirtualKey (scan_code, MAPVK_VSC_TO_VK_EX); + WORD vk = LOWORD (msg.msg.wParam); + if (translated) + vk = LOWORD (translated); + + Lisp_Object key = make_fixnum (vk); + Lisp_Object modifier = Qnil; + + switch (vk) + { + case VK_LSHIFT: + case VK_RSHIFT: + modifier = Qshift; + break; + case VK_LCONTROL: + case VK_RCONTROL: + modifier = Qctrl; + break; + case VK_LMENU: + case VK_RMENU: + modifier = Qmeta; + break; + } + + if (kbd_low_level_key_is_enabled (vk, modifier)) + { + f = w32_window_to_frame (dpyinfo, msg.msg.hwnd); + inev.kind = LOW_LEVEL_KEY_EVENT; + XSETFRAME (inev.frame_or_window, f); + inev.timestamp = msg.msg.time; + inev.arg = list3 (is_wm_keyup ? Qnil : Qt, key, modifier); + kbd_buffer_store_event_hold (&inev, hold_quit); + } + + inev.kind = NO_EVENT; + + } + break; + case WM_UNICHAR: case WM_SYSCHAR: case WM_CHAR: diff --git a/src/w32term.h b/src/w32term.h index 2483ca9036c..7290531a3ac 100644 --- a/src/w32term.h +++ b/src/w32term.h @@ -713,7 +713,8 @@ #define WM_EMACS_FILENOTIFY (WM_EMACS_START + 25) #define WM_EMACS_IME_STATUS (WM_EMACS_START + 26) #define WM_EMACS_DRAGOVER (WM_EMACS_START + 27) #define WM_EMACS_DROP (WM_EMACS_START + 28) -#define WM_EMACS_END (WM_EMACS_START + 29) +#define WM_EMACS_LOW_LEVEL_KEY (WM_EMACS_START + 29) +#define WM_EMACS_END (WM_EMACS_START + 30) #define WND_FONTWIDTH_INDEX (0) #define WND_LINEHEIGHT_INDEX (4) diff --git a/src/xterm.c b/src/xterm.c index 01e3a931ae9..51af5fcab22 100644 --- a/src/xterm.c +++ b/src/xterm.c @@ -17839,6 +17839,141 @@ #define STORE_KEYSYM_FOR_DEBUG(keysym) ((void)0) static struct x_display_info *next_noop_dpyinfo; +Lisp_Object +x_get_modifier_for_keycode (struct x_display_info *dpyinfo, + int keycode) +{ +#ifdef HAVE_XKB + if (dpyinfo->xkb_desc) + for (int mod = 0; mod < XkbNumModifiers; mod++) + { + int mask = (1 << mod); + if (dpyinfo->xkb_desc->map->modmap[keycode] & mask) + { + if (mask == ShiftMask) + return Qshift; + if (mask == ControlMask) + return Qctrl; + if (mask == dpyinfo->meta_mod_mask) + return Qmeta; + if (mask == dpyinfo->alt_mod_mask) + return Qalt; + if (mask == dpyinfo->super_mod_mask) + return Qsuper; + if (mask == dpyinfo->hyper_mod_mask) + return Qhyper; + } + } +#endif + XModifierKeymap *map = dpyinfo->modmap; + if (map) + for (int mod = 0; mod < 8; mod++) + { + int mask = (1 << mod); + for (int key = 0; key < map->max_keypermod; key++) + if (map->modifiermap[mod * map->max_keypermod + key] == keycode) + { + if (mask == ShiftMask) + return Qshift; + if (mask == ControlMask) + return Qctrl; + if (mask == dpyinfo->meta_mod_mask) + return Qmeta; + if (mask == dpyinfo->alt_mod_mask) + return Qalt; + if (mask == dpyinfo->super_mod_mask) + return Qsuper; + if (mask == dpyinfo->hyper_mod_mask) + return Qhyper; + } + } + return Qnil; +} + +static void +x_maybe_send_low_level_key_event (struct x_display_info *dpyinfo, + const XEvent *xev, struct frame *f) +{ + XKeyEvent xkey; + bool is_press; + KeySym keysym; + Lisp_Object key, modifier; + struct input_event ie; + + if (NILP (Venable_low_level_key_events)) + return; + + switch (xev->type) + { + case KeyPress: + is_press = true; + xkey = xev->xkey; + break; + case KeyRelease: + is_press = false; + xkey = xev->xkey; + break; +#ifdef HAVE_XINPUT2 + case GenericEvent: + XIDeviceEvent *xiev = xev->xcookie.data; + switch (xev->xgeneric.evtype) + { + case XI_KeyPress: + is_press = true; + break; + case XI_KeyRelease: + is_press = false; + break; + default: + return; + } + + xkey.serial = xiev->serial; + xkey.send_event = xiev->send_event; + xkey.display = xiev->display; + xkey.window = xiev->event; + xkey.root = xiev->root; + xkey.subwindow = xiev->child; + xkey.time = xiev->time; + xkey.x = xiev->event_x; + xkey.y = xiev->event_y; + xkey.x_root = xiev->root_x; + xkey.y_root = xiev->root_y; + xkey.state = xiev->mods.effective; + xkey.keycode = xiev->detail; + xkey.same_screen = 1; + break; +#endif + default: + return; + } + + if (!f) + f = x_any_window_to_frame (dpyinfo, xkey.window); + if (!f) + return; + + XLookupString (&xkey, NULL, 0, &keysym, NULL); + + modifier = x_get_modifier_for_keycode (dpyinfo, xkey.keycode); + + /* Convert lowercase latin letter to uppercase. */ + if (keysym >= XK_a && keysym <= XK_z) + keysym -= XK_a - XK_A; + + if (!kbd_low_level_key_is_enabled (keysym, modifier)) + return; + + key = make_fixnum (keysym); + + EVENT_INIT (ie); + XSETFRAME (ie.frame_or_window, f); + ie.kind = LOW_LEVEL_KEY_EVENT; + ie.timestamp = xkey.time; + ie.arg = list3 (is_press ? Qt : Qnil, key, modifier); + kbd_buffer_store_event (&ie); +} + /* Filter events for the current X input method. DPYINFO is the display this event is for. EVENT is the X event to filter. @@ -20205,6 +20340,7 @@ handle_one_xevent (struct x_display_info *dpyinfo, goto OTHER; case KeyPress: + x_maybe_send_low_level_key_event (dpyinfo, event, any); x_display_set_last_user_time (dpyinfo, event->xkey.time, event->xkey.send_event, true); @@ -20714,6 +20850,7 @@ handle_one_xevent (struct x_display_info *dpyinfo, #endif case KeyRelease: + x_maybe_send_low_level_key_event (dpyinfo, event, any); #ifdef HAVE_X_I18N /* Don't dispatch this event since XtDispatchEvent calls XFilterEvent, and two calls in a row may freeze the @@ -23969,6 +24106,8 @@ handle_one_xevent (struct x_display_info *dpyinfo, struct xi_device_t *device, *source; XKeyPressedEvent xkey; + x_maybe_send_low_level_key_event (dpyinfo, event, any); + coding = Qlatin_1; /* The code under this label is quite desultory. There @@ -24585,6 +24724,8 @@ handle_one_xevent (struct x_display_info *dpyinfo, #endif case XI_KeyRelease: + x_maybe_send_low_level_key_event (dpyinfo, event, any); + #if defined HAVE_X_I18N || defined USE_GTK || defined USE_LUCID { XKeyPressedEvent xkey; @@ -32661,6 +32802,7 @@ syms_of_xterm (void) Vx_toolkit_scroll_bars = Qnil; #endif + DEFSYM (Qshift, "shift"); DEFSYM (Qmodifier_value, "modifier-value"); DEFSYM (Qctrl, "ctrl"); Fput (Qctrl, Qmodifier_value, make_fixnum (ctrl_modifier)); diff --git a/src/xterm.h b/src/xterm.h index 7c2fadbf094..5735e9dfe19 100644 --- a/src/xterm.h +++ b/src/xterm.h @@ -1906,6 +1906,8 @@ x_mutable_colormap (XVisualInfo *visual) extern void tear_down_x_back_buffer (struct frame *f); extern void initial_set_up_x_back_buffer (struct frame *f); +extern Lisp_Object x_get_modifier_for_keycode (struct x_display_info *, int); + /* Defined in xfns.c. */ extern void x_real_positions (struct frame *, int *, int *); extern void x_change_tab_bar_height (struct frame *, int); -- 2.35.1.windows.2