4 * Copyright 2003 CodeWeavers (Aric Stewart)
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
22 #include "wine/port.h"
31 #include "wine/library.h"
32 #include "wine/unicode.h"
33 #include "wine/debug.h"
36 WINE_DEFAULT_DEBUG_CHANNEL(wintab32);
38 #define WT_MAX_NAME_LEN 256
40 typedef struct tagWTI_CURSORS_INFO
42 WCHAR NAME[WT_MAX_NAME_LEN];
43 /* a displayable zero-terminated string containing the name of the
47 /* whether the cursor is currently connected. */
49 /* a bit mask indicating the packet data items supported when this
50 * cursor is connected.
53 /* the number of buttons on this cursor. */
55 /* the number of bits of raw button data returned by the hardware.*/
58 /* a list of zero-terminated strings containing the names of the
59 * cursor's buttons. The number of names in the list is the same as the
60 * number of buttons on the cursor. The names are separated by a single
61 * zero character; the list is terminated by two zero characters.
64 /* a 32 byte array of logical button numbers, one for each physical
68 /* a 32 byte array of button action codes, one for each logical
72 /* the physical button number of the button that is controlled by normal
76 /* an array of two UINTs, specifying the button marks for the normal
77 * pressure button. The first UINT contains the release mark; the second
78 * contains the press mark.
81 /* an array of UINTs describing the pressure response curve for normal
85 /* the physical button number of the button that is controlled by
86 * tangential pressure.
89 /* an array of two UINTs, specifying the button marks for the tangential
90 * pressure button. The first UINT contains the release mark; the second
91 * contains the press mark.
94 /* an array of UINTs describing the pressure response curve for
95 * tangential pressure.
98 /* a manufacturer-specific physical identifier for the cursor. This
99 * value will distinguish the physical cursor from others on the same
100 * device. This physical identifier allows applications to bind
101 * functions to specific physical cursors, even if category numbers
102 * change and multiple, otherwise identical, physical cursors are
106 /* the cursor mode number of this cursor type, if this cursor type has
107 * the CRC_MULTIMODE capability.
110 /* the minimum set of data available from a physical cursor in this
111 * cursor type, if this cursor type has the CRC_AGGREGATE capability.
114 /* the minimum number of buttons of physical cursors in the cursor type,
115 * if this cursor type has the CRC_AGGREGATE capability.
118 /* flags indicating cursor capabilities, as defined below:
120 Indicates this cursor type describes one of several modes of a
121 single physical cursor. Consecutive cursor type categories
122 describe the modes; the CSR_MODE data item gives the mode number
125 Indicates this cursor type describes several physical cursors
126 that cannot be distinguished by software.
128 Indicates this cursor type describes the physical cursor in its
129 inverted orientation; the previous consecutive cursor type
130 category describes the normal orientation.
133 /* Manufacturer Unique id for the item type */
134 } WTI_CURSORS_INFO, *LPWTI_CURSORS_INFO;
137 typedef struct tagWTI_DEVICES_INFO
139 WCHAR NAME[WT_MAX_NAME_LEN];
140 /* a displayable null- terminated string describing the device,
141 * manufacturer, and revision level.
144 /* flags indicating hardware and driver capabilities, as defined
147 Indicates that the display and digitizer share the same surface.
149 Indicates that the cursor must be in physical contact with the
150 device to report position.
152 Indicates that device can generate events when the cursor is
153 entering and leaving the physical detection range.
155 Indicates that device can uniquely identify the active cursor in
159 /* the number of supported cursor types.*/
161 /* the first cursor type number for the device. */
163 /* the maximum packet report rate in Hertz. */
165 /* a bit mask indicating which packet data items are always available.*/
167 /* a bit mask indicating which packet data items are physically
168 * relative, i.e., items for which the hardware can only report change,
169 * not absolute measurement.
172 /* a bit mask indicating which packet data items are only available when
173 * certain cursors are connected. The individual cursor descriptions
174 * must be consulted to determine which cursors return which data.
179 /* the size of tablet context margins in tablet native coordinates, in
180 * the x, y, and z directions, respectively.
185 /* the tablet's range and resolution capabilities, in the x, y, and z
186 * axes, respectively.
190 /* the tablet's range and resolution capabilities, for the normal and
191 * tangential pressure inputs, respectively.
194 /* a 3-element array describing the tablet's orientation range and
195 * resolution capabilities.
198 /* a 3-element array describing the tablet's rotation range and
199 * resolution capabilities.
201 WCHAR PNPID[WT_MAX_NAME_LEN];
202 /* a null-terminated string containing the devices Plug and Play ID.*/
203 } WTI_DEVICES_INFO, *LPWTI_DEVICES_INFO;
206 /***********************************************************************
207 * WACOM WINTAB EXTENSIONS TO SUPPORT CSR_TYPE
208 * In Wintab 1.2, a CSR_TYPE feature was added. This adds the
209 * ability to return a type of cursor on a tablet.
210 * Unfortunately, we cannot get the cursor type directly from X,
211 * and it is not specified directly anywhere. So we virtualize
212 * the type here. (This is unfortunate, the kernel module has
213 * the exact type, but we have no way of getting that module to
214 * pass us that type).
216 * Reference linuxwacom driver project wcmCommon.c function
217 * idtotype for a much larger list of CSR_TYPE.
219 * http://linuxwacom.cvs.sourceforge.net/linuxwacom/linuxwacom-prod/src/xdrv/wcmCommon.c?view=markup
221 * The WTI_CURSORS_INFO.TYPE data is supposed to be used like this:
222 * (cursor.TYPE & 0x0F06) == target_cursor_type
223 * Reference: Section Unique ID
224 * http://www.wacomeng.com/devsupport/ibmpc/gddevpc.html
226 #define CSR_TYPE_PEN 0x822
227 #define CSR_TYPE_ERASER 0x82a
228 #define CSR_TYPE_MOUSE_2D 0x007
229 #define CSR_TYPE_MOUSE_4D 0x094
230 /* CSR_TYPE_OTHER is a special value! assumed no real world significance
231 * if a stylus type or eraser type eventually have this value
232 * it'll be a bug. As of 2008 05 21 we can be sure because
233 * linux wacom lists all the known values and this isn't one of them */
234 #define CSR_TYPE_OTHER 0x000
236 typedef struct tagWTPACKET {
247 UINT pkNormalPressure;
248 UINT pkTangentPressure;
249 ORIENTATION pkOrientation;
250 ROTATION pkRotation; /* 1.1 */
251 } WTPACKET, *LPWTPACKET;
256 #include <X11/Xlib.h>
257 #include <X11/extensions/XInput.h>
259 static int motion_type;
260 static int button_press_type;
261 static int button_release_type;
262 static int key_press_type;
263 static int key_release_type;
264 static int proximity_in_type;
265 static int proximity_out_type;
267 static HWND hwndTabletDefault;
268 static WTPACKET gMsgPacket;
269 static DWORD gSerial;
271 /* Reference: http://www.wacomeng.com/devsupport/ibmpc/gddevpc.html
273 * Cursors come in sets of 3 normally
274 * Cursor #0 = puck device 1
275 * Cursor #1 = stylus device 1
276 * Cursor #2 = eraser device 1
277 * Cursor #3 = puck device 2
278 * Cursor #4 = stylus device 2
279 * Cursor #5 = eraser device 2
282 * A dual tracking/multimode tablet is one
283 * that supports 2 independent cursors of the same or
284 * different types simultaneously on a single tablet.
285 * This makes our cursor layout potentially like this
286 * Cursor #0 = puck 1 device 1
287 * Cursor #1 = stylus 1 device 1
288 * Cursor #2 = eraser 1 device 1
289 * Cursor #3 = puck 2 device 1
290 * Cursor #4 = stylus 2 device 1
291 * Cursor #5 = eraser 2 device 1
292 * Cursor #6 = puck 1 device 2
295 * So with multimode tablets we could potentially need
296 * 2 slots of the same type per tablet i.e.
297 * you are using 2 styluses at once so they would
298 * get placed in Cursors #1 and Cursor #4
300 * Now say someone has 2 multimode tablets with 2 erasers each
301 * now we would need Cursor #2, #5, #8, #11
302 * So to support that we need CURSORMAX of 12 (0 to 11)
303 * FIXME: we don't support more than 4 regular tablets or 2 multimode tablets */
305 static INT button_state[CURSORMAX];
307 static LOGCONTEXTW gSysContext;
308 static WTI_DEVICES_INFO gSysDevice;
309 static WTI_CURSORS_INFO gSysCursor[CURSORMAX];
310 static INT gNumCursors; /* do NOT use this to iterate through gSysCursor slots */
314 static void *xinput_handle;
316 #define MAKE_FUNCPTR(f) static typeof(f) * p##f;
317 MAKE_FUNCPTR(XListInputDevices)
318 MAKE_FUNCPTR(XFreeDeviceList)
319 MAKE_FUNCPTR(XOpenDevice)
320 MAKE_FUNCPTR(XQueryDeviceState)
321 MAKE_FUNCPTR(XGetDeviceButtonMapping)
322 MAKE_FUNCPTR(XCloseDevice)
323 MAKE_FUNCPTR(XSelectExtensionEvent)
324 MAKE_FUNCPTR(XFreeDeviceState)
327 static INT X11DRV_XInput_Init(void)
329 xinput_handle = wine_dlopen(SONAME_LIBXI, RTLD_NOW, NULL, 0);
332 #define LOAD_FUNCPTR(f) if((p##f = wine_dlsym(xinput_handle, #f, NULL, 0)) == NULL) goto sym_not_found;
333 LOAD_FUNCPTR(XListInputDevices)
334 LOAD_FUNCPTR(XFreeDeviceList)
335 LOAD_FUNCPTR(XOpenDevice)
336 LOAD_FUNCPTR(XGetDeviceButtonMapping)
337 LOAD_FUNCPTR(XCloseDevice)
338 LOAD_FUNCPTR(XSelectExtensionEvent)
339 LOAD_FUNCPTR(XQueryDeviceState)
340 LOAD_FUNCPTR(XFreeDeviceState)
348 static int Tablet_ErrorHandler(Display *dpy, XErrorEvent *event, void* arg)
353 static void trace_axes(XValuatorInfoPtr val)
358 for (i = 0, axis = val->axes ; i < val->num_axes; i++, axis++)
359 TRACE(" Axis %d: [resolution %d|min_value %d|max_value %d]\n", i, axis->resolution, axis->min_value, axis->max_value);
362 static BOOL match_token(const char *haystack, const char *needle)
365 for (p = haystack; *p; )
367 while (*p && isspace(*p))
372 for (q = needle; *q && *p && tolower(*p) == tolower(*q); q++)
374 if (! *q && (isspace(*p) || !*p))
377 while (*p && ! isspace(*p))
383 /* Determining if an X device is a Tablet style device is an imperfect science.
384 ** We rely on common conventions around device names as well as the type reported
385 ** by Wacom tablets. This code will likely need to be expanded for alternate tablet types
387 ** Wintab refers to any device that interacts with the tablet as a cursor,
388 ** (stylus, eraser, tablet mouse, airbrush, etc)
389 ** this is not to be confused with wacom x11 configuration "cursor" device.
390 ** Wacoms x11 config "cursor" refers to its device slot (which we mirror with
391 ** our gSysCursors) for puck like devices (tablet mice essentially).
394 static BOOL is_tablet_cursor(const char *name, const char *type)
397 static const char *tablet_cursor_whitelist[] = {
409 for (i=0; tablet_cursor_whitelist[i] != NULL; i++) {
410 if (name && match_token(name, tablet_cursor_whitelist[i]))
412 if (type && match_token(type, tablet_cursor_whitelist[i]))
418 static BOOL is_stylus(const char *name, const char *type)
421 static const char* tablet_stylus_whitelist[] = {
428 for (i=0; tablet_stylus_whitelist[i] != NULL; i++) {
429 if (name && match_token(name, tablet_stylus_whitelist[i]))
431 if (type && match_token(type, tablet_stylus_whitelist[i]))
438 static BOOL is_eraser(const char *name, const char *type)
440 if (name && match_token(name, "eraser"))
442 if (type && match_token(type, "eraser"))
447 /* cursors are placed in gSysCursor rows depending on their type
448 * see CURSORMAX comments for more detail */
449 static BOOL add_system_cursor(LPWTI_CURSORS_INFO cursor)
453 if (cursor->TYPE == CSR_TYPE_PEN)
455 else if (cursor->TYPE == CSR_TYPE_ERASER)
458 for (; offset < CURSORMAX; offset += 3)
460 if (!gSysCursor[offset].ACTIVE)
462 gSysCursor[offset] = *cursor;
471 static void disable_system_cursors(void)
475 for (i = 0; i < CURSORMAX; ++i)
477 gSysCursor[i].ACTIVE = 0;
484 /***********************************************************************
485 * X11DRV_LoadTabletInfo (X11DRV.@)
487 BOOL CDECL X11DRV_LoadTabletInfo(HWND hwnddefault)
489 const WCHAR SZ_CONTEXT_NAME[] = {'W','i','n','e',' ','T','a','b','l','e','t',' ','C','o','n','t','e','x','t',0};
490 const WCHAR SZ_DEVICE_NAME[] = {'W','i','n','e',' ','T','a','b','l','e','t',' ','D','e','v','i','c','e',0};
491 const WCHAR SZ_NON_PLUGINPLAY[] = {'n','o','n','-','p','l','u','g','i','n','p','l','a','y',0};
493 struct x11drv_thread_data *data = x11drv_init_thread_data();
496 XDeviceInfo *devices;
497 XDeviceInfo *target = NULL;
498 BOOL axis_read_complete= FALSE;
501 XButtonInfoPtr Button;
502 XValuatorInfoPtr Val;
507 if (!X11DRV_XInput_Init())
509 ERR("Unable to initialize the XInput library.\n");
513 hwndTabletDefault = hwnddefault;
515 /* Do base initialization */
516 strcpyW(gSysContext.lcName, SZ_CONTEXT_NAME);
517 strcpyW(gSysDevice.NAME, SZ_DEVICE_NAME);
519 gSysContext.lcOptions = CXO_SYSTEM;
520 gSysContext.lcLocks = CXL_INSIZE | CXL_INASPECT | CXL_MARGIN |
521 CXL_SENSITIVITY | CXL_SYSOUT;
523 gSysContext.lcMsgBase= WT_DEFBASE;
524 gSysContext.lcDevice = 0;
525 gSysContext.lcPktData =
526 PK_CONTEXT | PK_STATUS | PK_SERIAL_NUMBER| PK_TIME | PK_CURSOR |
527 PK_BUTTONS | PK_X | PK_Y | PK_NORMAL_PRESSURE | PK_ORIENTATION;
528 gSysContext.lcMoveMask=
529 PK_BUTTONS | PK_X | PK_Y | PK_NORMAL_PRESSURE | PK_ORIENTATION;
530 gSysContext.lcStatus = CXS_ONTOP;
531 gSysContext.lcPktRate = 100;
532 gSysContext.lcBtnDnMask = 0xffffffff;
533 gSysContext.lcBtnUpMask = 0xffffffff;
534 gSysContext.lcSensX = 65536;
535 gSysContext.lcSensY = 65536;
536 gSysContext.lcSensX = 65536;
537 gSysContext.lcSensZ = 65536;
538 gSysContext.lcSysSensX= 65536;
539 gSysContext.lcSysSensY= 65536;
541 /* initialize cursors */
542 disable_system_cursors();
544 /* Device Defaults */
545 gSysDevice.HARDWARE = HWC_HARDPROX|HWC_PHYSID_CURSORS;
546 gSysDevice.FIRSTCSR= 0;
547 gSysDevice.PKTRATE = 100;
549 PK_CONTEXT | PK_STATUS | PK_SERIAL_NUMBER| PK_TIME | PK_CURSOR |
550 PK_BUTTONS | PK_X | PK_Y | PK_NORMAL_PRESSURE | PK_ORIENTATION;
551 strcpyW(gSysDevice.PNPID, SZ_NON_PLUGINPLAY);
553 devices = pXListInputDevices(data->display, &num_devices);
556 WARN("XInput Extensions reported as not available\n");
559 TRACE("XListInputDevices reports %d devices\n", num_devices);
560 for (loop=0; loop < num_devices; loop++)
563 char *device_type = devices[loop].type ? XGetAtomName(data->display, devices[loop].type) : NULL;
564 WTI_CURSORS_INFO cursor;
566 TRACE("Device %i: [id %d|name %s|type %s|num_classes %d|use %d]\n",
567 loop, (int) devices[loop].id, devices[loop].name, device_type ? device_type : "",
568 devices[loop].num_classes, devices[loop].use );
570 switch (devices[loop].use)
572 case IsXExtensionDevice:
573 #ifdef IsXExtensionPointer
574 case IsXExtensionPointer:
576 #ifdef IsXExtensionKeyboard
577 case IsXExtensionKeyboard:
579 TRACE("Is XExtension: Device, Keyboard, or Pointer\n");
580 target = &devices[loop];
582 if (strlen(target->name) >= WT_MAX_NAME_LEN)
584 ERR("Input device '%s' name too long - skipping\n", wine_dbgstr_a(target->name));
588 X11DRV_expect_error(data->display, Tablet_ErrorHandler, NULL);
589 opendevice = pXOpenDevice(data->display,target->id);
590 if (!X11DRV_check_error() && opendevice)
592 unsigned char map[32];
596 X11DRV_expect_error(data->display,Tablet_ErrorHandler,NULL);
597 cursor.BUTTONS = pXGetDeviceButtonMapping(data->display, opendevice, map, 32);
598 if (X11DRV_check_error() || cursor.BUTTONS <= 0)
600 TRACE("No buttons, Non Tablet Device\n");
601 pXCloseDevice(data->display, opendevice);
605 for (i=0; i< cursor.BUTTONS; i++,shft++)
607 cursor.BUTTONMAP[i] = map[i];
608 cursor.SYSBTNMAP[i] = (1<<shft);
610 pXCloseDevice(data->display, opendevice);
614 WARN("Unable to open device %s\n",target->name);
617 MultiByteToWideChar(CP_UNIXCP, 0, target->name, -1, cursor.NAME, WT_MAX_NAME_LEN);
619 if (! is_tablet_cursor(target->name, device_type))
621 WARN("Skipping device %d [name %s|type %s]; not apparently a tablet cursor type device. If this is wrong, please report it to wine-devel@winehq.org\n",
622 loop, devices[loop].name, device_type ? device_type : "");
627 cursor.PKTDATA = PK_TIME | PK_CURSOR | PK_BUTTONS | PK_X | PK_Y |
628 PK_NORMAL_PRESSURE | PK_TANGENT_PRESSURE |
631 cursor.PHYSID = target->id;
633 cursor.NPBTNMARKS[0] = 0 ;
634 cursor.NPBTNMARKS[1] = 1 ;
635 cursor.CAPABILITIES = CRC_MULTIMODE;
637 /* prefer finding TYPE_PEN(most capable) */
638 if (is_stylus(target->name, device_type))
639 cursor.TYPE = CSR_TYPE_PEN;
640 else if (is_eraser(target->name, device_type))
641 cursor.TYPE = CSR_TYPE_ERASER;
643 cursor.TYPE = CSR_TYPE_OTHER;
645 any = target->inputclassinfo;
647 for (class_loop = 0; class_loop < target->num_classes; class_loop++)
653 Val = (XValuatorInfoPtr) any;
654 TRACE(" ValidatorInput %d: [class %d|length %d|num_axes %d|mode %d|motion_buffer %ld]\n",
655 class_loop, (int) Val->class, Val->length, Val->num_axes, Val->mode, Val->motion_buffer);
656 if (TRACE_ON(wintab32))
659 /* FIXME: This is imperfect; we compute our devices capabilities based upon the
660 ** first pen type device we find. However, a more correct implementation
661 ** would require acquiring a wide variety of tablets and running through
662 ** the various inputs to see what the values are. Odds are that a
663 ** more 'correct' algorithm would condense to this one anyway.
665 if (!axis_read_complete && cursor.TYPE == CSR_TYPE_PEN)
667 Axis = (XAxisInfoPtr) ((char *) Val + sizeof
670 if (Val->num_axes>=1)
673 gSysDevice.X.axMin = Axis->min_value;
674 gSysDevice.X.axMax= Axis->max_value;
675 gSysDevice.X.axUnits = TU_INCHES;
676 gSysDevice.X.axResolution = Axis->resolution;
677 gSysContext.lcInOrgX = Axis->min_value;
678 gSysContext.lcSysOrgX = Axis->min_value;
679 gSysContext.lcInExtX = Axis->max_value;
680 gSysContext.lcSysExtX = Axis->max_value;
683 if (Val->num_axes>=2)
686 gSysDevice.Y.axMin = Axis->min_value;
687 gSysDevice.Y.axMax= Axis->max_value;
688 gSysDevice.Y.axUnits = TU_INCHES;
689 gSysDevice.Y.axResolution = Axis->resolution;
690 gSysContext.lcInOrgY = Axis->min_value;
691 gSysContext.lcSysOrgY = Axis->min_value;
692 gSysContext.lcInExtY = Axis->max_value;
693 gSysContext.lcSysExtY = Axis->max_value;
696 if (Val->num_axes>=3)
698 /* Axis 3 is Normal Pressure */
699 gSysDevice.NPRESSURE.axMin = Axis->min_value;
700 gSysDevice.NPRESSURE.axMax= Axis->max_value;
701 gSysDevice.NPRESSURE.axUnits = TU_INCHES;
702 gSysDevice.NPRESSURE.axResolution =
706 if (Val->num_axes >= 5)
708 /* Axis 4 and 5 are X and Y tilt */
709 XAxisInfoPtr XAxis = Axis;
711 if (max (abs(Axis->max_value),
712 abs(XAxis->max_value)))
714 gSysDevice.ORIENTATION[0].axMin = 0;
715 gSysDevice.ORIENTATION[0].axMax = 3600;
716 gSysDevice.ORIENTATION[0].axUnits = TU_CIRCLE;
717 gSysDevice.ORIENTATION[0].axResolution
719 gSysDevice.ORIENTATION[1].axMin = -1000;
720 gSysDevice.ORIENTATION[1].axMax = 1000;
721 gSysDevice.ORIENTATION[1].axUnits = TU_CIRCLE;
722 gSysDevice.ORIENTATION[1].axResolution
727 axis_read_complete = TRUE;
736 Button = (XButtonInfoPtr) any;
737 TRACE(" ButtonInput %d: [class %d|length %d|num_buttons %d]\n",
738 class_loop, (int) Button->class, Button->length, Button->num_buttons);
739 cursor.BTNNAMES = HeapAlloc(GetProcessHeap(), 0, sizeof(WCHAR)*cchBuf);
740 for (i = 0; i < cursor.BUTTONS; i++)
742 /* FIXME - these names are probably incorrect */
743 int cch = strlenW(cursor.NAME) + 1;
744 while (cch > cchBuf - cchPos - 1) /* we want one extra byte for the last NUL */
747 cursor.BTNNAMES = HeapReAlloc(GetProcessHeap(), 0, cursor.BTNNAMES, sizeof(WCHAR)*cchBuf);
750 strcpyW(cursor.BTNNAMES + cchPos, cursor.NAME);
753 cursor.BTNNAMES[cchPos++] = 0;
754 cursor.BTNNAMES = HeapReAlloc(GetProcessHeap(), 0, cursor.BTNNAMES, sizeof(WCHAR)*cchPos);
755 cursor.cchBTNNAMES = cchPos;
758 } /* switch any->class */
759 any = (XAnyClassPtr) ((char*) any + any->length);
760 } /* for class_loop */
761 if (!add_system_cursor(&cursor))
762 FIXME("Skipping this cursor due to lack of system cursor slots.\n");
764 } /* switch devices.use */
766 } /* for XListInputDevices */
767 pXFreeDeviceList(devices);
769 if (axis_read_complete)
770 gSysDevice.NCSRTYPES = gNumCursors;
773 disable_system_cursors();
774 WARN("Did not find a valid stylus, unable to determine system context parameters. Wintab is disabled.\n");
780 static int figure_deg(int x, int y)
784 angle = atan2((float)y, (float)x);
789 return (0.5 + (angle * 1800.0 / M_PI));
792 static int get_button_state(int curnum)
794 return button_state[curnum];
797 static void set_button_state(int curnum, XID deviceid)
799 struct x11drv_thread_data *data = x11drv_thread_data();
806 device = pXOpenDevice(data->display,deviceid);
807 state = pXQueryDeviceState(data->display,device);
812 for (loop = 0; loop < state->num_classes; loop++)
814 if (class->class == ButtonClass)
817 XButtonState *button_state = (XButtonState*)class;
818 for (loop2 = 0; loop2 < button_state->num_buttons; loop2++)
820 if (button_state->buttons[loop2 / 8] & (1 << (loop2 % 8)))
826 class = (XInputClass *) ((char *) class + class->length);
829 pXFreeDeviceState(state);
830 button_state[curnum] = rc;
833 static int cursor_from_device(DWORD deviceid, LPWTI_CURSORS_INFO *cursorp)
836 for (i = 0; i < CURSORMAX; i++)
837 if (gSysCursor[i].ACTIVE && gSysCursor[i].PHYSID == deviceid)
839 *cursorp = &gSysCursor[i];
843 ERR("Could not map device id %d to a cursor\n", (int) deviceid);
847 static void motion_event( HWND hwnd, XEvent *event )
849 XDeviceMotionEvent *motion = (XDeviceMotionEvent *)event;
850 LPWTI_CURSORS_INFO cursor;
851 int curnum = cursor_from_device(motion->deviceid, &cursor);
855 memset(&gMsgPacket,0,sizeof(WTPACKET));
857 TRACE("Received tablet motion event (%p); device id %d, cursor num %d\n",hwnd, (int) motion->deviceid, curnum);
859 /* Set cursor to inverted if cursor is the eraser */
860 gMsgPacket.pkStatus = (cursor->TYPE == CSR_TYPE_ERASER ? TPS_INVERT:0);
861 gMsgPacket.pkTime = EVENT_x11_time_to_win32_time(motion->time);
862 gMsgPacket.pkSerialNumber = gSerial++;
863 gMsgPacket.pkCursor = curnum;
864 gMsgPacket.pkX = motion->axis_data[0];
865 gMsgPacket.pkY = motion->axis_data[1];
866 gMsgPacket.pkOrientation.orAzimuth = figure_deg(motion->axis_data[3],motion->axis_data[4]);
867 gMsgPacket.pkOrientation.orAltitude = ((1000 - 15 * max
868 (abs(motion->axis_data[3]),
869 abs(motion->axis_data[4])))
870 * (gMsgPacket.pkStatus & TPS_INVERT?-1:1));
871 gMsgPacket.pkNormalPressure = motion->axis_data[2];
872 gMsgPacket.pkButtons = get_button_state(curnum);
873 SendMessageW(hwndTabletDefault,WT_PACKET,gMsgPacket.pkSerialNumber,(LPARAM)hwnd);
876 static void button_event( HWND hwnd, XEvent *event )
878 XDeviceButtonEvent *button = (XDeviceButtonEvent *) event;
879 LPWTI_CURSORS_INFO cursor;
880 int curnum = cursor_from_device(button->deviceid, &cursor);
884 memset(&gMsgPacket,0,sizeof(WTPACKET));
886 TRACE("Received tablet button %s event\n", (event->type == button_press_type)?"press":"release");
888 /* Set cursor to inverted if cursor is the eraser */
889 gMsgPacket.pkStatus = (cursor->TYPE == CSR_TYPE_ERASER ? TPS_INVERT:0);
890 set_button_state(curnum, button->deviceid);
891 gMsgPacket.pkTime = EVENT_x11_time_to_win32_time(button->time);
892 gMsgPacket.pkSerialNumber = gSerial++;
893 gMsgPacket.pkCursor = curnum;
894 gMsgPacket.pkX = button->axis_data[0];
895 gMsgPacket.pkY = button->axis_data[1];
896 gMsgPacket.pkOrientation.orAzimuth = figure_deg(button->axis_data[3],button->axis_data[4]);
897 gMsgPacket.pkOrientation.orAltitude = ((1000 - 15 * max(abs(button->axis_data[3]),
898 abs(button->axis_data[4])))
899 * (gMsgPacket.pkStatus & TPS_INVERT?-1:1));
900 gMsgPacket.pkNormalPressure = button->axis_data[2];
901 gMsgPacket.pkButtons = get_button_state(curnum);
902 SendMessageW(hwndTabletDefault,WT_PACKET,gMsgPacket.pkSerialNumber,(LPARAM)hwnd);
905 static void key_event( HWND hwnd, XEvent *event )
907 if (event->type == key_press_type)
908 FIXME("Received tablet key press event\n");
910 FIXME("Received tablet key release event\n");
913 static void proximity_event( HWND hwnd, XEvent *event )
915 XProximityNotifyEvent *proximity = (XProximityNotifyEvent *) event;
916 LPWTI_CURSORS_INFO cursor;
917 int curnum = cursor_from_device(proximity->deviceid, &cursor);
918 LPARAM proximity_info;
920 TRACE("hwnd=%p\n", hwnd);
925 memset(&gMsgPacket,0,sizeof(WTPACKET));
927 /* Set cursor to inverted if cursor is the eraser */
928 gMsgPacket.pkStatus = (cursor->TYPE == CSR_TYPE_ERASER ? TPS_INVERT:0);
929 gMsgPacket.pkStatus |= (event->type==proximity_out_type)?TPS_PROXIMITY:0;
930 gMsgPacket.pkTime = EVENT_x11_time_to_win32_time(proximity->time);
931 gMsgPacket.pkSerialNumber = gSerial++;
932 gMsgPacket.pkCursor = curnum;
933 gMsgPacket.pkX = proximity->axis_data[0];
934 gMsgPacket.pkY = proximity->axis_data[1];
935 gMsgPacket.pkOrientation.orAzimuth = figure_deg(proximity->axis_data[3],proximity->axis_data[4]);
936 gMsgPacket.pkOrientation.orAltitude = ((1000 - 15 * max(abs(proximity->axis_data[3]),
937 abs(proximity->axis_data[4])))
938 * (gMsgPacket.pkStatus & TPS_INVERT?-1:1));
939 gMsgPacket.pkNormalPressure = proximity->axis_data[2];
940 gMsgPacket.pkButtons = get_button_state(curnum);
942 /* FIXME: LPARAM loword is true when cursor entering context, false when leaving context
943 * This needs to be handled here or in wintab32. Using the proximity_in_type is not correct
945 * LPARAM hiword is "non-zero when the cursor is leaving or entering hardware proximity"
946 * WPARAM contains context handle.
947 * HWND to HCTX is handled by wintab32.
949 proximity_info = MAKELPARAM((event->type == proximity_in_type),
950 (event->type == proximity_in_type) || (event->type == proximity_out_type));
951 SendMessageW(hwndTabletDefault, WT_PROXIMITY, (WPARAM)hwnd, proximity_info);
954 /***********************************************************************
955 * X11DRV_AttachEventQueueToTablet (X11DRV.@)
957 int CDECL X11DRV_AttachEventQueueToTablet(HWND hOwner)
959 struct x11drv_thread_data *data = x11drv_init_thread_data();
963 XDeviceInfo *devices;
964 XDeviceInfo *target = NULL;
966 XEventClass event_list[7];
967 Window win = X11DRV_get_whole_window( hOwner );
969 if (!win || !xinput_handle) return 0;
971 TRACE("Creating context for window %p (%lx) %i cursors\n", hOwner, win, gNumCursors);
973 devices = pXListInputDevices(data->display, &num_devices);
975 X11DRV_expect_error(data->display,Tablet_ErrorHandler,NULL);
976 for (cur_loop=0; cur_loop < CURSORMAX; cur_loop++)
978 char cursorNameA[WT_MAX_NAME_LEN];
981 if (!gSysCursor[cur_loop].ACTIVE) continue;
983 /* the cursor name fits in the buffer because too long names are skipped */
984 WideCharToMultiByte(CP_UNIXCP, 0, gSysCursor[cur_loop].NAME, -1, cursorNameA, WT_MAX_NAME_LEN, NULL, NULL);
985 for (loop=0; loop < num_devices; loop ++)
986 if (strcmp(devices[loop].name, cursorNameA) == 0)
987 target = &devices[loop];
989 WARN("Cursor Name %s not found in list of targets.\n", cursorNameA);
993 TRACE("Opening cursor %i id %i\n",cur_loop,(INT)target->id);
995 the_device = pXOpenDevice(data->display, target->id);
999 WARN("Unable to Open device\n");
1003 if (the_device->num_classes > 0)
1005 DeviceKeyPress(the_device, key_press_type, event_list[event_number]);
1006 if (key_press_type) event_number++;
1007 DeviceKeyRelease(the_device, key_release_type, event_list[event_number]);
1008 if (key_release_type) event_number++;
1009 DeviceButtonPress(the_device, button_press_type, event_list[event_number]);
1010 if (button_press_type) event_number++;
1011 DeviceButtonRelease(the_device, button_release_type, event_list[event_number]);
1012 if (button_release_type) event_number++;
1013 DeviceMotionNotify(the_device, motion_type, event_list[event_number]);
1014 if (motion_type) event_number++;
1015 ProximityIn(the_device, proximity_in_type, event_list[event_number]);
1016 if (proximity_in_type) event_number++;
1017 ProximityOut(the_device, proximity_out_type, event_list[event_number]);
1018 if (proximity_out_type) event_number++;
1021 X11DRV_register_event_handler( key_press_type, key_event, "XInput KeyPress" );
1022 if (key_release_type)
1023 X11DRV_register_event_handler( key_release_type, key_event, "XInput KeyRelease" );
1024 if (button_press_type)
1025 X11DRV_register_event_handler( button_press_type, button_event, "XInput ButtonPress" );
1026 if (button_release_type)
1027 X11DRV_register_event_handler( button_release_type, button_event, "XInput ButtonRelease" );
1029 X11DRV_register_event_handler( motion_type, motion_event, "XInput MotionNotify" );
1030 if (proximity_in_type)
1031 X11DRV_register_event_handler( proximity_in_type, proximity_event, "XInput ProximityIn" );
1032 if (proximity_out_type)
1033 X11DRV_register_event_handler( proximity_out_type, proximity_event, "XInput ProximityOut" );
1035 pXSelectExtensionEvent(data->display, win, event_list, event_number);
1038 XSync(data->display, False);
1039 X11DRV_check_error();
1041 if (NULL != devices) pXFreeDeviceList(devices);
1045 /***********************************************************************
1046 * X11DRV_GetCurrentPacket (X11DRV.@)
1048 int CDECL X11DRV_GetCurrentPacket(LPWTPACKET packet)
1050 *packet = gMsgPacket;
1055 static inline int CopyTabletData(LPVOID target, LPCVOID src, INT size)
1058 * It is valid to call CopyTabletData with NULL.
1059 * This handles the WTInfo() case where lpOutput is null.
1062 memcpy(target,src,size);
1066 /***********************************************************************
1067 * X11DRV_WTInfoW (X11DRV.@)
1069 UINT CDECL X11DRV_WTInfoW(UINT wCategory, UINT nIndex, LPVOID lpOutput)
1072 * It is valid to call WTInfoA with lpOutput == NULL, as per standard.
1073 * lpOutput == NULL signifies the user only wishes
1074 * to find the size of the data.
1076 * From now on use CopyTabletData to fill lpOutput. memcpy will break
1080 LPWTI_CURSORS_INFO tgtcursor;
1081 TRACE("(%u, %u, %p)\n", wCategory, nIndex, lpOutput);
1083 if (!xinput_handle) return 0;
1088 /* return largest necessary buffer */
1089 TRACE("%i cursors\n",gNumCursors);
1092 FIXME("Return proper size\n");
1103 static const WCHAR driver[] = {'W','i','n','e',' ','W','i','n','t','a','b',' ','1','.','1',0};
1104 rc = CopyTabletData(lpOutput, driver, (strlenW(driver) + 1) * sizeof(WCHAR));
1107 case IFC_SPECVERSION:
1108 version = (0x01) | (0x01 << 8);
1109 rc = CopyTabletData(lpOutput, &version,sizeof(WORD));
1111 case IFC_IMPLVERSION:
1112 version = (0x00) | (0x01 << 8);
1113 rc = CopyTabletData(lpOutput, &version,sizeof(WORD));
1117 rc = CopyTabletData(lpOutput, &num,sizeof(num));
1121 rc = CopyTabletData(lpOutput, &num,sizeof(num));
1124 FIXME("WTI_INTERFACE unhandled index %i\n",nIndex);
1130 case WTI_DEFCONTEXT:
1134 /* report 0 if wintab is disabled */
1135 if (0 == gNumCursors)
1138 rc = CopyTabletData(lpOutput, &gSysContext,
1139 sizeof(LOGCONTEXTW));
1142 rc = CopyTabletData(lpOutput, gSysContext.lcName,
1143 (strlenW(gSysContext.lcName)+1) * sizeof(WCHAR));
1146 rc = CopyTabletData(lpOutput, &gSysContext.lcOptions,
1150 rc = CopyTabletData(lpOutput, &gSysContext.lcStatus,
1154 rc= CopyTabletData (lpOutput, &gSysContext.lcLocks,
1158 rc = CopyTabletData(lpOutput, &gSysContext.lcMsgBase,
1162 rc = CopyTabletData(lpOutput, &gSysContext.lcDevice,
1166 rc = CopyTabletData(lpOutput, &gSysContext.lcPktRate,
1170 rc = CopyTabletData(lpOutput, &gSysContext.lcPktData,
1174 rc = CopyTabletData(lpOutput, &gSysContext.lcPktMode,
1178 rc = CopyTabletData(lpOutput, &gSysContext.lcMoveMask,
1182 rc = CopyTabletData(lpOutput, &gSysContext.lcBtnDnMask,
1186 rc = CopyTabletData(lpOutput, &gSysContext.lcBtnUpMask,
1190 rc = CopyTabletData(lpOutput, &gSysContext.lcInOrgX,
1194 rc = CopyTabletData(lpOutput, &gSysContext.lcInOrgY,
1198 rc = CopyTabletData(lpOutput, &gSysContext.lcInOrgZ,
1202 rc = CopyTabletData(lpOutput, &gSysContext.lcInExtX,
1206 rc = CopyTabletData(lpOutput, &gSysContext.lcInExtY,
1210 rc = CopyTabletData(lpOutput, &gSysContext.lcInExtZ,
1214 rc = CopyTabletData(lpOutput, &gSysContext.lcOutOrgX,
1218 rc = CopyTabletData(lpOutput, &gSysContext.lcOutOrgY,
1222 rc = CopyTabletData(lpOutput, &gSysContext.lcOutOrgZ,
1226 rc = CopyTabletData(lpOutput, &gSysContext.lcOutExtX,
1230 rc = CopyTabletData(lpOutput, &gSysContext.lcOutExtY,
1234 rc = CopyTabletData(lpOutput, &gSysContext.lcOutExtZ,
1238 rc = CopyTabletData(lpOutput, &gSysContext.lcSensX,
1242 rc = CopyTabletData(lpOutput, &gSysContext.lcSensY,
1246 rc = CopyTabletData(lpOutput, &gSysContext.lcSensZ,
1250 rc = CopyTabletData(lpOutput, &gSysContext.lcSysMode,
1254 rc = CopyTabletData(lpOutput, &gSysContext.lcSysOrgX,
1258 rc = CopyTabletData(lpOutput, &gSysContext.lcSysOrgY,
1262 rc = CopyTabletData(lpOutput, &gSysContext.lcSysExtX,
1266 rc = CopyTabletData(lpOutput, &gSysContext.lcSysExtY,
1270 rc = CopyTabletData(lpOutput, &gSysContext.lcSysSensX,
1274 rc = CopyTabletData(lpOutput, &gSysContext.lcSysSensY,
1278 FIXME("WTI_DEFSYSCTX unhandled index %i\n",nIndex);
1292 case WTI_CURSORS+10:
1293 case WTI_CURSORS+11:
1294 /* CURSORMAX == 12 */
1295 /* FIXME: dynamic cursor support */
1296 /* Apps will poll different slots to detect what cursors are available
1297 * if there isn't a cursor for this slot return 0 */
1298 if (!gSysCursor[wCategory - WTI_CURSORS].ACTIVE)
1302 tgtcursor = &gSysCursor[wCategory - WTI_CURSORS];
1306 rc = CopyTabletData(lpOutput, tgtcursor->NAME,
1307 (strlenW(tgtcursor->NAME)+1) * sizeof(WCHAR));
1310 rc = CopyTabletData(lpOutput,&tgtcursor->ACTIVE,
1314 rc = CopyTabletData(lpOutput,&tgtcursor->PKTDATA,
1318 rc = CopyTabletData(lpOutput,&tgtcursor->BUTTONS,
1321 case CSR_BUTTONBITS:
1322 rc = CopyTabletData(lpOutput,&tgtcursor->BUTTONBITS,
1326 FIXME("Button Names not returned correctly\n");
1327 rc = CopyTabletData(lpOutput,&tgtcursor->BTNNAMES,
1328 tgtcursor->cchBTNNAMES*sizeof(WCHAR));
1331 rc = CopyTabletData(lpOutput,tgtcursor->BUTTONMAP,
1335 rc = CopyTabletData(lpOutput,tgtcursor->SYSBTNMAP,
1338 case CSR_NPBTNMARKS:
1339 rc = CopyTabletData(lpOutput,tgtcursor->NPBTNMARKS,
1343 rc = CopyTabletData(lpOutput,&tgtcursor->NPBUTTON,
1346 case CSR_NPRESPONSE:
1347 FIXME("Not returning CSR_NPRESPONSE correctly\n");
1351 rc = CopyTabletData(lpOutput,&tgtcursor->TPBUTTON,
1354 case CSR_TPBTNMARKS:
1355 rc = CopyTabletData(lpOutput,tgtcursor->TPBTNMARKS,
1358 case CSR_TPRESPONSE:
1359 FIXME("Not returning CSR_TPRESPONSE correctly\n");
1365 id = tgtcursor->PHYSID;
1366 rc = CopyTabletData(lpOutput,&id,sizeof(DWORD));
1370 rc = CopyTabletData(lpOutput,&tgtcursor->MODE,sizeof(UINT));
1372 case CSR_MINPKTDATA:
1373 rc = CopyTabletData(lpOutput,&tgtcursor->MINPKTDATA,
1376 case CSR_MINBUTTONS:
1377 rc = CopyTabletData(lpOutput,&tgtcursor->MINBUTTONS,
1380 case CSR_CAPABILITIES:
1381 rc = CopyTabletData(lpOutput,&tgtcursor->CAPABILITIES,
1385 rc = CopyTabletData(lpOutput,&tgtcursor->TYPE,
1389 FIXME("WTI_CURSORS unhandled index %i\n",nIndex);
1398 rc = CopyTabletData(lpOutput,gSysDevice.NAME,
1399 (strlenW(gSysDevice.NAME)+1) * sizeof(WCHAR));
1402 rc = CopyTabletData(lpOutput,&gSysDevice.HARDWARE,
1406 rc = CopyTabletData(lpOutput,&gSysDevice.NCSRTYPES,
1410 rc = CopyTabletData(lpOutput,&gSysDevice.FIRSTCSR,
1414 rc = CopyTabletData(lpOutput,&gSysDevice.PKTRATE,
1418 rc = CopyTabletData(lpOutput,&gSysDevice.PKTDATA,
1422 rc = CopyTabletData(lpOutput,&gSysDevice.PKTMODE,
1426 rc = CopyTabletData(lpOutput,&gSysDevice.CSRDATA,
1430 rc = CopyTabletData(lpOutput,&gSysDevice.XMARGIN,
1434 rc = CopyTabletData(lpOutput,&gSysDevice.YMARGIN,
1438 rc = 0; /* unsupported */
1440 rc = CopyTabletData(lpOutput,&gSysDevice.ZMARGIN,
1445 rc = CopyTabletData(lpOutput,&gSysDevice.X,
1449 rc = CopyTabletData(lpOutput,&gSysDevice.Y,
1453 rc = 0; /* unsupported */
1455 rc = CopyTabletData(lpOutput,&gSysDevice.Z,
1460 rc = CopyTabletData(lpOutput,&gSysDevice.NPRESSURE,
1464 rc = 0; /* unsupported */
1466 rc = CopyTabletData(lpOutput,&gSysDevice.TPRESSURE,
1470 case DVC_ORIENTATION:
1471 rc = CopyTabletData(lpOutput,gSysDevice.ORIENTATION,
1475 rc = 0; /* unsupported */
1477 rc = CopyTabletData(lpOutput,&gSysDevice.ROTATION,
1482 rc = CopyTabletData(lpOutput,gSysDevice.PNPID,
1483 (strlenW(gSysDevice.PNPID)+1)*sizeof(WCHAR));
1486 FIXME("WTI_DEVICES unhandled index %i\n",nIndex);
1491 FIXME("Unhandled Category %i\n",wCategory);
1496 #else /* SONAME_LIBXI */
1498 /***********************************************************************
1499 * AttachEventQueueToTablet (X11DRV.@)
1501 int CDECL X11DRV_AttachEventQueueToTablet(HWND hOwner)
1506 /***********************************************************************
1507 * GetCurrentPacket (X11DRV.@)
1509 int CDECL X11DRV_GetCurrentPacket(LPWTPACKET packet)
1514 /***********************************************************************
1515 * LoadTabletInfo (X11DRV.@)
1517 BOOL CDECL X11DRV_LoadTabletInfo(HWND hwnddefault)
1522 /***********************************************************************
1523 * WTInfoW (X11DRV.@)
1525 UINT CDECL X11DRV_WTInfoW(UINT wCategory, UINT nIndex, LPVOID lpOutput)
1530 #endif /* SONAME_LIBXI */