X-Git-Url: https://diplodocus.org/git/xorg-xinput/blobdiff_plain/d02601e5c88d1d40e12cd71c2c10c7822919f7b8..refs/heads/master:/src/list.c?ds=sidebyside diff --git a/src/list.c b/src/list.c index e288642..38bc777 100644 --- a/src/list.c +++ b/src/list.c @@ -5,15 +5,15 @@ * documentation for any purpose is hereby granted without fee, provided that * the above copyright notice appear in all copies and that both that * copyright notice and this permission notice appear in supporting - * documentation, and that the name of Frederic Lepied not be used in + * documentation, and that the name of the authors not be used in * advertising or publicity pertaining to distribution of the software without - * specific, written prior permission. Frederic Lepied makes no + * specific, written prior permission. The authors make no * representations about the suitability of this software for any purpose. It * is provided "as is" without express or implied warranty. * - * FREDERIC LEPIED DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, + * THE AUTHORS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO - * EVENT SHALL FREDERIC LEPIED BE LIABLE FOR ANY SPECIAL, INDIRECT OR + * EVENT SHALL THE AUTHORS BE LIABLE FOR ANY SPECIAL, INDIRECT OR * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR @@ -23,10 +23,18 @@ #include "xinput.h" #include -#include /* for XI_Device***ChangedNotify */ + +enum print_format { + FORMAT_NONE, + FORMAT_SHORT, + FORMAT_LONG, + FORMAT_NAME, + FORMAT_ID, +}; + static void -print_info(XDeviceInfo *info, Bool shortformat) +print_info(Display* dpy, XDeviceInfo *info, enum print_format format) { int i,j; XAnyClassPtr any; @@ -34,7 +42,16 @@ print_info(XDeviceInfo *info, Bool shortformat) XButtonInfoPtr b; XValuatorInfoPtr v; XAxisInfoPtr a; - XAttachInfoPtr att; + + if (format == FORMAT_NAME) + { + printf("%s\n", info->name); + return; + } else if (format == FORMAT_ID) + { + printf("%ld\n", info->id); + return; + } printf("\"%s\"\tid=%ld\t[", info->name, info->id); @@ -60,9 +77,15 @@ print_info(XDeviceInfo *info, Bool shortformat) } printf("]\n"); - if (shortformat) + if (format == FORMAT_SHORT) return; + if (info->type != None) { + char *type = XGetAtomName(dpy, info->type); + printf("\tType is %s\n", type); + XFree(type); + } + if (info->num_classes > 0) { any = (XAnyClassPtr) (info->inputclassinfo); for (i=0; inum_classes; i++) { @@ -93,11 +116,6 @@ print_info(XDeviceInfo *info, Bool shortformat) printf ("\t\tResolution is %d\n", a->resolution); } break; - case AttachClass: - att = (XAttachInfoPtr)any; - printf("\tAttached to %d\n", att->attached); - break; - default: printf ("unknown class\n"); } @@ -106,75 +124,287 @@ print_info(XDeviceInfo *info, Bool shortformat) } } -int -list(Display *display, - int argc, - char *argv[], - char *name, - char *desc) +static int list_xi1(Display *display, + enum print_format format) { XDeviceInfo *info; int loop; - int shortformat = False; - int daemon = False; + int num_devices; - shortformat = (argc == 1 && strcmp(argv[0], "--short") == 0); - daemon = (argc == 1 && strcmp(argv[0], "--loop") == 0); + info = XListInputDevices(display, &num_devices); + for(loop=0; loopsourceid); + switch(classes[i]->type) { - XiSelectEvent(display, DefaultRootWindow(display), - XI_DeviceHierarchyChangedMask | - XI_DeviceClassesChangedMask); + case XIButtonClass: + { + XIButtonClassInfo *b = (XIButtonClassInfo*)classes[i]; + char *name; + printf("XIButtonClass\n"); + printf("\t\tButtons supported: %d\n", b->num_buttons); + printf("\t\tButton labels:"); + for (j = 0; j < b->num_buttons; j++) + { + name = (b->labels[j]) ? XGetAtomName(display, b->labels[j]) : NULL; + if (name) + printf(" \"%s\"", name); + else + printf(" None"); + XFree(name); + } + printf("\n"); + printf("\t\tButton state:"); + for (j = 0; j < b->state.mask_len * 8; j++) + if (XIMaskIsSet(b->state.mask, j)) + printf(" %d", j); + printf("\n"); + + } + break; + case XIKeyClass: + { + XIKeyClassInfo *k = (XIKeyClassInfo*)classes[i]; + printf("XIKeyClass\n"); + printf("\t\tKeycodes supported: %d\n", k->num_keycodes); + } + break; + case XIValuatorClass: + { + XIValuatorClassInfo *v = (XIValuatorClassInfo*)classes[i]; + char *name = v->label ? XGetAtomName(display, v->label) : NULL; + + /* Bug in X servers 1.7..1.8.1, mode was | OutOfProximity */ + v->mode &= DeviceMode; + + printf("XIValuatorClass\n"); + printf("\t\tDetail for Valuator %d:\n", v->number); + printf("\t\t Label: %s\n", (name) ? name : "None"); + printf("\t\t Range: %f - %f\n", v->min, v->max); + printf("\t\t Resolution: %d units/m\n", v->resolution); + printf("\t\t Mode: %s\n", v->mode == Absolute ? "absolute" : + "relative"); + if (v->mode == Absolute) + printf("\t\t Current value: %f\n", v->value); + XFree(name); + } + break; +#if HAVE_XI21 + case XIScrollClass: + { + XIScrollClassInfo *s = (XIScrollClassInfo*)classes[i]; + + printf("XIScrollClass\n"); + printf("\t\tScroll info for Valuator %d\n", s->number); + printf("\t\t type: %d (%s)\n", s->scroll_type, + (s->scroll_type == XIScrollTypeHorizontal) ? "horizontal" : + (s->scroll_type == XIScrollTypeVertical) ? "vertical" : "unknown"); + printf("\t\t increment: %f\n", s->increment); + printf("\t\t flags: 0x%x", s->flags); + if (s->flags) { + printf(" ("); + if (s->flags & XIScrollFlagNoEmulation) + printf(" no-emulation "); + if (s->flags & XIScrollFlagPreferred) + printf(" preferred "); + printf(")"); + } + printf("\n"); + } + break; +#endif +#if HAVE_XI22 + case XITouchClass: + { + XITouchClassInfo *t = (XITouchClassInfo*)classes[i]; + + printf("XITouchClass\n"); + printf("\t\tTouch mode: %s\n", + (t->mode == XIDirectTouch) ? "direct" : "dependent"); + printf("\t\tMax number of touches: %d\n", t->num_touches); + } +#endif } + } + + printf("\n"); +} - do { - info = XListInputDevices(display, &num_devices); - for(loop=0; loopname); + return; + } else if (format == FORMAT_ID) + { + printf("%d\n", dev->deviceid); + return; + } + + printf("%-40s\tid=%d\t[", dev->name, dev->deviceid); + switch(dev->use) + { + case XIMasterPointer: + printf("master pointer (%d)]\n", dev->attachment); + break; + case XIMasterKeyboard: + printf("master keyboard (%d)]\n", dev->attachment); + break; + case XISlavePointer: + printf("slave pointer (%d)]\n", dev->attachment); + break; + case XISlaveKeyboard: + printf("slave keyboard (%d)]\n", dev->attachment); + break; + case XIFloatingSlave: + printf("floating slave]\n"); + break; + } + + if (format == FORMAT_SHORT) + return; + + if (!dev->enabled) + printf("\tThis device is disabled\n"); + + print_classes_xi2(display, dev->classes, dev->num_classes); +} + + +static int +list_xi2(Display *display, + enum print_format format) +{ + int ndevices; + int i, j; + XIDeviceInfo *info, *dev; + + info = XIQueryDevice(display, XIAllDevices, &ndevices); + + for(i = 0; i < ndevices; i++) + { + dev = &info[i]; + if (dev->use == XIMasterPointer || dev->use == XIMasterKeyboard) + { + if (format == FORMAT_SHORT || format == FORMAT_LONG) + { + if (dev->use == XIMasterPointer) + printf("⎡ "); + else + printf("⎣ "); } - /* just wait for the next generic event to come along */ - while (daemon && !XNextEvent(display, &ev)) + print_info_xi2(display, dev, format); + for (j = 0; j < ndevices; j++) { - if (ev.type == GenericEvent) + XIDeviceInfo* sd = &info[j]; + + if ((sd->use == XISlavePointer || sd->use == XISlaveKeyboard) && + (sd->attachment == dev->deviceid)) { - XGenericEvent* gev = (XGenericEvent*)&ev; - /* we just assume that extension is IReqCode, pretty save - since we don't register for other events. */ - if (gev->evtype == XI_DeviceHierarchyChangedNotify) - { - printf("Hierarchy change.\n"); - } else if (gev->evtype == XI_DeviceClassesChangedNotify) - { - printf("Device classes changed.\n"); - free(((XDeviceClassesChangedEvent*)&ev)->inputclassinfo); - } - break; + if (format == FORMAT_SHORT || format == FORMAT_LONG) + printf("%s ↳ ", dev->use == XIMasterPointer ? "⎜" : " "); + print_info_xi2(display, sd, format); } } - } while(daemon); - } else { - int ret = EXIT_SUCCESS; - - for(loop=0; loopuse == XIFloatingSlave) + { + printf("∼ "); + print_info_xi2(display, dev, format); + } } + + + XIFreeDeviceInfo(info); return EXIT_SUCCESS; } +#endif + +int +list(Display *display, + int argc, + char *argv[], + char *name, + char *desc) +{ + enum print_format format = FORMAT_NONE; + int arg_dev = 1; + + if (argc >= 1) + { + if (strcmp(argv[0], "--short") == 0) + format = FORMAT_SHORT; + else if (strcmp(argv[0], "--long") == 0) + format = FORMAT_LONG; + else if (strcmp(argv[0], "--name-only") == 0) + format = FORMAT_NAME; + else if (strcmp(argv[0], "--id-only") == 0) + format = FORMAT_ID; + else + arg_dev--; + } + + if (argc > arg_dev) + { + if (format == FORMAT_NONE) + format = FORMAT_LONG; +#if HAVE_XI2 + if (xinput_version(display) == XI_2_Major) + { + XIDeviceInfo *info = xi2_find_device_info(display, argv[arg_dev]); + + if (!info) { + fprintf(stderr, "unable to find device %s\n", argv[arg_dev]); + return EXIT_FAILURE; + } else { + print_info_xi2(display, info, format); + return EXIT_SUCCESS; + } + } else +#endif + { + XDeviceInfo *info = find_device_info(display, argv[arg_dev], False); + + if (!info) { + fprintf(stderr, "unable to find device %s\n", argv[arg_dev]); + return EXIT_FAILURE; + } else { + print_info(display, info, format); + return EXIT_SUCCESS; + } + } + } else { + if (format == FORMAT_NONE) + format = FORMAT_SHORT; +#if HAVE_XI2 + if (xinput_version(display) == XI_2_Major) + return list_xi2(display, format); +#endif + return list_xi1(display, format); + } +} /* end of list.c */