1 module xkbcommon.x11;
2 /*
3  * Copyright © 2013 Ran Benita
4  *
5  * Permission is hereby granted, free of charge, to any person obtaining a
6  * copy of this software and associated documentation files (the "Software"),
7  * to deal in the Software without restriction, including without limitation
8  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
9  * and/or sell copies of the Software, and to permit persons to whom the
10  * Software is furnished to do so, subject to the following conditions:
11  *
12  * The above copyright notice and this permission notice (including the next
13  * paragraph) shall be included in all copies or substantial portions of the
14  * Software.
15  *
16  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
19  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
22  * DEALINGS IN THE SOFTWARE.
23  */
24 
25 import xcb.xcb;
26 import xkbcommon.xkbcommon;
27 
28 extern(C):
29 /**
30  * @file
31  * libxkbcommon-x11 API - Additional X11 support for xkbcommon.
32  */
33 
34 /**
35  * @defgroup x11 X11 support
36  * Additional X11 support for xkbcommon.
37  * @since 0.4.0
38  *
39  * @{
40  */
41 
42 /**
43  * @page x11-overview Overview
44  * @parblock
45  *
46  * The xkbcommon-x11 module provides a means for creating an xkb_keymap
47  * corresponding to the currently active keymap on the X server.  To do
48  * so, it queries the XKB X11 extension using the xcb-xkb library.  It
49  * can be used as a replacement for Xlib's keyboard handling.
50  *
51  * Following is an example workflow using xkbcommon-x11.  A complete
52  * example may be found in the test/interactive-x11.c file in the
53  * xkbcommon source repository.  On startup:
54  *
55  * 1. Connect to the X server using xcb_connect().
56  * 2. Setup the XKB X11 extension.  You can do this either by using the
57  *    xcb_xkb_use_extension() request directly, or by using the
58  *    xkb_x11_setup_xkb_extension() helper function.
59  *
60  * The XKB extension supports using separate keymaps and states for
61  * different keyboard devices.  The devices are identified by an integer
62  * device ID and are managed by another X11 extension, XInput (or its
63  * successor, XInput2).  The original X11 protocol only had one keyboard
64  * device, called the "core keyboard", which is still supported as a
65  * "virtual device".
66  *
67  * 3. We will use the core keyboard as an example.  To get its device ID,
68  *    use either the xcb_xkb_get_device_info() request directly, or the
69  *    xkb_x11_get_core_keyboard_device_id() helper function.
70  * 4. Create an initial xkb_keymap for this device, using the
71  *    xkb_x11_keymap_new_from_device() function.
72  * 5. Create an initial xkb_state for this device, using the
73  *    xkb_x11_state_new_from_device() function.
74  *
75  * @note At this point, you may consider setting various XKB controls and
76  * XKB per-client flags.  For example, enabling detectable autorepeat: \n
77  * http://www.x.org/releases/current/doc/kbproto/xkbproto.html#Detectable_Autorepeat
78  *
79  * Next, you need to react to state changes (e.g. a modifier was pressed,
80  * the layout was changed) and to keymap changes (e.g. a tool like xkbcomp,
81  * setxkbmap or xmodmap was used):
82  *
83  * 6. Select to listen to at least the following XKB events:
84  *    NewKeyboardNotify, MapNotify, StateNotify; using the
85  *    xcb_xkb_select_events_aux() request.
86  * 7. When NewKeyboardNotify or MapNotify are received, recreate the
87  *    xkb_keymap and xkb_state as described above.
88  * 8. When StateNotify is received, update the xkb_state accordingly
89  *    using the xkb_state_update_mask() function.
90  *
91  * @note It is also possible to use the KeyPress/KeyRelease @p state
92  * field to find the effective modifier and layout state, instead of
93  * using XkbStateNotify: \n
94  * http://www.x.org/releases/current/doc/kbproto/xkbproto.html#Computing_A_State_Field_from_an_XKB_State
95  * \n However, XkbStateNotify is more accurate.
96  *
97  * @note There is no need to call xkb_state_update_key(); the state is
98  * already synchronized.
99  *
100  * Finally, when a key event is received, you can use ordinary xkbcommon
101  * functions, like xkb_state_key_get_one_sym() and xkb_state_key_get_utf8(),
102  * as you normally would.
103  *
104  * @endparblock
105  */
106 
107 /**
108  * The minimal compatible major version of the XKB X11 extension which
109  * this library can use.
110  */
111 enum XKB_X11_MIN_MAJOR_XKB_VERSION = 1;
112 /**
113  * The minimal compatible minor version of the XKB X11 extension which
114  * this library can use (for the minimal major version).
115  */
116 enum XKB_X11_MIN_MINOR_XKB_VERSION = 0;
117 
118 /** Flags for the xkb_x11_setup_xkb_extension() function. */
119 enum xkb_x11_setup_xkb_extension_flags {
120     /** Do not apply any flags. */
121     XKB_X11_SETUP_XKB_EXTENSION_NO_FLAGS = 0
122 }
123 alias XKB_X11_SETUP_XKB_EXTENSION_NO_FLAGS = xkb_x11_setup_xkb_extension_flags.XKB_X11_SETUP_XKB_EXTENSION_NO_FLAGS;
124 
125 /**
126  * Setup the XKB X11 extension for this X client.
127  *
128  * The xkbcommon-x11 library uses various XKB requests.  Before doing so,
129  * an X client must notify the server that it will be using the extension.
130  * This function (or an XCB equivalent) must be called before any other
131  * function in this library is used.
132  *
133  * Some X servers may not support or disable the XKB extension.  If you
134  * want to support such servers, you need to use a different fallback.
135  *
136  * You may call this function several times; it is idempotent.
137  *
138  * @param connection
139  *     An XCB connection to the X server.
140  * @param major_xkb_version
141  *     See @p minor_xkb_version.
142  * @param minor_xkb_version
143  *     The XKB extension version to request.  To operate correctly, you
144  *     must have (major_xkb_version, minor_xkb_version) >=
145  *     (XKB_X11_MIN_MAJOR_XKB_VERSION, XKB_X11_MIN_MINOR_XKB_VERSION),
146  *     though this is not enforced.
147  * @param flags
148  *     Optional flags, or 0.
149  * @param[out] major_xkb_version_out
150  *     See @p minor_xkb_version_out.
151  * @param[out] minor_xkb_version_out
152  *     Backfilled with the compatible XKB extension version numbers picked
153  *     by the server.  Can be NULL.
154  * @param[out] base_event_out
155  *     Backfilled with the XKB base (also known as first) event code, needed
156  *     to distinguish XKB events.  Can be NULL.
157  * @param[out] base_error_out
158  *     Backfilled with the XKB base (also known as first) error code, needed
159  *     to distinguish XKB errors.  Can be NULL.
160  *
161  * @returns 1 on success, or 0 on failure.
162  */
163 int
164 xkb_x11_setup_xkb_extension(xcb_connection_t *connection,
165                             ushort major_xkb_version,
166                             ushort minor_xkb_version,
167                             xkb_x11_setup_xkb_extension_flags flags,
168                             ushort *major_xkb_version_out,
169                             ushort *minor_xkb_version_out,
170                             ubyte *base_event_out,
171                             ubyte *base_error_out);
172 
173 /**
174  * Get the keyboard device ID of the core X11 keyboard.
175  *
176  * @param connection An XCB connection to the X server.
177  *
178  * @returns A device ID which may be used with other xkb_x11_* functions,
179  *          or -1 on failure.
180  */
181 int
182 xkb_x11_get_core_keyboard_device_id(xcb_connection_t *connection);
183 
184 /**
185  * Create a keymap from an X11 keyboard device.
186  *
187  * This function queries the X server with various requests, fetches the
188  * details of the active keymap on a keyboard device, and creates an
189  * xkb_keymap from these details.
190  *
191  * @param context
192  *     The context in which to create the keymap.
193  * @param connection
194  *     An XCB connection to the X server.
195  * @param device_id
196  *     An XInput 1 device ID (in the range 0-255) with input class KEY.
197  *     Passing values outside of this range is an error.
198  * @param flags
199  *     Optional flags for the keymap, or 0.
200  *
201  * @returns A keymap retrieved from the X server, or NULL on failure.
202  *
203  * @memberof xkb_keymap
204  */
205 xkb_keymap *
206 xkb_x11_keymap_new_from_device(xkb_context *context,
207                                xcb_connection_t *connection,
208                                int device_id,
209                                xkb_keymap_compile_flags flags);
210 
211 /**
212  * Create a new keyboard state object from an X11 keyboard device.
213  *
214  * This function is the same as xkb_state_new(), only pre-initialized
215  * with the state of the device at the time this function is called.
216  *
217  * @param keymap
218  *     The keymap for which to create the state.
219  * @param connection
220  *     An XCB connection to the X server.
221  * @param device_id
222  *     An XInput 1 device ID (in the range 0-255) with input class KEY.
223  *     Passing values outside of this range is an error.
224  *
225  * @returns A new keyboard state object, or NULL on failure.
226  *
227  * @memberof xkb_state
228  */
229 xkb_state *
230 xkb_x11_state_new_from_device(xkb_keymap *keymap,
231                               xcb_connection_t *connection,
232                               int device_id);
233