QT 之wayland 事件处理分析基于qt5wayland5.14.2
1. Qt wayland 初始化 接收鼠标/案件,触摸屏等事件事件
QWaylandNativeInterface : public QPlatformNativeInterface
在QWaylandNativeInterface 继承qpa 接口类QPlatformNativeInterface;
1.1 初始化鼠标:
void *QWaylandNativeInterface::nativeResourceForIntegration(const QByteArray &resourceString)
{QByteArray lowerCaseResource = resourceString.toLower();if (lowerCaseResource == "display" || lowerCaseResource == "wl_display" || lowerCaseResource == "nativedisplay")return m_integration->display()->wl_display();if (lowerCaseResource == "compositor")return const_cast<wl_compositor *>(m_integration->display()->wl_compositor());if (lowerCaseResource == "server_buffer_integration")return m_integration->serverBufferIntegration();if (lowerCaseResource == "egldisplay" && m_integration->clientBufferIntegration())return m_integration->clientBufferIntegration()->nativeResource(QWaylandClientBufferIntegration::EglDisplay);if (lowerCaseResource == "wl_seat")return m_integration->display()->defaultInputDevice()->wl_seat();if (lowerCaseResource == "wl_keyboard") {auto *keyboard = m_integration->display()->defaultInputDevice()->keyboard();if (keyboard)return keyboard->wl_keyboard();return nullptr;}if (lowerCaseResource == "wl_pointer") {auto *pointer = m_integration->display()->defaultInputDevice()->pointer();if (pointer)return pointer->wl_pointer();return nullptr;}if (lowerCaseResource == "wl_touch") {auto *touch = m_integration->display()->defaultInputDevice()->touch();if (touch)return touch->wl_touch();return nullptr;}return nullptr;
}
QWaylandInputDevice::Point 主要实现 QtWayland::wl_pointer 类函数
class Q_WAYLAND_CLIENT_WAYLAND_EXPORT wl_pointer{protected:virtual void pointer_enter(uint32_t serial, struct ::wl_surface *surface, wl_fixed_t surface_x, wl_fixed_t surface_y);virtual void pointer_leave(uint32_t serial, struct ::wl_surface *surface);virtual void pointer_motion(uint32_t time, wl_fixed_t surface_x, wl_fixed_t surface_y);virtual void pointer_button(uint32_t serial, uint32_t time, uint32_t button, uint32_t state);virtual void pointer_axis(uint32_t time, uint32_t axis, wl_fixed_t value);virtual void pointer_frame();virtual void pointer_axis_source(uint32_t axis_source);virtual void pointer_axis_stop(uint32_t time, uint32_t axis);virtual void pointer_axis_discrete(uint32_t axis, int32_t discrete);private:void init_listener();static const struct wl_pointer_listener m_wl_pointer_listener;static void handle_enter(void *data,struct ::wl_pointer *object,uint32_t serial,struct ::wl_surface *surface,wl_fixed_t surface_x,wl_fixed_t surface_y);static void handle_leave(void *data,struct ::wl_pointer *object,uint32_t serial,struct ::wl_surface *surface);static void handle_motion(void *data,struct ::wl_pointer *object,uint32_t time,wl_fixed_t surface_x,wl_fixed_t surface_y);static void handle_button(void *data,struct ::wl_pointer *object,uint32_t serial,uint32_t time,uint32_t button,uint32_t state);static void handle_axis(void *data,struct ::wl_pointer *object,uint32_t time,uint32_t axis,wl_fixed_t value);static void handle_frame(void *data,struct ::wl_pointer *object);static void handle_axis_source(void *data,struct ::wl_pointer *object,uint32_t axis_source);static void handle_axis_stop(void *data,struct ::wl_pointer *object,uint32_t time,uint32_t axis);static void handle_axis_discrete(void *data,struct ::wl_pointer *object,uint32_t axis,int32_t discrete);struct ::wl_pointer *m_wl_pointer;};
2. wl_pointer 类向weston 注册监听鼠标事件
const struct wl_pointer_listener wl_pointer::m_wl_pointer_listener = {wl_pointer::handle_enter,wl_pointer::handle_leave,wl_pointer::handle_motion,wl_pointer::handle_button,wl_pointer::handle_axis,wl_pointer::handle_frame,wl_pointer::handle_axis_source,wl_pointer::handle_axis_stop,wl_pointer::handle_axis_discrete,};void wl_pointer::init_listener(){wl_pointer_add_listener(m_wl_pointer, &m_wl_pointer_listener, this);}
weston 服务接收鼠标事件就调用m_wl_pointer_listener 函数集。
3. 事件产生流程:分析hand_enter 鼠标进入事件为例:
void wl_pointer::handle_enter(void *data,struct ::wl_pointer *object,uint32_t serial,struct ::wl_surface *surface,wl_fixed_t surface_x,wl_fixed_t surface_y){Q_UNUSED(object);static_cast<wl_pointer *>(data)->pointer_enter(serial,surface,surface_x,surface_y);}
hand_enter 处理调用虚函数pointer_enter。pointer_enter函数在QWaylandInputDevice::Pointe实现。
void QWaylandInputDevice::Pointer::pointer_enter(uint32_t serial, struct wl_surface *surface,wl_fixed_t sx, wl_fixed_t sy)
{if (!surface)return;QWaylandWindow *window = QWaylandWindow::fromWlSurface(surface);if (!window)return; // Ignore foreign surfacesif (mFocus) {qCWarning(lcQpaWayland) << "The compositor sent a wl_pointer.enter event before sending a"<< "leave event first, this is not allowed by the wayland protocol"<< "attempting to work around it by invalidating the current focus";invalidateFocus();}mFocus = window->waylandSurface();connect(mFocus.data(), &QObject::destroyed, this, &Pointer::handleFocusDestroyed);mSurfacePos = QPointF(wl_fixed_to_double(sx), wl_fixed_to_double(sy));mGlobalPos = window->window()->mapToGlobal(mSurfacePos.toPoint());mParent->mSerial = serial;mEnterSerial = serial;#if QT_CONFIG(cursor)// Depends on mEnterSerial being updatedupdateCursor();
#endifprintf("%s %d \n",__FUNCTION__,__LINE__);QWaylandWindow *grab = QWaylandWindow::mouseGrab();if (!grab)setFrameEvent(new EnterEvent(window, mSurfacePos, mGlobalPos));
}
setFrameEvent 发送EnterEvent 事件。
void QWaylandInputDevice::Pointer::setFrameEvent(QWaylandPointerEvent *event)
{qCDebug(lcQpaWaylandInput) << "Setting frame event " << event->type;if (mFrameData.event && mFrameData.event->type != event->type) {qCDebug(lcQpaWaylandInput) << "Flushing; previous was " << mFrameData.event->type;flushFrameEvent();}mFrameData.event = event;if (mParent->mVersion < WL_POINTER_FRAME_SINCE_VERSION) {qCDebug(lcQpaWaylandInput) << "Flushing new event; no frame event in this version";flushFrameEvent();}
}void QWaylandInputDevice::Pointer::flushFrameEvent()
{if (auto *event = mFrameData.event) {if (auto window = event->surface) {window->handleMouse(mParent, *event);} else if (mFrameData.event->type == QWaylandPointerEvent::Type::Release) {// If the window has been destroyed, we still need to report an up event, but it can't// be handled by the destroyed window (obviously), so send the event here instead.QWindowSystemInterface::handleMouseEvent(nullptr, event->timestamp, event->local,event->global, event->buttons, event->modifiers);}delete mFrameData.event;mFrameData.event = nullptr;}//TODO: do modifiers get passed correctly here?flushScrollEvent();
}void QWaylandInputDevice::Pointer::flushFrameEvent()
{if (auto *event = mFrameData.event) {if (auto window = event->surface) {window->handleMouse(mParent, *event);} else if (mFrameData.event->type == QWaylandPointerEvent::Type::Release) {// If the window has been destroyed, we still need to report an up event, but it can't// be handled by the destroyed window (obviously), so send the event here instead.QWindowSystemInterface::handleMouseEvent(nullptr, event->timestamp, event->local,event->global, event->buttons, event->modifiers);}delete mFrameData.event;mFrameData.event = nullptr;}//TODO: do modifiers get passed correctly here?flushScrollEvent();
}
window->handleMouse
void QWaylandWindow::handleMouse(QWaylandInputDevice *inputDevice, const QWaylandPointerEvent &e)
{if (e.type == QWaylandPointerEvent::Leave) {if (mWindowDecoration) {if (mMouseEventsInContentArea)QWindowSystemInterface::handleLeaveEvent(window());} else {QWindowSystemInterface::handleLeaveEvent(window());}
#if QT_CONFIG(cursor)restoreMouseCursor(inputDevice);
#endifreturn;}if (mWindowDecoration) {handleMouseEventWithDecoration(inputDevice, e);} else {switch (e.type) {case QWaylandPointerEvent::Enter:QWindowSystemInterface::handleEnterEvent(window(), e.local, e.global);break;case QWaylandPointerEvent::Press:case QWaylandPointerEvent::Release:case QWaylandPointerEvent::Motion:QWindowSystemInterface::handleMouseEvent(window(), e.timestamp, e.local, e.global, e.buttons, e.modifiers);break;case QWaylandPointerEvent::Wheel:QWindowSystemInterface::handleWheelEvent(window(), e.timestamp, e.local, e.global,e.pixelDelta, e.angleDelta, e.modifiers,e.phase, e.source, false);break;}}#if QT_CONFIG(cursor)if (e.type == QWaylandPointerEvent::Enter) {QRect contentGeometry = windowContentGeometry().marginsRemoved(frameMargins());if (contentGeometry.contains(e.local.toPoint()))restoreMouseCursor(inputDevice);}
#endif
}
调用 QWindowSystemInterface::handleMouseEvent(window()。 handleMouseEvent 函数是qtbase 代码中实现的。
QT_DEFINE_QPA_EVENT_HANDLER(void, handleMouseEvent, QWindow *window,const QPointF &local, const QPointF &global, Qt::MouseButtons state,Qt::MouseButton button, QEvent::Type type, Qt::KeyboardModifiers mods,Qt::MouseEventSource source)
{unsigned long time = QWindowSystemInterfacePrivate::eventTime.elapsed();handleMouseEvent<Delivery>(window, time, local, global, state, button, type, mods, source);
}QT_DEFINE_QPA_EVENT_HANDLER(void, handleMouseEvent, QWindow *window, ulong timestamp,const QPointF &local, const QPointF &global, Qt::MouseButtons state,Qt::MouseButton button, QEvent::Type type, Qt::KeyboardModifiers mods,Qt::MouseEventSource source)
{Q_ASSERT_X(type != QEvent::MouseButtonDblClick && type != QEvent::NonClientAreaMouseButtonDblClick,"QWindowSystemInterface::handleMouseEvent","QTBUG-71263: Native double clicks are not implemented.");auto localPos = QHighDpi::fromNativeLocalPosition(local, window);auto globalPos = QHighDpi::fromNativePixels(global, window);QWindowSystemInterfacePrivate::MouseEvent *e =new QWindowSystemInterfacePrivate::MouseEvent(window, timestamp, localPos, globalPos,state, mods, button, type, source);QWindowSystemInterfacePrivate::handleWindowSystemEvent<Delivery>(e);
}
handleMouseEvent 调用handleWindowSystemEvent 函数。
template<>
bool QWindowSystemInterfacePrivate::handleWindowSystemEvent<QWindowSystemInterface::DefaultDelivery>(QWindowSystemInterfacePrivate::WindowSystemEvent *ev)
{if (synchronousWindowSystemEvents)return handleWindowSystemEvent<QWindowSystemInterface::SynchronousDelivery>(ev);elsereturn handleWindowSystemEvent<QWindowSystemInterface::AsynchronousDelivery>(ev);
}
handleWindowSystemEvent 同步处理及异步处理
template<>
bool QWindowSystemInterfacePrivate::handleWindowSystemEvent<QWindowSystemInterface::AsynchronousDelivery>(WindowSystemEvent *ev)
{windowSystemEventQueue.append(ev);if (QAbstractEventDispatcher *dispatcher = QGuiApplicationPrivate::qt_qpa_core_dispatcher())dispatcher->wakeUp();return true;
}
异步处理实现:将事件加入到windowSystemEventQueue 队列处理。
同步处理:
template<>
bool QWindowSystemInterfacePrivate::handleWindowSystemEvent<QWindowSystemInterface::SynchronousDelivery>(WindowSystemEvent *ev)
{bool accepted = true;if (QThread::currentThread() == QGuiApplication::instance()->thread()) {// Process the event immediately on the current thread and return the accepted state.QGuiApplicationPrivate::processWindowSystemEvent(ev);accepted = ev->eventAccepted;delete ev;} else {// Post the event on the Qt main thread queue and flush the queue.// This will wake up the Gui thread which will process the event.// Return the accepted state for the last event on the queue,// which is the event posted by this function.handleWindowSystemEvent<QWindowSystemInterface::AsynchronousDelivery>(ev);accepted = QWindowSystemInterface::flushWindowSystemEvents();}return accepted;
}/*!Make Qt Gui process all events on the event queue immediately. Return theaccepted state for the last event on the queue.
*/
bool QWindowSystemInterface::flushWindowSystemEvents(QEventLoop::ProcessEventsFlags flags)
{const int count = QWindowSystemInterfacePrivate::windowSystemEventQueue.count();if (!count)return false;if (!QGuiApplication::instance()) {qWarning().nospace()<< "QWindowSystemInterface::flushWindowSystemEvents() invoked after ""QGuiApplication destruction, discarding " << count << " events.";QWindowSystemInterfacePrivate::windowSystemEventQueue.clear();return false;}if (QThread::currentThread() != QGuiApplication::instance()->thread()) {// Post a FlushEvents event which will trigger a call back to// deferredFlushWindowSystemEvents from the Gui thread.QMutexLocker locker(&QWindowSystemInterfacePrivate::flushEventMutex);QWindowSystemInterfacePrivate::FlushEventsEvent *e = new QWindowSystemInterfacePrivate::FlushEventsEvent(flags);QWindowSystemInterfacePrivate::handleWindowSystemEvent<AsynchronousDelivery>(e);QWindowSystemInterfacePrivate::eventsFlushed.wait(&QWindowSystemInterfacePrivate::flushEventMutex);} else {sendWindowSystemEvents(flags);}return QWindowSystemInterfacePrivate::eventAccepted.loadRelaxed() > 0;
}
ool QWindowSystemInterface::sendWindowSystemEvents(QEventLoop::ProcessEventsFlags flags)
{int nevents = 0;while (QWindowSystemInterfacePrivate::windowSystemEventsQueued()) {QWindowSystemInterfacePrivate::WindowSystemEvent *event = nullptr;if (QWindowSystemInterfacePrivate::platformFiltersEvents) {event = QWindowSystemInterfacePrivate::getWindowSystemEvent();} else {event = flags & QEventLoop::ExcludeUserInputEvents ?QWindowSystemInterfacePrivate::getNonUserInputWindowSystemEvent() :QWindowSystemInterfacePrivate::getWindowSystemEvent();}if (!event)break;if (QWindowSystemInterfacePrivate::eventHandler) {if (QWindowSystemInterfacePrivate::eventHandler->sendEvent(event))nevents++;} else {nevents++;QGuiApplicationPrivate::processWindowSystemEvent(event);}// Record the accepted state for the processed event// (excluding flush events). This state can then be// returned by flushWindowSystemEvents().if (event->type != QWindowSystemInterfacePrivate::FlushEvents)QWindowSystemInterfacePrivate::eventAccepted.storeRelaxed(event->eventAccepted);delete event;}return (nevents > 0);
}void QGuiApplicationPrivate::processWindowSystemEvent(QWindowSystemInterfacePrivate::WindowSystemEvent *e)
{Q_TRACE_SCOPE(QGuiApplicationPrivate_processWindowSystemEvent, e->type);switch(e->type) {case QWindowSystemInterfacePrivate::Mouse:QGuiApplicationPrivate::processMouseEvent(static_cast<QWindowSystemInterfacePrivate::MouseEvent *>(e));break;case QWindowSystemInterfacePrivate::Wheel:QGuiApplicationPrivate::processWheelEvent(static_cast<QWindowSystemInterfacePrivate::WheelEvent *>(e));break;case QWindowSystemInterfacePrivate::Key:QGuiApplicationPrivate::processKeyEvent(static_cast<QWindowSystemInterfacePrivate::KeyEvent *>(e));break;case QWindowSystemInterfacePrivate::Touch:QGuiApplicationPrivate::processTouchEvent(static_cast<QWindowSystemInterfacePrivate::TouchEvent *>(e));break;case QWindowSystemInterfacePrivate::GeometryChange:QGuiApplicationPrivate::processGeometryChangeEvent(static_cast<QWindowSystemInterfacePrivate::GeometryChangeEvent*>(e));break;case QWindowSystemInterfacePrivate::Enter:QGuiApplicationPrivate::processEnterEvent(static_cast<QWindowSystemInterfacePrivate::EnterEvent *>(e));break;case QWindowSystemInterfacePrivate::Leave:QGuiApplicationPrivate::processLeaveEvent(static_cast<QWindowSystemInterfacePrivate::LeaveEvent *>(e));break;case QWindowSystemInterfacePrivate::ActivatedWindow:QGuiApplicationPrivate::processActivatedEvent(static_cast<QWindowSystemInterfacePrivate::ActivatedWindowEvent *>(e));break;case QWindowSystemInterfacePrivate::WindowStateChanged:QGuiApplicationPrivate::processWindowStateChangedEvent(static_cast<QWindowSystemInterfacePrivate::WindowStateChangedEvent *>(e));break;case QWindowSystemInterfacePrivate::WindowScreenChanged:QGuiApplicationPrivate::processWindowScreenChangedEvent(static_cast<QWindowSystemInterfacePrivate::WindowScreenChangedEvent *>(e));break;case QWindowSystemInterfacePrivate::SafeAreaMarginsChanged:QGuiApplicationPrivate::processSafeAreaMarginsChangedEvent(static_cast<QWindowSystemInterfacePrivate::SafeAreaMarginsChangedEvent *>(e));break;case QWindowSystemInterfacePrivate::ApplicationStateChanged: {QWindowSystemInterfacePrivate::ApplicationStateChangedEvent * changeEvent = static_cast<QWindowSystemInterfacePrivate::ApplicationStateChangedEvent *>(e);QGuiApplicationPrivate::setApplicationState(changeEvent->newState, changeEvent->forcePropagate); }break;case QWindowSystemInterfacePrivate::ApplicationTermination:QGuiApplicationPrivate::processApplicationTermination(e);break;case QWindowSystemInterfacePrivate::FlushEvents: {QWindowSystemInterfacePrivate::FlushEventsEvent *flushEventsEvent = static_cast<QWindowSystemInterfacePrivate::FlushEventsEvent *>(e);QWindowSystemInterface::deferredFlushWindowSystemEvents(flushEventsEvent->flags); }break;case QWindowSystemInterfacePrivate::Close:QGuiApplicationPrivate::processCloseEvent(static_cast<QWindowSystemInterfacePrivate::CloseEvent *>(e));break;case QWindowSystemInterfacePrivate::ScreenOrientation:QGuiApplicationPrivate::processScreenOrientationChange(static_cast<QWindowSystemInterfacePrivate::ScreenOrientationEvent *>(e));break;case QWindowSystemInterfacePrivate::ScreenGeometry:QGuiApplicationPrivate::processScreenGeometryChange(static_cast<QWindowSystemInterfacePrivate::ScreenGeometryEvent *>(e));break;case QWindowSystemInterfacePrivate::ScreenLogicalDotsPerInch:QGuiApplicationPrivate::processScreenLogicalDotsPerInchChange(static_cast<QWindowSystemInterfacePrivate::ScreenLogicalDotsPerInchEvent *>(e));break;case QWindowSystemInterfacePrivate::ScreenRefreshRate:QGuiApplicationPrivate::processScreenRefreshRateChange(static_cast<QWindowSystemInterfacePrivate::ScreenRefreshRateEvent *>(e));break;case QWindowSystemInterfacePrivate::ThemeChange:QGuiApplicationPrivate::processThemeChanged(static_cast<QWindowSystemInterfacePrivate::ThemeChangeEvent *>(e));break;case QWindowSystemInterfacePrivate::Expose:QGuiApplicationPrivate::processExposeEvent(static_cast<QWindowSystemInterfacePrivate::ExposeEvent *>(e));break;case QWindowSystemInterfacePrivate::Tablet:QGuiApplicationPrivate::processTabletEvent(static_cast<QWindowSystemInterfacePrivate::TabletEvent *>(e));break;case QWindowSystemInterfacePrivate::TabletEnterProximity:QGuiApplicationPrivate::processTabletEnterProximityEvent(static_cast<QWindowSystemInterfacePrivate::TabletEnterProximityEvent *>(e));break;case QWindowSystemInterfacePrivate::TabletLeaveProximity:QGuiApplicationPrivate::processTabletLeaveProximityEvent(static_cast<QWindowSystemInterfacePrivate::TabletLeaveProximityEvent *>(e));break;
#ifndef QT_NO_GESTUREScase QWindowSystemInterfacePrivate::Gesture:QGuiApplicationPrivate::processGestureEvent(static_cast<QWindowSystemInterfacePrivate::GestureEvent *>(e));break;
#endifcase QWindowSystemInterfacePrivate::PlatformPanel:QGuiApplicationPrivate::processPlatformPanelEvent(static_cast<QWindowSystemInterfacePrivate::PlatformPanelEvent *>(e));break;case QWindowSystemInterfacePrivate::FileOpen:QGuiApplicationPrivate::processFileOpenEvent(static_cast<QWindowSystemInterfacePrivate::FileOpenEvent *>(e));break;
#ifndef QT_NO_CONTEXTMENUcase QWindowSystemInterfacePrivate::ContextMenu:QGuiApplicationPrivate::processContextMenuEvent(static_cast<QWindowSystemInterfacePrivate::ContextMenuEvent *>(e));break;
#endifcase QWindowSystemInterfacePrivate::EnterWhatsThisMode:QGuiApplication::postEvent(QGuiApplication::instance(), new QEvent(QEvent::EnterWhatsThisMode));break;default:qWarning() << "Unknown user input event type:" << e->type;break;}
}void QGuiApplicationPrivate::processMouseEvent(QWindowSystemInterfacePrivate::MouseEvent *e)
{QEvent::Type type = QEvent::None;Qt::MouseButton button = Qt::NoButton;QWindow *window = e->window.data();bool positionChanged = QGuiApplicationPrivate::lastCursorPosition != e->globalPos;bool mouseMove = false;bool mousePress = false;if (e->enhancedMouseEvent()) {type = e->buttonType;button = e->button;if (type == QEvent::NonClientAreaMouseMove || type == QEvent::MouseMove)mouseMove = true;else if (type == QEvent::NonClientAreaMouseButtonPress || type == QEvent::MouseButtonPress)mousePress = true;if (!mouseMove && positionChanged) {QWindowSystemInterfacePrivate::MouseEvent moveEvent(window, e->timestamp,e->localPos, e->globalPos, e->buttons ^ button, e->modifiers, Qt::NoButton,e->nonClientArea ? QEvent::NonClientAreaMouseMove : QEvent::MouseMove,e->source, e->nonClientArea);if (e->synthetic())moveEvent.flags |= QWindowSystemInterfacePrivate::WindowSystemEvent::Synthetic;processMouseEvent(&moveEvent); // mouse move excluding state changeprocessMouseEvent(e); // the original mouse eventreturn;}} else {Qt::MouseButtons stateChange = e->buttons ^ mouse_buttons;if (positionChanged && (stateChange != Qt::NoButton)) {QWindowSystemInterfacePrivate::MouseEvent moveEvent(window, e->timestamp, e->localPos,e->globalPos, mouse_buttons, e->modifiers, Qt::NoButton, QEvent::None, e->source,e->nonClientArea);if (e->synthetic())moveEvent.flags |= QWindowSystemInterfacePrivate::WindowSystemEvent::Synthetic;processMouseEvent(&moveEvent); // mouse move excluding state changeprocessMouseEvent(e); // the original mouse eventreturn;}// In the compatibility path we deduce event type and button that caused the eventif (positionChanged) {mouseMove = true;type = e->nonClientArea ? QEvent::NonClientAreaMouseMove : QEvent::MouseMove;} else {// Check to see if a new button has been pressed/released.for (uint mask = Qt::LeftButton; mask <= Qt::MaxMouseButton; mask <<= 1) {if (stateChange & mask) {button = Qt::MouseButton(mask);break;}}if (button == Qt::NoButton) {// Ignore mouse events that don't change the current state. This shouldn't// really happen, getting here can only mean that the stored button state// is out of sync with the actual physical button state.return;}if (button & e->buttons) {mousePress = true;type = e->nonClientArea ? QEvent::NonClientAreaMouseButtonPress: QEvent::MouseButtonPress;} else {type = e->nonClientArea ? QEvent::NonClientAreaMouseButtonRelease: QEvent::MouseButtonRelease;}}}modifier_buttons = e->modifiers;QPointF localPoint = e->localPos;QPointF globalPoint = e->globalPos;bool doubleClick = false;if (mouseMove) {QGuiApplicationPrivate::lastCursorPosition = globalPoint;const auto doubleClickDistance = e->source == Qt::MouseEventNotSynthesized ?mouseDoubleClickDistance : touchDoubleTapDistance;if (qAbs(globalPoint.x() - mousePressX) > doubleClickDistance ||qAbs(globalPoint.y() - mousePressY) > doubleClickDistance)mousePressButton = Qt::NoButton;} else {mouse_buttons = e->buttons;if (mousePress) {ulong doubleClickInterval = static_cast<ulong>(QGuiApplication::styleHints()->mouseDoubleClickInterval());doubleClick = e->timestamp - mousePressTime < doubleClickInterval && button == mousePressButton;mousePressTime = e->timestamp;mousePressButton = button;const QPoint point = QGuiApplicationPrivate::lastCursorPosition.toPoint();mousePressX = point.x();mousePressY = point.y();}}if (e->nullWindow()) {window = QGuiApplication::topLevelAt(globalPoint.toPoint());if (window) {// Moves and the release following a press must go to the same// window, even if the cursor has moved on over another window.if (e->buttons != Qt::NoButton) {if (!currentMousePressWindow)currentMousePressWindow = window;elsewindow = currentMousePressWindow;} else if (currentMousePressWindow) {window = currentMousePressWindow;currentMousePressWindow = 0;}QPointF delta = globalPoint - globalPoint.toPoint();localPoint = window->mapFromGlobal(globalPoint.toPoint()) + delta;}}if (!window)return;#ifndef QT_NO_CURSORif (!e->synthetic()) {if (const QScreen *screen = window->screen())if (QPlatformCursor *cursor = screen->handle()->cursor()) {const QPointF nativeLocalPoint = QHighDpi::toNativePixels(localPoint, screen);const QPointF nativeGlobalPoint = QHighDpi::toNativePixels(globalPoint, screen);QMouseEvent ev(type, nativeLocalPoint, nativeLocalPoint, nativeGlobalPoint,button, e->buttons, e->modifiers, e->source);ev.setTimestamp(e->timestamp);cursor->pointerEvent(ev);}}
#endifQMouseEvent ev(type, localPoint, localPoint, globalPoint, button, e->buttons, e->modifiers, e->source);ev.setTimestamp(e->timestamp);if (window->d_func()->blockedByModalWindow && !qApp->d_func()->popupActive()) {// a modal window is blocking this window, don't allow mouse events throughreturn;}if (doubleClick && (ev.type() == QEvent::MouseButtonPress)) {// QtBUG-25831, used to suppress delivery in qwidgetwindow.cppsetMouseEventFlags(&ev, ev.flags() | Qt::MouseEventCreatedDoubleClick);}QGuiApplication::sendSpontaneousEvent(window, &ev);e->eventAccepted = ev.isAccepted();if (!e->synthetic() && !ev.isAccepted()&& !e->nonClientArea&& qApp->testAttribute(Qt::AA_SynthesizeTouchForUnhandledMouseEvents)) {if (!m_fakeTouchDevice) {m_fakeTouchDevice = new QTouchDevice;QWindowSystemInterface::registerTouchDevice(m_fakeTouchDevice);}QList<QWindowSystemInterface::TouchPoint> points;QWindowSystemInterface::TouchPoint point;point.id = 1;point.area = QRectF(globalPoint.x() - 2, globalPoint.y() - 2, 4, 4);// only translate left button related events to// avoid strange touch event sequences when several// buttons are pressedif (type == QEvent::MouseButtonPress && button == Qt::LeftButton) {point.state = Qt::TouchPointPressed;} else if (type == QEvent::MouseButtonRelease && button == Qt::LeftButton) {point.state = Qt::TouchPointReleased;} else if (type == QEvent::MouseMove && (e->buttons & Qt::LeftButton)) {point.state = Qt::TouchPointMoved;} else {return;}points << point;QEvent::Type type;QList<QTouchEvent::TouchPoint> touchPoints =QWindowSystemInterfacePrivate::fromNativeTouchPoints(points, window, QTouchDevicePrivate::get(m_fakeTouchDevice)->id, &type);QWindowSystemInterfacePrivate::TouchEvent fake(window, e->timestamp, type, m_fakeTouchDevice, touchPoints, e->modifiers);fake.flags |= QWindowSystemInterfacePrivate::WindowSystemEvent::Synthetic;processTouchEvent(&fake);}if (doubleClick) {mousePressButton = Qt::NoButton;if (!e->window.isNull() || e->nullWindow()) { // QTBUG-36364, check if window closed in response to pressconst QEvent::Type doubleClickType = e->nonClientArea ? QEvent::NonClientAreaMouseButtonDblClick : QEvent::MouseButtonDblClick;QMouseEvent dblClickEvent(doubleClickType, localPoint, localPoint, globalPoint,button, e->buttons, e->modifiers, e->source);dblClickEvent.setTimestamp(e->timestamp);QGuiApplication::sendSpontaneousEvent(window, &dblClickEvent);}}
}/*!\internal
*/
bool QCoreApplication::sendSpontaneousEvent(QObject *receiver, QEvent *event)
{Q_TRACE(QCoreApplication_sendSpontaneousEvent, receiver, event, event->type());if (event)event->spont = true;return notifyInternal2(receiver, event);
}
/*!\internal
*/
bool QCoreApplication::sendSpontaneousEvent(QObject *receiver, QEvent *event)
{Q_TRACE(QCoreApplication_sendSpontaneousEvent, receiver, event, event->type());if (event)event->spont = true;return notifyInternal2(receiver, event);
}/*!\internal\since 5.6This function is here to make it possible for Qt extensions tohook into event notification without subclassing QApplication.
*/
bool QCoreApplication::notifyInternal2(QObject *receiver, QEvent *event)
{bool selfRequired = QCoreApplicationPrivate::threadRequiresCoreApplication();if (!self && selfRequired)return false;// Make it possible for Qt Script to hook into events even// though QApplication is subclassed...bool result = false;void *cbdata[] = { receiver, event, &result };if (QInternal::activateCallbacks(QInternal::EventNotifyCallback, cbdata)) {return result;}// Qt enforces the rule that events can only be sent to objects in// the current thread, so receiver->d_func()->threadData is// equivalent to QThreadData::current(), just without the function// call overhead.QObjectPrivate *d = receiver->d_func();QThreadData *threadData = d->threadData;QScopedScopeLevelCounter scopeLevelCounter(threadData);if (!selfRequired)return doNotify(receiver, event);return self->notify(receiver, event);
}bool QCoreApplication::notify(QObject *receiver, QEvent *event)
{// no events are delivered after ~QCoreApplication() has startedif (QCoreApplicationPrivate::is_app_closing)return true;return doNotify(receiver, event);
}static bool doNotify(QObject *receiver, QEvent *event)
{if (receiver == 0) { // serious errorqWarning("QCoreApplication::notify: Unexpected null receiver");return true;}#ifndef QT_NO_DEBUGQCoreApplicationPrivate::checkReceiverThread(receiver);
#endifreturn receiver->isWidgetType() ? false : QCoreApplicationPrivate::notify_helper(receiver, event);
}Helper function called by QCoreApplicationPrivate::notify() and qapplication.cpp*/
bool QCoreApplicationPrivate::notify_helper(QObject *receiver, QEvent * event)
{// Note: when adjusting the tracepoints in here// consider adjusting QApplicationPrivate::notify_helper too.Q_TRACE(QCoreApplication_notify_entry, receiver, event, event->type());bool consumed = false;bool filtered = false;Q_TRACE_EXIT(QCoreApplication_notify_exit, consumed, filtered);// send to all application event filters (only does anything in the main thread)if (QCoreApplication::self&& receiver->d_func()->threadData->thread.loadAcquire() == mainThread()&& QCoreApplication::self->d_func()->sendThroughApplicationEventFilters(receiver, event)) {filtered = true;return filtered;}// send to all receiver event filtersif (sendThroughObjectEventFilters(receiver, event)) {filtered = true;return filtered;}// deliver the eventconsumed = receiver->event(event);return consumed;
}receiver->event 指向QObject ;
可以看到,流程中涉及两个事件过滤器的调用:sendThroughApplicationEventFilters和sendThroughObjectEventFilters,事件过滤器调用完后,才是调用接收者的event函数。
QApplication的事件过滤器
上一小节提到的sendThroughApplicationEventFilters是处理app的事件过滤器。代码里会调用给app安装的所有事件过滤器(从代码中的注释看到,app的事件过滤器只能在主线程中被调用),我们给app安装的事件过滤器就是在这个阶段被执行的。
bool QCoreApplicationPrivate::sendThroughApplicationEventFilters(QObject *receiver, QEvent *event)
{// We can't access the application event filters outside of the main thread (race conditions)Q_ASSERT(receiver->d_func()->threadData->thread.loadAcquire() == mainThread());if (extraData) {// application event filters are only called for objects in the GUI threadfor (int i = 0; i < extraData->eventFilters.size(); ++i) {QObject *obj = extraData->eventFilters.at(i);...if (obj->eventFilter(receiver, event))return true;}}return false;
}
QObject的事件过滤器
sendThroughObjectEventFilters是处理对象的事件过滤器。里面会调用给接收者安装的全部事件过滤器,我们通过installEventFilter给某个对象安装的事件过滤器就是在这个阶段执行的。
bool QCoreApplicationPrivate::sendThroughObjectEventFilters(QObject *receiver, QEvent *event)
{...for (int i = 0; i < receiver->d_func()->extraData->eventFilters.size(); ++i) {QObject *obj = receiver->d_func()->extraData->eventFilters.at(i);...if (obj->eventFilter(receiver, event))return true;}}return false;
}
对象的event方法
从QCoreApplicationPrivate::notify_helper代码可以看到,处理完事件接收者的事件过滤器后,就会调用接收者的event方法来处理事件。
每一个从QObject继承出来的子类都有event方法,都可以处理自己的事件。
我们简单看下QWidget的event方法:
bool QWidget::event(QEvent *event)
{...switch (event->type()) {...case QEvent::MouseMove:mouseMoveEvent((QMouseEvent*)event);break;case QEvent::MouseButtonPress:mousePressEvent((QMouseEvent*)event);break;case QEvent::MouseButtonRelease:mouseReleaseEvent((QMouseEvent*)event);break;case QEvent::MouseButtonDblClick:mouseDoubleClickEvent((QMouseEvent*)event);break;...
}到这里
是不是很熟悉,各个具体的事件处理函数就是在这个阶段被调用的。
相关文章:
QT 之wayland 事件处理分析基于qt5wayland5.14.2
1. Qt wayland 初始化 接收鼠标/案件,触摸屏等事件事件 QWaylandNativeInterface : public QPlatformNativeInterface 在QWaylandNativeInterface 继承qpa 接口类QPlatformNativeInterface; 1.1 初始化鼠标: void *QWaylandNativeInterface::nativeR…...
【this 和 super 的区别】
在 Java 中,this 和 super 都是关键字,表示当前对象和父类对象。 this 关键字可以用于以下几种情况: 引用当前对象的成员变量,方法和构造方法,用于区分局部变量和成员变量重名的情况; 调用当前类的另外一…...

K8s:Monokle Desktop 一个集Yaml资源编写、项目管理、集群管理的 K8s IDE
写在前面 Monokle Desktop 是 kubeshop 推出的一个开源的 K8s IDE相关项目还有 Monokle CLI 和 Monokle Cloud相比其他的工具,Monokle Desktop 功能较全面,涉及 k8s 管理的整个生命周期博文内容:Monokle Desktop 下载安装,项目管理…...

自动化测试实战篇(8),jmeter并发测试登录接口,模拟从100到1000个用户同时登录测试服务器压力
首先进行使用jmeter进行并发测试之前就需要搞清楚线程和进程的区别还需要理解什么是并发、高并发、并行。还需要理解高并发中的以及老生常谈的,TCP三次握手协议和TCP四次握手协议**TCP三次握手协议指:****TCP四次挥手协议:**进入Jmeter&#…...

ATTCK v12版本战术实战研究—持久化(二)
一、前言前几期文章中,我们介绍了ATT&CK中侦察、资源开发、初始访问、执行战术、持久化战术的知识。那么从前文中介绍的相关持久化子技术来开展测试,进行更深一步的分析。本文主要内容是介绍攻击者在运用持久化子技术时,在相关的资产服务…...
python函数式编程
1 callable内建函数判断一个名字是否为一个可调用函数 >>> import math >>> x 1 >>> y math.sqrt >>> callable(x) False >>> callable(y) True 2 记录函数(文档字符串) >>> def square(x): …...
3.linux下安装mysql
1.安装前的环境准备 查看是否安装过mysql 首先检测Linux操作系统中是否安装了MySQL: # rpm -qa | grep -i mysql 卸载安装包 如果有信息出现,则进行删除,命令如下: # rpm -e --nodeps 包名 删除老版本mysql的开发头文件和…...
17、MySQL分库分表,原理实战
MySQL分库分表,原理实战 1.MyCAT分布式架构入门及双主架构1.1 主从架构1.2 MyCAT安装1.3 启动和连接1.4 配置文件介绍2.MyCAT读写分离架构2.1 架构说明2.2 创建用户2.3 schema.xml2.4 连接说明2.5 读写测试2.6 当前是单节点3.MyCAT高可用读写分离架构3.1 架构说明3.3 schema.xm…...

【C++的OpenCV】第九课-OpenCV图像常用操作(六):图像形态学-阈值的概念、功能及操作(threshold()函数))
目录一、阈值(thresh)的概念二、阈值在图形学中的用途三、阈值的作用和操作3.1 在OpenCV中可以进行的阈值操作3.2 操作实例3.2.1 threshold()函数介绍3.2.2 实例3.2.3 结果上节课的内容(作者还是鼓励各位同学按照顺序进行学习哦)&…...

[Java代码审计]—MCMS
环境搭建 MCMS 5.2.4:https://gitee.com/mingSoft/MCMS/tree/5.2.4/利用 idea 打开项目 创建数据库 mcms,导入 doc/mcms-5.2.8.sql 修改 src/main/resources/application-dev.yml 中关于数据库设置参数 启动项目登录后台 http://localhost:8080/ms/l…...
《程序员面试金典(第6版)》面试题 01.08. 零矩阵
题目描述 编写代码,移除未排序链表中的重复节点。保留最开始出现的节点。 示例1: 输入:[1, 2, 3, 3, 2, 1] 输出:[1, 2, 3] -示例2: 输入:[1, 1, 1, 1, 2] 输出:[1, 2] 提示: 链表长度在[0, 20000]范…...
初识 Python
文章目录简介用途解释器命令行模式交互模式输入和输出简介 高级编程语言,解释型语言代码在执行时会逐行翻译成 CPU 能理解的机器码代码精简,但运行速度慢基础代码库丰富,还有大量第三方库代码不能加密 用途 网络应用工具软件包装其他语言开…...
常用sql语句分享
SELECT COUNT(DISTINCT money) FROM ac_association_course;#COUNT() 函数返回匹配指定条件的行数SELECT AVG(money) FROM ac_association_course;#AVG 函数返回数值列的平均值。NULL 值不包括在计算中SELECT id FROM ac_association_course order by id desc limit 1;#返回最大…...

极狐GitLab DevSecOps 为企业许可证安全合规保驾护航
本文来自: 小马哥 极狐(GitLab) 技术布道师 开源许可证是开源软件的法律武器,是第三方正确使用开源软件的安全合规依据。 根据 Linux 发布的 SBOM 报告显示,98% 的企业都在使用开源软件(中文版报告详情)。随着开源使用…...
后端程序员的前端基础-前端三剑客之HTML
文章目录1 HTML简介1.1 什么是HTML1.2 HTML能做什么1.3 HTML书写规范2 HTML基本标签2.1 结构标签2.2 排版标签2.3 块标签2.4 基本文字标签2.5 文本格式化标签2.6 标题标签2.7 列表标签(清单标签)2.8 图片标签2.9 链接标签2.10 表格标签3 HTML表单标签3.1 form元素常用属性3.2 i…...

VS2019加载解决方案时不能自动打开之前的文档(回忆消失)
✏️作者:枫霜剑客 📋系列专栏:C实战宝典 🌲上一篇: 错误error c3861 :“_T“:找不到标识符 逐梦编程,让中华屹立世界之巅。 简单的事情重复做,重复的事情用心做,用心的事情坚持做; 文章目录前言一、问题描…...

ConcurrentHashMap-Java八股面试(五)
系列文章目录 第一章 ArrayList-Java八股面试(一) 第二章 HashMap-Java八股面试(二) 第三章 单例模式-Java八股面试(三) 第四章 线程池和Volatile关键字-Java八股面试(四) 提示:动态每日更新算法题,想要学习的可以关注一下 文章目录系列文章目录一、…...

互联网衰退期,测试工程师35岁的路该怎么走...
国内的互联网行业发展较快,所以造成了技术研发类员工工作强度比较大,同时技术的快速更新又需要员工不断的学习新的技术。因此淘汰率也比较高,超过35岁的基层研发类员工,往往因为家庭原因、身体原因,比较难以跟得上工作…...

Windows Cannot Initialize Data Bindings 问题的解决方法
前言 拿到一个调试程序, 怎么折腾都打不开, 在客户那边, 尝试了几个系统版本, 发现Windows 10 21H2 版本可以正常运行。 尝试 系统篇 系统结果公司电脑 Windows 8有问题…下载安装 Windows10 22H2问题依旧下载安装 Windows10 21H2问题依旧家里的 笔记本Window 11正常 网上…...

Leetcode每日一题 1487. 保证文件名唯一
Halo,这里是Ppeua。平时主要更新C语言,C,数据结构算法......感兴趣就关注我吧!你定不会失望。 🌈个人主页:主页链接 🌈算法专栏:专栏链接 我会一直往里填充内容哒! &…...

wordpress后台更新后 前端没变化的解决方法
使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…...

通过Wrangler CLI在worker中创建数据库和表
官方使用文档:Getting started Cloudflare D1 docs 创建数据库 在命令行中执行完成之后,会在本地和远程创建数据库: npx wranglerlatest d1 create prod-d1-tutorial 在cf中就可以看到数据库: 现在,您的Cloudfla…...

3.3.1_1 检错编码(奇偶校验码)
从这节课开始,我们会探讨数据链路层的差错控制功能,差错控制功能的主要目标是要发现并且解决一个帧内部的位错误,我们需要使用特殊的编码技术去发现帧内部的位错误,当我们发现位错误之后,通常来说有两种解决方案。第一…...
pam_env.so模块配置解析
在PAM(Pluggable Authentication Modules)配置中, /etc/pam.d/su 文件相关配置含义如下: 配置解析 auth required pam_env.so1. 字段分解 字段值说明模块类型auth认证类模块,负责验证用户身份&am…...

苍穹外卖--缓存菜品
1.问题说明 用户端小程序展示的菜品数据都是通过查询数据库获得,如果用户端访问量比较大,数据库访问压力随之增大 2.实现思路 通过Redis来缓存菜品数据,减少数据库查询操作。 缓存逻辑分析: ①每个分类下的菜品保持一份缓存数据…...
什么是EULA和DPA
文章目录 EULA(End User License Agreement)DPA(Data Protection Agreement)一、定义与背景二、核心内容三、法律效力与责任四、实际应用与意义 EULA(End User License Agreement) 定义: EULA即…...

【开发技术】.Net使用FFmpeg视频特定帧上绘制内容
目录 一、目的 二、解决方案 2.1 什么是FFmpeg 2.2 FFmpeg主要功能 2.3 使用Xabe.FFmpeg调用FFmpeg功能 2.4 使用 FFmpeg 的 drawbox 滤镜来绘制 ROI 三、总结 一、目的 当前市场上有很多目标检测智能识别的相关算法,当前调用一个医疗行业的AI识别算法后返回…...

stm32wle5 lpuart DMA数据不接收
配置波特率9600时,需要使用外部低速晶振...
TJCTF 2025
还以为是天津的。这个比较容易,虽然绕了点弯,可还是把CP AK了,不过我会的别人也会,还是没啥名次。记录一下吧。 Crypto bacon-bits with open(flag.txt) as f: flag f.read().strip() with open(text.txt) as t: text t.read…...
从实验室到产业:IndexTTS 在六大核心场景的落地实践
一、内容创作:重构数字内容生产范式 在短视频创作领域,IndexTTS 的语音克隆技术彻底改变了配音流程。B 站 UP 主通过 5 秒参考音频即可克隆出郭老师音色,生成的 “各位吴彦祖们大家好” 语音相似度达 97%,单条视频播放量突破百万…...