/* SPDX-FileCopyrightText: 2014 Martin Gräßlin SPDX-FileCopyrightText: 2020 David Edmundson SPDX-FileCopyrightText: 2021 Vlad Zahorodnii SPDX-License-Identifier: LGPL-2.1-only OR LGPL-3.0-only OR LicenseRef-KDE-Accepted-LGPL */ #include "seat.h" #include "abstract_data_source.h" #include "datacontroldevice_v1.h" #include "datacontrolsource_v1.h" #include "datadevice.h" #include "datadevice_p.h" #include "datasource.h" #include "display.h" #include "display_p.h" #include "keyboard.h" #include "keyboard_p.h" #include "pointer.h" #include "pointer_p.h" #include "pointerconstraints_v1.h" #include "pointergestures_v1_p.h" #include "primaryselectiondevice_v1.h" #include "primaryselectionsource_v1.h" #include "relativepointer_v1_p.h" #include "seat_p.h" #include "surface.h" #include "textinput_v1_p.h" #include "textinput_v2_p.h" #include "textinput_v3_p.h" #include "touch_p.h" #include "utils/common.h" #include "utils/resource.h" #include "xdgtopleveldrag_v1.h" #include #include namespace KWin { static const int s_version = 9; SeatInterfacePrivate *SeatInterfacePrivate::get(SeatInterface *seat) { return seat->d.get(); } SeatInterfacePrivate::SeatInterfacePrivate(SeatInterface *q, Display *display) : QtWaylandServer::wl_seat(*display, s_version) , q(q) , display(display) { textInputV1 = new TextInputV1Interface(q); textInputV2 = new TextInputV2Interface(q); textInputV3 = new TextInputV3Interface(q); pointer.reset(new PointerInterface(q)); keyboard.reset(new KeyboardInterface(q)); touch.reset(new TouchInterface(q)); } void SeatInterfacePrivate::seat_bind_resource(Resource *resource) { send_capabilities(resource->handle, capabilities); if (resource->version() >= WL_SEAT_NAME_SINCE_VERSION) { send_name(resource->handle, name); } } void SeatInterfacePrivate::seat_get_pointer(Resource *resource, uint32_t id) { PointerInterfacePrivate *pointerPrivate = PointerInterfacePrivate::get(pointer.get()); pointerPrivate->add(resource->client(), id, resource->version()); } void SeatInterfacePrivate::seat_get_keyboard(Resource *resource, uint32_t id) { KeyboardInterfacePrivate *keyboardPrivate = KeyboardInterfacePrivate::get(keyboard.get()); keyboardPrivate->add(resource->client(), id, resource->version()); } void SeatInterfacePrivate::seat_get_touch(Resource *resource, uint32_t id) { TouchInterfacePrivate *touchPrivate = TouchInterfacePrivate::get(touch.get()); touchPrivate->add(resource->client(), id, resource->version()); } void SeatInterfacePrivate::seat_release(Resource *resource) { wl_resource_destroy(resource->handle); } SeatInterface::SeatInterface(Display *display, QObject *parent) : QObject(parent) , d(new SeatInterfacePrivate(this, display)) { DisplayPrivate *displayPrivate = DisplayPrivate::get(d->display); displayPrivate->seats.append(this); } SeatInterface::~SeatInterface() { if (d->display) { DisplayPrivate *displayPrivate = DisplayPrivate::get(d->display); displayPrivate->seats.removeOne(this); } } void SeatInterfacePrivate::updatePointerButtonSerial(quint32 button, quint32 serial) { auto it = globalPointer.buttonSerials.find(button); if (it == globalPointer.buttonSerials.end()) { globalPointer.buttonSerials.insert(button, serial); return; } it.value() = serial; } void SeatInterfacePrivate::updatePointerButtonState(quint32 button, Pointer::State state) { auto it = globalPointer.buttonStates.find(button); if (it == globalPointer.buttonStates.end()) { globalPointer.buttonStates.insert(button, state); return; } it.value() = state; } QList SeatInterfacePrivate::dataDevicesForSurface(SurfaceInterface *surface) const { if (!surface) { return {}; } QList primarySelectionDevices; for (auto it = dataDevices.constBegin(); it != dataDevices.constEnd(); ++it) { if ((*it)->client() == *surface->client()) { primarySelectionDevices << *it; } } return primarySelectionDevices; } void SeatInterfacePrivate::registerDataDevice(DataDeviceInterface *dataDevice) { Q_ASSERT(dataDevice->seat() == q); dataDevices << dataDevice; auto dataDeviceCleanup = [this, dataDevice] { dataDevices.removeOne(dataDevice); globalKeyboard.focus.selections.removeOne(dataDevice); }; QObject::connect(dataDevice, &QObject::destroyed, q, dataDeviceCleanup); QObject::connect(dataDevice, &DataDeviceInterface::selectionChanged, q, [this, dataDevice] { updateSelection(dataDevice); }); QObject::connect(dataDevice, &DataDeviceInterface::dragStarted, q, [this](AbstractDataSource *source, SurfaceInterface *origin, quint32 serial, DragAndDropIcon *dragIcon) { q->startDrag(source, origin, serial, dragIcon); }); // is the new DataDevice for the current keyoard focus? if (globalKeyboard.focus.surface) { // same client? if (*globalKeyboard.focus.surface->client() == dataDevice->client()) { globalKeyboard.focus.selections.append(dataDevice); if (currentSelection) { dataDevice->sendSelection(currentSelection); } } } } AbstractDropHandler *SeatInterface::dropHandlerForSurface(SurfaceInterface *surface) const { auto list = d->dataDevicesForSurface(surface); if (list.isEmpty()) { return nullptr; }; return list.first(); } void SeatInterface::cancelDrag() { if (d->drag.mode != SeatInterfacePrivate::Drag::Mode::None) { // cancel the drag, don't drop. serial does not matter d->cancelDrag(); } } void SeatInterfacePrivate::registerDataControlDevice(DataControlDeviceV1Interface *dataDevice) { Q_ASSERT(dataDevice->seat() == q); dataControlDevices << dataDevice; auto dataDeviceCleanup = [this, dataDevice] { dataControlDevices.removeOne(dataDevice); }; QObject::connect(dataDevice, &QObject::destroyed, q, dataDeviceCleanup); QObject::connect(dataDevice, &DataControlDeviceV1Interface::selectionChanged, q, [this, dataDevice] { // Special klipper workaround to avoid a race // If the mimetype x-kde-onlyReplaceEmpty is set, and we've had another update in the meantime, do nothing // but resend selection to mimic normal event flow upon cancel and not confuse the client // See https://github.com/swaywm/wlr-protocols/issues/92 if (dataDevice->selection() && dataDevice->selection()->mimeTypes().contains(QLatin1String("application/x-kde-onlyReplaceEmpty")) && currentSelection) { dataDevice->selection()->cancel(); dataDevice->sendSelection(currentSelection); return; } q->setSelection(dataDevice->selection()); }); QObject::connect(dataDevice, &DataControlDeviceV1Interface::primarySelectionChanged, q, [this, dataDevice] { // Special klipper workaround to avoid a race // If the mimetype x-kde-onlyReplaceEmpty is set, and we've had another update in the meantime, do nothing // but resend selection to mimic normal event flow upon cancel and not confuse the client // See https://github.com/swaywm/wlr-protocols/issues/92 if (dataDevice->primarySelection() && dataDevice->primarySelection()->mimeTypes().contains(QLatin1String("application/x-kde-onlyReplaceEmpty")) && currentPrimarySelection) { dataDevice->primarySelection()->cancel(); dataDevice->sendPrimarySelection(currentPrimarySelection); return; } q->setPrimarySelection(dataDevice->primarySelection()); }); dataDevice->sendSelection(currentSelection); dataDevice->sendPrimarySelection(currentPrimarySelection); } void SeatInterfacePrivate::registerPrimarySelectionDevice(PrimarySelectionDeviceV1Interface *primarySelectionDevice) { Q_ASSERT(primarySelectionDevice->seat() == q); primarySelectionDevices << primarySelectionDevice; auto dataDeviceCleanup = [this, primarySelectionDevice] { primarySelectionDevices.removeOne(primarySelectionDevice); globalKeyboard.focus.primarySelections.removeOne(primarySelectionDevice); }; QObject::connect(primarySelectionDevice, &QObject::destroyed, q, dataDeviceCleanup); QObject::connect(primarySelectionDevice, &PrimarySelectionDeviceV1Interface::selectionChanged, q, [this, primarySelectionDevice] { updatePrimarySelection(primarySelectionDevice); }); // is the new DataDevice for the current keyoard focus? if (globalKeyboard.focus.surface) { // same client? if (*globalKeyboard.focus.surface->client() == primarySelectionDevice->client()) { globalKeyboard.focus.primarySelections.append(primarySelectionDevice); if (currentPrimarySelection) { primarySelectionDevice->sendSelection(currentPrimarySelection); } } } } void SeatInterfacePrivate::cancelDrag() { if (drag.target) { drag.target->updateDragTarget(nullptr, 0); drag.target = nullptr; } QObject::disconnect(drag.dragSourceDestroyConnection); if (drag.source) { drag.source->dndCancelled(); } drag = Drag(); Q_EMIT q->dragSurfaceChanged(); Q_EMIT q->dragEnded(); } void SeatInterfacePrivate::endDrag() { QObject::disconnect(drag.dragSourceDestroyConnection); AbstractDropHandler *dragTargetDevice = drag.target.data(); AbstractDataSource *dragSource = drag.source; if (dragSource) { // TODO: Also check the current drag-and-drop action. if (dragTargetDevice && dragSource->isAccepted()) { Q_EMIT q->dragDropped(); dragTargetDevice->drop(); dragSource->dropPerformed(); } else { dragSource->dropPerformed(); dragSource->dndCancelled(); } } if (dragTargetDevice) { dragTargetDevice->updateDragTarget(nullptr, 0); } drag = Drag(); Q_EMIT q->dragSurfaceChanged(); Q_EMIT q->dragEnded(); } void SeatInterfacePrivate::updateSelection(DataDeviceInterface *dataDevice) { DataSourceInterface *selection = dataDevice->selection(); // if the update is from the focussed window we should inform the active client if (!(globalKeyboard.focus.surface && (*globalKeyboard.focus.surface->client() == dataDevice->client()))) { if (selection) { selection->cancel(); } return; } q->setSelection(selection); } void SeatInterfacePrivate::updatePrimarySelection(PrimarySelectionDeviceV1Interface *primarySelectionDevice) { PrimarySelectionSourceV1Interface *selection = primarySelectionDevice->selection(); // if the update is from the focussed window we should inform the active client if (!(globalKeyboard.focus.surface && (*globalKeyboard.focus.surface->client() == primarySelectionDevice->client()))) { if (selection) { selection->cancel(); } return; } q->setPrimarySelection(selection); } void SeatInterfacePrivate::sendCapabilities() { const auto seatResources = resourceMap(); for (SeatInterfacePrivate::Resource *resource : seatResources) { send_capabilities(resource->handle, capabilities); } } void SeatInterface::setHasKeyboard(bool has) { if (hasKeyboard() == has) { return; } if (has) { d->capabilities |= SeatInterfacePrivate::capability_keyboard; } else { d->capabilities &= ~SeatInterfacePrivate::capability_keyboard; } d->sendCapabilities(); Q_EMIT hasKeyboardChanged(has); } void SeatInterface::setHasPointer(bool has) { if (hasPointer() == has) { return; } if (has) { d->capabilities |= SeatInterfacePrivate::capability_pointer; } else { d->capabilities &= ~SeatInterfacePrivate::capability_pointer; } d->sendCapabilities(); Q_EMIT hasPointerChanged(has); } void SeatInterface::setHasTouch(bool has) { if (hasTouch() == has) { return; } if (has) { d->capabilities |= SeatInterfacePrivate::capability_touch; } else { d->capabilities &= ~SeatInterfacePrivate::capability_touch; } d->sendCapabilities(); Q_EMIT hasTouchChanged(has); } void SeatInterface::setName(const QString &name) { if (d->name == name) { return; } d->name = name; const auto seatResources = d->resourceMap(); for (SeatInterfacePrivate::Resource *resource : seatResources) { if (resource->version() >= WL_SEAT_NAME_SINCE_VERSION) { d->send_name(resource->handle, d->name); } } Q_EMIT nameChanged(d->name); } QString SeatInterface::name() const { return d->name; } bool SeatInterface::hasPointer() const { return d->capabilities & SeatInterfacePrivate::capability_pointer; } bool SeatInterface::hasKeyboard() const { return d->capabilities & SeatInterfacePrivate::capability_keyboard; } bool SeatInterface::hasTouch() const { return d->capabilities & SeatInterfacePrivate::capability_touch; } Display *SeatInterface::display() const { return d->display; } SeatInterface *SeatInterface::get(wl_resource *native) { if (SeatInterfacePrivate *seatPrivate = resource_cast(native)) { return seatPrivate->q; } return nullptr; } QPointF SeatInterface::pointerPos() const { return d->globalPointer.pos; } void SeatInterface::notifyPointerMotion(const QPointF &pos) { if (!d->pointer) { return; } if (d->globalPointer.pos == pos) { return; } d->globalPointer.pos = pos; Q_EMIT pointerPosChanged(pos); SurfaceInterface *focusedSurface = focusedPointerSurface(); if (!focusedSurface) { return; } if (isDragPointer()) { // data device will handle it directly // for xwayland cases we still want to send pointer events if (!d->dataDevicesForSurface(focusedSurface).isEmpty()) return; } if (focusedSurface->lockedPointer() && focusedSurface->lockedPointer()->isLocked()) { return; } QPointF localPosition = focusedPointerSurfaceTransformation().map(pos); SurfaceInterface *effectiveFocusedSurface = focusedSurface->inputSurfaceAt(localPosition); if (!effectiveFocusedSurface) { effectiveFocusedSurface = focusedSurface; } if (focusedSurface != effectiveFocusedSurface) { localPosition = focusedSurface->mapToChild(effectiveFocusedSurface, localPosition); } if (d->pointer->focusedSurface() != effectiveFocusedSurface) { d->pointer->sendEnter(effectiveFocusedSurface, localPosition, display()->nextSerial()); if (d->keyboard) { d->keyboard->setModifierFocusSurface(effectiveFocusedSurface); } } d->pointer->sendMotion(localPosition); } std::chrono::milliseconds SeatInterface::timestamp() const { return d->timestamp; } void SeatInterface::setTimestamp(std::chrono::microseconds time) { const auto milliseconds = std::chrono::duration_cast(time); if (d->timestamp == milliseconds) { return; } d->timestamp = milliseconds; Q_EMIT timestampChanged(); } void SeatInterface::setDragTarget(AbstractDropHandler *dropTarget, SurfaceInterface *surface, const QPointF &globalPosition, const QMatrix4x4 &inputTransformation) { if (surface == d->drag.surface) { // no change return; } const quint32 serial = d->display->nextSerial(); if (d->drag.target) { d->drag.target->updateDragTarget(nullptr, serial); } // TODO: technically we can have mulitple data devices // and we should send the drag to all of them, but that seems overly complicated // in practice so far the only case for mulitple data devices is for clipboard overriding d->drag.target = dropTarget; if (d->drag.mode == SeatInterfacePrivate::Drag::Mode::Pointer) { notifyPointerMotion(globalPosition); notifyPointerFrame(); } else if (d->drag.mode == SeatInterfacePrivate::Drag::Mode::Touch && d->globalTouch.focus.firstTouchPos != globalPosition) { notifyTouchMotion(d->globalTouch.ids.first(), globalPosition); } if (d->drag.target) { QMatrix4x4 surfaceInputTransformation = inputTransformation; surfaceInputTransformation.scale(surface->scaleOverride()); d->drag.surface = surface; d->drag.transformation = surfaceInputTransformation; d->drag.target->updateDragTarget(surface, serial); } else { d->drag.surface = nullptr; } Q_EMIT dragSurfaceChanged(); return; } void SeatInterface::setDragTarget(AbstractDropHandler *target, SurfaceInterface *surface, const QMatrix4x4 &inputTransformation) { if (d->drag.mode == SeatInterfacePrivate::Drag::Mode::Pointer) { setDragTarget(target, surface, pointerPos(), inputTransformation); } else { Q_ASSERT(d->drag.mode == SeatInterfacePrivate::Drag::Mode::Touch); setDragTarget(target, surface, d->globalTouch.focus.firstTouchPos, inputTransformation); } } SurfaceInterface *SeatInterface::focusedPointerSurface() const { return d->globalPointer.focus.surface; } void SeatInterface::notifyPointerEnter(SurfaceInterface *surface, const QPointF &position, const QPointF &surfacePosition) { QMatrix4x4 m; m.translate(-surfacePosition.x(), -surfacePosition.y()); notifyPointerEnter(surface, position, m); if (d->globalPointer.focus.surface) { d->globalPointer.focus.offset = surfacePosition; } } void SeatInterface::notifyPointerEnter(SurfaceInterface *surface, const QPointF &position, const QMatrix4x4 &transformation) { if (!d->pointer) { return; } if (d->drag.mode == SeatInterfacePrivate::Drag::Mode::Pointer) { // ignore return; } const quint32 serial = d->display->nextSerial(); if (d->globalPointer.focus.surface) { disconnect(d->globalPointer.focus.destroyConnection); } d->globalPointer.focus = SeatInterfacePrivate::Pointer::Focus(); d->globalPointer.focus.surface = surface; d->globalPointer.focus.destroyConnection = connect(surface, &QObject::destroyed, this, [this] { d->globalPointer.focus = SeatInterfacePrivate::Pointer::Focus(); }); d->globalPointer.focus.serial = serial; d->globalPointer.focus.transformation = transformation; d->globalPointer.focus.offset = QPointF(); d->globalPointer.pos = position; QPointF localPosition = focusedPointerSurfaceTransformation().map(position); SurfaceInterface *effectiveFocusedSurface = surface->inputSurfaceAt(localPosition); if (!effectiveFocusedSurface) { effectiveFocusedSurface = surface; } if (surface != effectiveFocusedSurface) { localPosition = surface->mapToChild(effectiveFocusedSurface, localPosition); } d->pointer->sendEnter(effectiveFocusedSurface, localPosition, serial); if (d->keyboard) { d->keyboard->setModifierFocusSurface(effectiveFocusedSurface); } } void SeatInterface::notifyPointerLeave() { if (!d->pointer) { return; } if (d->drag.mode == SeatInterfacePrivate::Drag::Mode::Pointer) { // ignore return; } if (d->globalPointer.focus.surface) { disconnect(d->globalPointer.focus.destroyConnection); } d->globalPointer.focus = SeatInterfacePrivate::Pointer::Focus(); const quint32 serial = d->display->nextSerial(); d->pointer->sendLeave(serial); if (d->keyboard) { d->keyboard->setModifierFocusSurface(nullptr); } } void SeatInterface::setFocusedPointerSurfacePosition(const QPointF &surfacePosition) { if (d->globalPointer.focus.surface) { d->globalPointer.focus.offset = surfacePosition; d->globalPointer.focus.transformation = QMatrix4x4(); d->globalPointer.focus.transformation.translate(-surfacePosition.x(), -surfacePosition.y()); } } QPointF SeatInterface::focusedPointerSurfacePosition() const { return d->globalPointer.focus.offset; } void SeatInterface::setFocusedPointerSurfaceTransformation(const QMatrix4x4 &transformation) { if (d->globalPointer.focus.surface) { d->globalPointer.focus.transformation = transformation; } } QMatrix4x4 SeatInterface::focusedPointerSurfaceTransformation() const { return d->globalPointer.focus.transformation; } PointerInterface *SeatInterface::pointer() const { return d->pointer.get(); } static quint32 qtToWaylandButton(Qt::MouseButton button) { static const QHash s_buttons({ {Qt::LeftButton, BTN_LEFT}, {Qt::RightButton, BTN_RIGHT}, {Qt::MiddleButton, BTN_MIDDLE}, {Qt::ExtraButton1, BTN_BACK}, // note: QtWayland maps BTN_SIDE {Qt::ExtraButton2, BTN_FORWARD}, // note: QtWayland maps BTN_EXTRA {Qt::ExtraButton3, BTN_TASK}, // note: QtWayland maps BTN_FORWARD {Qt::ExtraButton4, BTN_EXTRA}, // note: QtWayland maps BTN_BACK {Qt::ExtraButton5, BTN_SIDE}, // note: QtWayland maps BTN_TASK {Qt::ExtraButton6, BTN_TASK + 1}, {Qt::ExtraButton7, BTN_TASK + 2}, {Qt::ExtraButton8, BTN_TASK + 3}, {Qt::ExtraButton9, BTN_TASK + 4}, {Qt::ExtraButton10, BTN_TASK + 5}, {Qt::ExtraButton11, BTN_TASK + 6}, {Qt::ExtraButton12, BTN_TASK + 7}, {Qt::ExtraButton13, BTN_TASK + 8} // further mapping not possible, 0x120 is BTN_JOYSTICK }); return s_buttons.value(button, 0); } bool SeatInterface::isPointerButtonPressed(Qt::MouseButton button) const { return isPointerButtonPressed(qtToWaylandButton(button)); } bool SeatInterface::isPointerButtonPressed(quint32 button) const { auto it = d->globalPointer.buttonStates.constFind(button); if (it == d->globalPointer.buttonStates.constEnd()) { return false; } return it.value() == SeatInterfacePrivate::Pointer::State::Pressed; } void SeatInterface::notifyPointerAxis(Qt::Orientation orientation, qreal delta, qint32 deltaV120, PointerAxisSource source, PointerAxisRelativeDirection direction) { if (!d->pointer) { return; } if (d->drag.mode == SeatInterfacePrivate::Drag::Mode::Pointer) { // ignore return; } d->pointer->sendAxis(orientation, delta, deltaV120, source, direction); } void SeatInterface::notifyPointerButton(Qt::MouseButton button, PointerButtonState state) { const quint32 nativeButton = qtToWaylandButton(button); if (nativeButton == 0) { return; } notifyPointerButton(nativeButton, state); } void SeatInterface::notifyPointerButton(quint32 button, PointerButtonState state) { if (!d->pointer) { return; } const quint32 serial = d->display->nextSerial(); if (state == PointerButtonState::Pressed) { d->updatePointerButtonSerial(button, serial); d->updatePointerButtonState(button, SeatInterfacePrivate::Pointer::State::Pressed); if (d->drag.mode == SeatInterfacePrivate::Drag::Mode::Pointer) { // ignore return; } } else { const quint32 currentButtonSerial = pointerButtonSerial(button); d->updatePointerButtonSerial(button, serial); d->updatePointerButtonState(button, SeatInterfacePrivate::Pointer::State::Released); if (d->drag.mode == SeatInterfacePrivate::Drag::Mode::Pointer) { if (d->drag.dragImplicitGrabSerial != currentButtonSerial) { // not our drag button - ignore return; } d->pointer->sendButton(button, state, serial); d->endDrag(); return; } } d->pointer->sendButton(button, state, serial); } void SeatInterface::notifyPointerFrame() { if (!d->pointer) { return; } d->pointer->sendFrame(); } quint32 SeatInterface::pointerButtonSerial(Qt::MouseButton button) const { return pointerButtonSerial(qtToWaylandButton(button)); } quint32 SeatInterface::pointerButtonSerial(quint32 button) const { auto it = d->globalPointer.buttonSerials.constFind(button); if (it == d->globalPointer.buttonSerials.constEnd()) { return 0; } return it.value(); } void SeatInterface::relativePointerMotion(const QPointF &delta, const QPointF &deltaNonAccelerated, std::chrono::microseconds time) { if (!d->pointer) { return; } auto relativePointer = RelativePointerV1Interface::get(pointer()); if (relativePointer) { relativePointer->sendRelativeMotion(delta, deltaNonAccelerated, time); } } void SeatInterface::startPointerSwipeGesture(quint32 fingerCount) { if (!d->pointer) { return; } auto swipeGesture = PointerSwipeGestureV1Interface::get(pointer()); if (swipeGesture) { swipeGesture->sendBegin(d->display->nextSerial(), fingerCount); } } void SeatInterface::updatePointerSwipeGesture(const QPointF &delta) { if (!d->pointer) { return; } auto swipeGesture = PointerSwipeGestureV1Interface::get(pointer()); if (swipeGesture) { swipeGesture->sendUpdate(delta); } } void SeatInterface::endPointerSwipeGesture() { if (!d->pointer) { return; } auto swipeGesture = PointerSwipeGestureV1Interface::get(pointer()); if (swipeGesture) { swipeGesture->sendEnd(d->display->nextSerial()); } } void SeatInterface::cancelPointerSwipeGesture() { if (!d->pointer) { return; } auto swipeGesture = PointerSwipeGestureV1Interface::get(pointer()); if (swipeGesture) { swipeGesture->sendCancel(d->display->nextSerial()); } } void SeatInterface::startPointerPinchGesture(quint32 fingerCount) { if (!d->pointer) { return; } auto pinchGesture = PointerPinchGestureV1Interface::get(pointer()); if (pinchGesture) { pinchGesture->sendBegin(d->display->nextSerial(), fingerCount); } } void SeatInterface::updatePointerPinchGesture(const QPointF &delta, qreal scale, qreal rotation) { if (!d->pointer) { return; } auto pinchGesture = PointerPinchGestureV1Interface::get(pointer()); if (pinchGesture) { pinchGesture->sendUpdate(delta, scale, rotation); } } void SeatInterface::endPointerPinchGesture() { if (!d->pointer) { return; } auto pinchGesture = PointerPinchGestureV1Interface::get(pointer()); if (pinchGesture) { pinchGesture->sendEnd(d->display->nextSerial()); } } void SeatInterface::cancelPointerPinchGesture() { if (!d->pointer) { return; } auto pinchGesture = PointerPinchGestureV1Interface::get(pointer()); if (pinchGesture) { pinchGesture->sendCancel(d->display->nextSerial()); } } void SeatInterface::startPointerHoldGesture(quint32 fingerCount) { if (!d->pointer) { return; } auto holdGesture = PointerHoldGestureV1Interface::get(pointer()); if (holdGesture) { holdGesture->sendBegin(d->display->nextSerial(), fingerCount); } } void SeatInterface::endPointerHoldGesture() { if (!d->pointer) { return; } auto holdGesture = PointerHoldGestureV1Interface::get(pointer()); if (holdGesture) { holdGesture->sendEnd(d->display->nextSerial()); } } void SeatInterface::cancelPointerHoldGesture() { if (!d->pointer) { return; } auto holdGesture = PointerHoldGestureV1Interface::get(pointer()); if (holdGesture) { holdGesture->sendCancel(d->display->nextSerial()); } } SurfaceInterface *SeatInterface::focusedKeyboardSurface() const { return d->globalKeyboard.focus.surface; } void SeatInterface::setFocusedKeyboardSurface(SurfaceInterface *surface) { if (!d->keyboard) { return; } Q_EMIT focusedKeyboardSurfaceAboutToChange(surface); const quint32 serial = d->display->nextSerial(); if (d->globalKeyboard.focus.surface) { disconnect(d->globalKeyboard.focus.destroyConnection); } d->globalKeyboard.focus = SeatInterfacePrivate::Keyboard::Focus(); d->globalKeyboard.focus.surface = surface; d->keyboard->setFocusedSurface(surface, serial); if (d->globalKeyboard.focus.surface) { d->globalKeyboard.focus.destroyConnection = connect(surface, &QObject::destroyed, this, [this]() { d->globalKeyboard.focus = SeatInterfacePrivate::Keyboard::Focus(); }); d->globalKeyboard.focus.serial = serial; // selection? const QList dataDevices = d->dataDevicesForSurface(surface); d->globalKeyboard.focus.selections = dataDevices; for (auto dataDevice : dataDevices) { dataDevice->sendSelection(d->currentSelection); } // primary selection QList primarySelectionDevices; for (auto it = d->primarySelectionDevices.constBegin(); it != d->primarySelectionDevices.constEnd(); ++it) { if ((*it)->client() == *surface->client()) { primarySelectionDevices << *it; } } d->globalKeyboard.focus.primarySelections = primarySelectionDevices; for (auto primaryDataDevice : primarySelectionDevices) { primaryDataDevice->sendSelection(d->currentPrimarySelection); } } // focused text input surface follows keyboard if (hasKeyboard()) { setFocusedTextInputSurface(surface); } } KeyboardInterface *SeatInterface::keyboard() const { return d->keyboard.get(); } void SeatInterface::notifyKeyboardKey(quint32 keyCode, KeyboardKeyState state) { if (!d->keyboard) { return; } d->keyboard->sendKey(keyCode, state); } void SeatInterface::notifyKeyboardModifiers(quint32 depressed, quint32 latched, quint32 locked, quint32 group) { if (!d->keyboard) { return; } d->keyboard->sendModifiers(depressed, latched, locked, group); } void SeatInterface::notifyTouchCancel() { if (!d->touch) { return; } d->touch->sendCancel(); if (d->drag.mode == SeatInterfacePrivate::Drag::Mode::Touch) { // cancel the drag, don't drop. serial does not matter d->cancelDrag(); } d->globalTouch.ids.clear(); } SurfaceInterface *SeatInterface::focusedTouchSurface() const { return d->globalTouch.focus.surface; } QPointF SeatInterface::focusedTouchSurfacePosition() const { return d->globalTouch.focus.offset; } bool SeatInterface::isTouchSequence() const { return !d->globalTouch.ids.isEmpty(); } TouchInterface *SeatInterface::touch() const { return d->touch.get(); } QPointF SeatInterface::firstTouchPointPosition() const { return d->globalTouch.focus.firstTouchPos; } void SeatInterface::setFocusedTouchSurface(SurfaceInterface *surface, const QPointF &surfacePosition) { if (!d->touch) { return; } if (isTouchSequence()) { // changing surface not allowed during a touch sequence return; } if (isDragTouch()) { return; } if (d->globalTouch.focus.surface) { disconnect(d->globalTouch.focus.destroyConnection); } d->globalTouch.focus = SeatInterfacePrivate::Touch::Focus(); d->globalTouch.focus.surface = surface; setFocusedTouchSurfacePosition(surfacePosition); if (d->globalTouch.focus.surface) { d->globalTouch.focus.destroyConnection = connect(surface, &QObject::destroyed, this, [this]() { if (isTouchSequence()) { // Surface destroyed during touch sequence - send a cancel d->touch->sendCancel(); } d->globalTouch.focus = SeatInterfacePrivate::Touch::Focus(); }); } } void SeatInterface::setFocusedTouchSurfacePosition(const QPointF &surfacePosition) { d->globalTouch.focus.offset = surfacePosition; d->globalTouch.focus.transformation = QMatrix4x4(); d->globalTouch.focus.transformation.translate(-surfacePosition.x(), -surfacePosition.y()); } void SeatInterface::notifyTouchDown(qint32 id, const QPointF &globalPosition) { if (!d->touch || !focusedTouchSurface()) { return; } const qint32 serial = display()->nextSerial(); auto pos = globalPosition - d->globalTouch.focus.offset; SurfaceInterface *effectiveFocusedSurface = focusedTouchSurface()->inputSurfaceAt(pos); if (effectiveFocusedSurface && effectiveFocusedSurface != focusedTouchSurface()) { pos = focusedTouchSurface()->mapToChild(effectiveFocusedSurface, pos); } else if (!effectiveFocusedSurface) { effectiveFocusedSurface = focusedTouchSurface(); } d->touch->sendDown(id, serial, pos, effectiveFocusedSurface); if (id == 0) { d->globalTouch.focus.firstTouchPos = globalPosition; } if (id == 0 && hasPointer() && focusedTouchSurface()) { TouchInterfacePrivate *touchPrivate = TouchInterfacePrivate::get(d->touch.get()); if (!touchPrivate->hasTouchesForClient(effectiveFocusedSurface->client())) { // If the client did not bind the touch interface fall back // to at least emulating touch through pointer events. d->pointer->sendEnter(effectiveFocusedSurface, pos, serial); d->pointer->sendMotion(pos); d->pointer->sendFrame(); } } d->globalTouch.ids[id] = serial; } void SeatInterface::notifyTouchMotion(qint32 id, const QPointF &globalPosition) { if (!d->touch) { return; } auto itTouch = d->globalTouch.ids.constFind(id); if (itTouch == d->globalTouch.ids.constEnd()) { // This can happen in cases where the interaction started while the device was asleep qCWarning(KWIN_CORE) << "Detected a touch move that never has been down, discarding"; return; } auto pos = globalPosition - d->globalTouch.focus.offset; SurfaceInterface *effectiveFocusedSurface = d->touch->focusedSurface(); if (effectiveFocusedSurface && focusedTouchSurface() != effectiveFocusedSurface) { pos = focusedTouchSurface()->mapToChild(effectiveFocusedSurface, pos); } if (isDragTouch()) { // handled by DataDevice } else { d->touch->sendMotion(id, pos); } if (id == 0) { d->globalTouch.focus.firstTouchPos = globalPosition; if (hasPointer() && focusedTouchSurface()) { TouchInterfacePrivate *touchPrivate = TouchInterfacePrivate::get(d->touch.get()); if (!touchPrivate->hasTouchesForClient(focusedTouchSurface()->client())) { // Client did not bind touch, fall back to emulating with pointer events. d->pointer->sendMotion(pos); d->pointer->sendFrame(); } } } Q_EMIT touchMoved(id, *itTouch, globalPosition); } void SeatInterface::notifyTouchUp(qint32 id) { if (!d->touch) { return; } auto itTouch = d->globalTouch.ids.find(id); if (itTouch == d->globalTouch.ids.end()) { // This can happen in cases where the interaction started while the device was asleep qCWarning(KWIN_CORE) << "Detected a touch that never started, discarding"; return; } const qint32 serial = d->display->nextSerial(); if (d->drag.mode == SeatInterfacePrivate::Drag::Mode::Touch && d->drag.dragImplicitGrabSerial == d->globalTouch.ids.value(id)) { // the implicitly grabbing touch point has been upped d->endDrag(); } d->touch->sendUp(id, serial); if (id == 0 && hasPointer() && focusedTouchSurface()) { TouchInterfacePrivate *touchPrivate = TouchInterfacePrivate::get(d->touch.get()); if (!touchPrivate->hasTouchesForClient(focusedTouchSurface()->client())) { // Client did not bind touch, fall back to emulating with pointer events. const quint32 serial = display()->nextSerial(); d->pointer->sendButton(BTN_LEFT, PointerButtonState::Released, serial); d->pointer->sendFrame(); } } d->globalTouch.ids.erase(itTouch); } void SeatInterface::notifyTouchFrame() { if (!d->touch) { return; } d->touch->sendFrame(); } bool SeatInterface::hasImplicitTouchGrab(quint32 serial) const { if (!d->globalTouch.focus.surface) { // origin surface has been destroyed return false; } return d->globalTouch.ids.key(serial, -1) != -1; } bool SeatInterface::isDrag() const { return d->drag.mode != SeatInterfacePrivate::Drag::Mode::None; } bool SeatInterface::isDragPointer() const { return d->drag.mode == SeatInterfacePrivate::Drag::Mode::Pointer; } bool SeatInterface::isDragTouch() const { return d->drag.mode == SeatInterfacePrivate::Drag::Mode::Touch; } bool SeatInterface::hasImplicitPointerGrab(quint32 serial) const { const auto &serials = d->globalPointer.buttonSerials; for (auto it = serials.constBegin(), end = serials.constEnd(); it != end; it++) { if (it.value() == serial) { return isPointerButtonPressed(it.key()); } } return false; } QMatrix4x4 SeatInterface::dragSurfaceTransformation() const { return d->drag.transformation; } SurfaceInterface *SeatInterface::dragSurface() const { return d->drag.surface; } AbstractDataSource *SeatInterface::dragSource() const { return d->drag.source; } XdgToplevelDragV1Interface *SeatInterface::xdgTopleveldrag() const { if (auto source = qobject_cast(d->drag.source)) { return source->xdgToplevelDrag(); } return nullptr; } void SeatInterface::setFocusedTextInputSurface(SurfaceInterface *surface) { const quint32 serial = d->display->nextSerial(); if (d->focusedTextInputSurface == surface) { return; } if (d->focusedTextInputSurface) { disconnect(d->focusedSurfaceDestroyConnection); d->textInputV1->d->sendLeave(d->focusedTextInputSurface); d->textInputV2->d->sendLeave(serial, d->focusedTextInputSurface); d->textInputV3->d->sendLeave(d->focusedTextInputSurface); } d->focusedTextInputSurface = surface; if (surface) { d->focusedSurfaceDestroyConnection = connect(surface, &SurfaceInterface::aboutToBeDestroyed, this, [this] { setFocusedTextInputSurface(nullptr); }); d->textInputV1->d->sendEnter(surface); d->textInputV2->d->sendEnter(surface, serial); d->textInputV3->d->sendEnter(surface); } Q_EMIT focusedTextInputSurfaceChanged(); } SurfaceInterface *SeatInterface::focusedTextInputSurface() const { return d->focusedTextInputSurface; } TextInputV1Interface *SeatInterface::textInputV1() const { return d->textInputV1; } TextInputV2Interface *SeatInterface::textInputV2() const { return d->textInputV2; } TextInputV3Interface *SeatInterface::textInputV3() const { return d->textInputV3; } AbstractDataSource *SeatInterface::selection() const { return d->currentSelection; } void SeatInterface::setSelection(AbstractDataSource *selection) { if (d->currentSelection == selection) { return; } if (d->currentSelection) { d->currentSelection->cancel(); disconnect(d->currentSelection, nullptr, this, nullptr); } if (selection) { auto cleanup = [this]() { setSelection(nullptr); }; connect(selection, &AbstractDataSource::aboutToBeDestroyed, this, cleanup); } d->currentSelection = selection; for (auto focussedSelection : std::as_const(d->globalKeyboard.focus.selections)) { focussedSelection->sendSelection(selection); } for (auto control : std::as_const(d->dataControlDevices)) { control->sendSelection(selection); } Q_EMIT selectionChanged(selection); } AbstractDataSource *SeatInterface::primarySelection() const { return d->currentPrimarySelection; } void SeatInterface::setPrimarySelection(AbstractDataSource *selection) { if (d->currentPrimarySelection == selection) { return; } if (d->currentPrimarySelection) { d->currentPrimarySelection->cancel(); disconnect(d->currentPrimarySelection, nullptr, this, nullptr); } if (selection) { auto cleanup = [this]() { setPrimarySelection(nullptr); }; connect(selection, &AbstractDataSource::aboutToBeDestroyed, this, cleanup); } d->currentPrimarySelection = selection; for (auto focussedSelection : std::as_const(d->globalKeyboard.focus.primarySelections)) { focussedSelection->sendSelection(selection); } for (auto control : std::as_const(d->dataControlDevices)) { control->sendPrimarySelection(selection); } Q_EMIT primarySelectionChanged(selection); } void SeatInterface::startDrag(AbstractDataSource *dragSource, SurfaceInterface *originSurface, int dragSerial, DragAndDropIcon *dragIcon) { if (hasImplicitPointerGrab(dragSerial)) { d->drag.mode = SeatInterfacePrivate::Drag::Mode::Pointer; d->drag.transformation = d->globalPointer.focus.transformation; } else if (hasImplicitTouchGrab(dragSerial)) { d->drag.mode = SeatInterfacePrivate::Drag::Mode::Touch; d->drag.transformation = d->globalTouch.focus.transformation; } else { // no implicit grab, abort drag return; } d->drag.dragImplicitGrabSerial = dragSerial; // set initial drag target to ourself d->drag.surface = originSurface; d->drag.source = dragSource; if (dragSource) { d->drag.dragSourceDestroyConnection = QObject::connect(dragSource, &AbstractDataSource::aboutToBeDestroyed, this, [this] { d->cancelDrag(); }); } d->drag.dragIcon = dragIcon; if (!d->dataDevicesForSurface(originSurface).isEmpty()) { d->drag.target = d->dataDevicesForSurface(originSurface)[0]; } if (d->drag.target) { d->drag.target->updateDragTarget(originSurface, dragSerial); } Q_EMIT dragStarted(); Q_EMIT dragSurfaceChanged(); } DragAndDropIcon *SeatInterface::dragIcon() const { return d->drag.dragIcon; } } #include "moc_seat.cpp"