6#include <QtWaylandCompositor/QWaylandXdgSurface>
7#include <QtWaylandCompositor/QWaylandCompositor>
8#include <QtWaylandCompositor/QWaylandSeat>
14static void handlePopupCreated(QWaylandQuickShellSurfaceItem *parentItem, QWaylandXdgPopup *popup)
16 if (parentItem->shellSurface() == popup->parentXdgSurface())
17 QWaylandQuickShellSurfaceItemPrivate::get(parentItem)->maybeCreateAutoPopup(popup->xdgSurface());
21 : QWaylandQuickShellIntegration(item)
23 , m_xdgSurface(qobject_cast<QWaylandXdgSurface *>(item->shellSurface()))
24 , m_toplevel(m_xdgSurface->toplevel())
25 , grabberState(GrabberState::Default)
29 m_item->setSurface(m_xdgSurface->surface());
31 connect(m_toplevel, &QWaylandXdgToplevel::startMove,
this, &XdgToplevelIntegration::handleStartMove);
32 connect(m_toplevel, &QWaylandXdgToplevel::startResize,
this, &XdgToplevelIntegration::handleStartResize);
33 connect(m_toplevel, &QWaylandXdgToplevel::setMaximized,
this, &XdgToplevelIntegration::handleSetMaximized);
34 connect(m_toplevel, &QWaylandXdgToplevel::unsetMaximized,
this, &XdgToplevelIntegration::handleUnsetMaximized);
35 connect(m_toplevel, &QWaylandXdgToplevel::maximizedChanged,
this, &XdgToplevelIntegration::handleMaximizedChanged);
36 connect(m_toplevel, &QWaylandXdgToplevel::setFullscreen,
this, &XdgToplevelIntegration::handleSetFullscreen);
37 connect(m_toplevel, &QWaylandXdgToplevel::unsetFullscreen,
this, &XdgToplevelIntegration::handleUnsetFullscreen);
38 connect(m_toplevel, &QWaylandXdgToplevel::fullscreenChanged,
this, &XdgToplevelIntegration::handleFullscreenChanged);
39 connect(m_toplevel, &QWaylandXdgToplevel::activatedChanged,
this, &XdgToplevelIntegration::handleActivatedChanged);
40 connect(m_xdgSurface->shell(), &QWaylandXdgShell::popupCreated,
this, [item](QWaylandXdgPopup *popup, QWaylandXdgSurface *){
41 handlePopupCreated(item, popup);
43 connect(m_xdgSurface->surface(), &QWaylandSurface::destinationSizeChanged,
this, &XdgToplevelIntegration::handleSurfaceSizeChanged);
44 connect(m_toplevel, &QObject::destroyed,
this, &XdgToplevelIntegration::handleToplevelDestroyed);
49 if (event->type() == QEvent::MouseMove) {
50 QMouseEvent *mouseEvent =
static_cast<QMouseEvent *>(event);
51 return filterMouseMoveEvent(mouseEvent);
52 }
else if (event->type() == QEvent::MouseButtonRelease || event->type() == QEvent::TouchEnd || event->type() == QEvent::TouchCancel) {
53 return filterPointerReleaseEvent();
54 }
else if (event->type() == QEvent::TouchUpdate) {
55 QTouchEvent *touchEvent =
static_cast<QTouchEvent *>(event);
56 return filterTouchUpdateEvent(touchEvent);
58 return QWaylandQuickShellIntegration::eventFilter(object, event);
63 if (grabberState == GrabberState::Resize) {
64 if (!resizeState.initialized) {
65 resizeState.initialMousePos = scenePosition;
66 resizeState.initialized =
true;
69 QPointF delta = m_item->mapToSurface(scenePosition - resizeState.initialMousePos);
70 QSize newSize = m_toplevel->sizeForResize(resizeState.initialWindowSize, delta, resizeState.resizeEdges);
71 m_toplevel->sendResizing(newSize);
72 }
else if (grabberState == GrabberState::Move) {
73 QQuickItem *moveItem = m_item->moveItem();
74 if (!moveState.initialized) {
75 moveState.initialOffset = moveItem->mapFromItem(
nullptr, scenePosition);
76 moveState.initialized =
true;
79 if (!moveItem->parentItem())
81 QPointF parentPos = moveItem->parentItem()->mapFromItem(
nullptr, scenePosition);
82 moveItem->setPosition(parentPos - moveState.initialOffset);
89 if (event->pointCount() == 0)
92 Q_ASSERT(grabberState != GrabberState::Move || moveState.seat == m_item->compositor()->seatFor(event));
93 Q_ASSERT(grabberState != GrabberState::Resize || resizeState.seat == m_item->compositor()->seatFor(event));
95 QEventPoint point = event->points().first();
96 return filterPointerMoveEvent(point.scenePosition());
101 Q_ASSERT(grabberState != GrabberState::Move || moveState.seat == m_item->compositor()->seatFor(event));
102 Q_ASSERT(grabberState != GrabberState::Resize || resizeState.seat == m_item->compositor()->seatFor(event));
104 return filterPointerMoveEvent(event->scenePosition());
109 if (grabberState != GrabberState::Default) {
110 grabberState = GrabberState::Default;
118 grabberState = GrabberState::Move;
119 moveState.seat = seat;
120 moveState.initialized =
false;
125 grabberState = GrabberState::Resize;
126 resizeState.seat = seat;
127 resizeState.resizeEdges = edges;
128 resizeState.initialWindowSize = m_xdgSurface->windowGeometry().size();
129 resizeState.initialPosition = m_item->moveItem()->position();
130 resizeState.initialSurfaceSize = m_item->surface()->destinationSize();
131 resizeState.initialized =
false;
136 if (!m_item->view()->isPrimary())
139 QList<QWaylandXdgToplevel::State> states = m_toplevel->states();
141 if (!states.contains(QWaylandXdgToplevel::State::FullscreenState) && !states.contains(QWaylandXdgToplevel::State::MaximizedState)) {
142 windowedGeometry.initialWindowSize = m_xdgSurface->windowGeometry().size();
143 windowedGeometry.initialPosition = m_item->moveItem()->position();
147 disconnect(nonwindowedState.sizeChangedConnection);
148 nonwindowedState.output = m_item->view()->output();
149 nonwindowedState.sizeChangedConnection = connect(nonwindowedState.output, &QWaylandOutput::availableGeometryChanged,
this, &XdgToplevelIntegration::handleMaximizedSizeChanged);
150 handleMaximizedSizeChanged();
157 if (m_toplevel ==
nullptr)
160 m_toplevel->sendMaximized(nonwindowedState.output->availableGeometry().size() / nonwindowedState.output->scaleFactor());
165 if (!m_item->view()->isPrimary())
170 if (windowedGeometry.initialWindowSize.isValid())
171 m_toplevel->sendUnmaximized(windowedGeometry.initialWindowSize);
173 m_toplevel->sendUnmaximized();
178 if (m_toplevel->maximized()) {
179 if (
auto *output = m_item->view()->output()) {
180 m_item->moveItem()->setPosition(output->position() + output->availableGeometry().topLeft());
182 qCWarning(qLcWaylandCompositor) <<
"The view does not have a corresponding output,"
183 <<
"ignoring maximized state";
186 m_item->moveItem()->setPosition(windowedGeometry.initialPosition);
192 if (!m_item->view()->isPrimary())
195 QList<QWaylandXdgToplevel::State> states = m_toplevel->states();
197 if (!states.contains(QWaylandXdgToplevel::State::FullscreenState) && !states.contains(QWaylandXdgToplevel::State::MaximizedState)) {
198 windowedGeometry.initialWindowSize = m_xdgSurface->windowGeometry().size();
199 windowedGeometry.initialPosition = m_item->moveItem()->position();
203 disconnect(nonwindowedState.sizeChangedConnection);
204 nonwindowedState.output = m_item->view()->output();
205 nonwindowedState.sizeChangedConnection = connect(nonwindowedState.output, &QWaylandOutput::geometryChanged,
this, &XdgToplevelIntegration::handleFullscreenSizeChanged);
206 handleFullscreenSizeChanged();
213 if (m_toplevel ==
nullptr)
216 m_toplevel->sendFullscreen(nonwindowedState.output->geometry().size() / nonwindowedState.output->scaleFactor());
221 if (!m_item->view()->isPrimary())
226 if (windowedGeometry.initialWindowSize.isValid())
227 m_toplevel->sendUnmaximized(windowedGeometry.initialWindowSize);
229 m_toplevel->sendUnmaximized();
234 if (m_toplevel->fullscreen()) {
235 if (
auto *output = m_item->view()->output()) {
236 m_item->moveItem()->setPosition(output->position() + output->geometry().topLeft());
238 qCWarning(qLcWaylandCompositor) <<
"The view does not have a corresponding output,"
239 <<
"ignoring fullscreen state";
242 m_item->moveItem()->setPosition(windowedGeometry.initialPosition);
248 if (m_toplevel->activated())
254 if (grabberState == GrabberState::Resize) {
257 if (resizeState.resizeEdges & Qt::TopEdge)
258 dy = resizeState.initialSurfaceSize.height() - m_item->surface()->destinationSize().height();
259 if (resizeState.resizeEdges & Qt::LeftEdge)
260 dx = resizeState.initialSurfaceSize.width() - m_item->surface()->destinationSize().width();
261 QPointF offset = m_item->mapFromSurface({dx, dy});
262 m_item->moveItem()->setPosition(resizeState.initialPosition + offset);
269 nonwindowedState.output =
nullptr;
270 disconnect(nonwindowedState.sizeChangedConnection);
275 , m_xdgSurface(qobject_cast<QWaylandXdgSurface *>(item->shellSurface()))
276 , m_popup(m_xdgSurface->popup())
280 m_item->setSurface(m_xdgSurface->surface());
281 handleGeometryChanged();
283 connect(m_popup, &QWaylandXdgPopup::configuredGeometryChanged,
this, &XdgPopupIntegration::handleGeometryChanged);
284 connect(m_xdgSurface->shell(), &QWaylandXdgShell::popupCreated,
this, [item](QWaylandXdgPopup *popup, QWaylandXdgSurface *){
285 handlePopupCreated(item, popup);
291 if (m_item->view()->output()) {
292 const QPoint windowOffset = m_popup->parentXdgSurface()->windowGeometry().topLeft();
293 const QPoint surfacePosition = m_popup->unconstrainedPosition() + windowOffset;
294 const QPoint itemPosition = m_item->mapFromSurface(surfacePosition).toPoint();
297 m_item->moveItem()->setPosition(itemPosition);
299 qWarning() <<
"XdgPopupIntegration popup item without output" << m_item;
307#include "moc_qwaylandxdgshellintegration_p.cpp"
bool eventFilter(QObject *object, QEvent *event) override
Filters events if this object has been installed as an event filter for the watched object.
static void handlePopupCreated(QWaylandQuickShellSurfaceItem *parentItem, QWaylandXdgPopup *popup)