/* * Copyright (C) 2010 Andriy Rysin (rysin@kde.org) * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "xinput_helper.h" #include #include #include #include #include #include #ifdef HAVE_XINPUT_AND_DEVICE_NOTIFY #include #endif #include #include "x11_helper.h" #include static int DEVICE_NONE = 0; static int DEVICE_KEYBOARD = 1; static int DEVICE_POINTER = 2; XInputEventNotifier::XInputEventNotifier(QWidget* parent): XEventNotifier(parent), xinputEventType(-1) { } void XInputEventNotifier::start() { if( KApplication::kApplication() != NULL ) { registerForNewDeviceEvent(QX11Info::display()); } XEventNotifier::start(); } void XInputEventNotifier::stop() { XEventNotifier::stop(); if( KApplication::kApplication() != NULL ) { // XEventNotifier::unregisterForNewDeviceEvent(QX11Info::display()); } } bool XInputEventNotifier::processOtherEvents(XEvent* event) { int newDeviceType = getNewDeviceEventType(event); if( newDeviceType == DEVICE_KEYBOARD ) { emit(newKeyboardDevice()); } else if( newDeviceType == DEVICE_POINTER ) { emit(newPointerDevice()); emit(newKeyboardDevice()); // arghhh, looks like X resets xkb map even when only pointer device is connected } return true; } #ifdef HAVE_XINPUT_AND_DEVICE_NOTIFY extern "C" { extern int _XiGetDevicePresenceNotifyEvent(Display *); } // This is ugly but allows to skip multiple execution of setxkbmap // for all keyboard devices that don't care about layouts static bool isRealKeyboard(const char* deviceName) { return strstr(deviceName, "Video Bus") == NULL && strstr(deviceName, "Sleep Button") == NULL && strstr(deviceName, "Power Button") == NULL && strstr(deviceName, "WMI hotkeys") == NULL; } int XInputEventNotifier::getNewDeviceEventType(XEvent* event) { int newDeviceType = DEVICE_NONE; if( xinputEventType != -1 && event->type == xinputEventType ) { XDevicePresenceNotifyEvent *xdpne = (XDevicePresenceNotifyEvent*) event; if( xdpne->devchange == DeviceEnabled ) { int ndevices; XDeviceInfo *devices = XListInputDevices(xdpne->display, &ndevices); if( devices != NULL ) { // kDebug() << "New device id:" << xdpne->deviceid; for(int i=0; ideviceid ) { if( devices[i].use == IsXKeyboard || devices[i].use == IsXExtensionKeyboard ) { if( isRealKeyboard(devices[i].name) ) { newDeviceType = DEVICE_KEYBOARD; kDebug() << "new keyboard device, id:" << devices[i].id << "name:" << devices[i].name << "used as:" << devices[i].use; break; } } if( devices[i].use == IsXPointer || devices[i].use == IsXExtensionPointer ) { newDeviceType = DEVICE_POINTER; kDebug() << "new pointer device, id:" << devices[i].id << "name:" << devices[i].name << "used as:" << devices[i].use; break; } } } XFreeDeviceList(devices); } } } return newDeviceType; } int XInputEventNotifier::registerForNewDeviceEvent(Display* display) { int xitype; XEventClass xiclass; DevicePresence(display, xitype, xiclass); XSelectExtensionEvent(display, DefaultRootWindow(display), &xiclass, 1); kDebug() << "Registered for new device events from XInput, class" << xitype; xinputEventType = xitype; return xitype; } #else #warning "Keyboard daemon is compiled without XInput, keyboard settings will be reset when new keyboard device is plugged in!" int XInputEventNotifier::registerForNewDeviceEvent(Display* /*display*/) { kWarning() << "Keyboard kded daemon is compiled without XInput, xkb configuration will be reset when new keyboard device is plugged in!"; return -1; } int XInputEventNotifier::getNewDeviceEventType(XEvent* /*event*/) { return DEVICE_NONE; } #endif