7#include <QtWaylandCompositor/QWaylandXdgSurface>
8#include <QtWaylandCompositor/QWaylandCompositor>
9#include <QtWaylandCompositor/QWaylandSeat>
15static void handlePopupCreated(QWaylandQuickShellSurfaceItem *parentItem, QWaylandXdgPopup *popup)
17 if (parentItem->shellSurface() == popup->parentXdgSurface())
18 QWaylandQuickShellSurfaceItemPrivate::get(parentItem)->maybeCreateAutoPopup(popup->xdgSurface());
22 : QWaylandQuickShellIntegration(item)
24 , m_xdgSurface(qobject_cast<QWaylandXdgSurface *>(item->shellSurface()))
25 , m_toplevel(m_xdgSurface->toplevel())
26 , grabberState(GrabberState::Default)
30 m_item->setSurface(m_xdgSurface->surface());
32 connect(m_toplevel, &QWaylandXdgToplevel::startMove,
this, &XdgToplevelIntegration::handleStartMove);
33 connect(m_toplevel, &QWaylandXdgToplevel::startResize,
this, &XdgToplevelIntegration::handleStartResize);
34 connect(m_toplevel, &QWaylandXdgToplevel::setMaximized,
this, &
XdgToplevelIntegration::handleSetMaximized);
35 connect(m_toplevel, &QWaylandXdgToplevel::unsetMaximized,
this, &
XdgToplevelIntegration::handleUnsetMaximized);
36 connect(m_toplevel, &QWaylandXdgToplevel::maximizedChanged,
this, &
XdgToplevelIntegration::handleMaximizedChanged);
37 connect(m_toplevel, &QWaylandXdgToplevel::setFullscreen,
this, &
XdgToplevelIntegration::handleSetFullscreen);
38 connect(m_toplevel, &QWaylandXdgToplevel::unsetFullscreen,
this, &
XdgToplevelIntegration::handleUnsetFullscreen);
39 connect(m_toplevel, &QWaylandXdgToplevel::fullscreenChanged,
this, &
XdgToplevelIntegration::handleFullscreenChanged);
40 connect(m_toplevel, &QWaylandXdgToplevel::activatedChanged,
this, &
XdgToplevelIntegration::handleActivatedChanged);
41 connect(m_xdgSurface->shell(), &QWaylandXdgShell::popupCreated,
this, [item](QWaylandXdgPopup *popup, QWaylandXdgSurface *){
42 handlePopupCreated(item, popup);
44 connect(m_xdgSurface->surface(), &QWaylandSurface::destinationSizeChanged,
this, &
XdgToplevelIntegration::handleSurfaceSizeChanged);
50 if (event->type() == QEvent::MouseMove) {
51 QMouseEvent *mouseEvent =
static_cast<QMouseEvent *>(event);
52 return filterMouseMoveEvent(mouseEvent);
53 }
else if (event->type() == QEvent::MouseButtonRelease || event->type() == QEvent::TouchEnd || event->type() == QEvent::TouchCancel) {
54 return filterPointerReleaseEvent();
55 }
else if (event->type() == QEvent::TouchUpdate) {
56 QTouchEvent *touchEvent =
static_cast<QTouchEvent *>(event);
57 return filterTouchUpdateEvent(touchEvent);
59 return QWaylandQuickShellIntegration::eventFilter(object, event);
64 if (grabberState == GrabberState::Resize) {
65 if (!resizeState.initialized) {
66 resizeState.initialMousePos = scenePosition;
67 resizeState.initialized =
true;
70 QPointF delta = m_item->mapToSurface(scenePosition - resizeState.initialMousePos);
71 QSize newSize = m_toplevel->sizeForResize(resizeState.initialWindowSize, delta, resizeState.resizeEdges);
72 m_toplevel->sendResizing(newSize);
73 }
else if (grabberState == GrabberState::Move) {
74 QQuickItem *moveItem = m_item->moveItem();
75 if (!moveState.initialized) {
76 moveState.initialOffset = moveItem->mapFromItem(
nullptr, scenePosition);
77 moveState.initialized =
true;
80 if (!moveItem->parentItem())
82 QPointF parentPos = moveItem->parentItem()->mapFromItem(
nullptr, scenePosition);
83 moveItem->setPosition(parentPos - moveState.initialOffset);
90 if (event->pointCount() == 0)
93 Q_ASSERT(grabberState != GrabberState::Move || moveState.seat == m_item->compositor()->seatFor(event));
94 Q_ASSERT(grabberState != GrabberState::Resize || resizeState.seat == m_item->compositor()->seatFor(event));
96 QEventPoint point = event->points().first();
97 return filterPointerMoveEvent(point.scenePosition());
102 Q_ASSERT(grabberState != GrabberState::Move || moveState.seat == m_item->compositor()->seatFor(event));
103 Q_ASSERT(grabberState != GrabberState::Resize || resizeState.seat == m_item->compositor()->seatFor(event));
105 return filterPointerMoveEvent(event->scenePosition());
110 if (grabberState != GrabberState::Default) {
111 grabberState = GrabberState::Default;
119 grabberState = GrabberState::Move;
120 moveState.seat = seat;
121 moveState.initialized =
false;
126 grabberState = GrabberState::Resize;
127 resizeState.seat = seat;
128 resizeState.resizeEdges = edges;
129 resizeState.initialWindowSize = m_xdgSurface->windowGeometry().size();
130 resizeState.initialPosition = m_item->moveItem()->position();
131 resizeState.initialSurfaceSize = m_item->surface()->destinationSize();
132 resizeState.initialized =
false;
137 if (!m_item->view()->isPrimary())
140 QList<QWaylandXdgToplevel::State> states = m_toplevel->states();
142 if (!states.contains(QWaylandXdgToplevel::State::FullscreenState) && !states.contains(QWaylandXdgToplevel::State::MaximizedState)) {
143 windowedGeometry.initialWindowSize = m_xdgSurface->windowGeometry().size();
144 windowedGeometry.initialPosition = m_item->moveItem()->position();
148 disconnect(nonwindowedState.sizeChangedConnection);
149 nonwindowedState.output = m_item->view()->output();
150 nonwindowedState.sizeChangedConnection = connect(nonwindowedState.output, &QWaylandOutput::availableGeometryChanged,
this, &XdgToplevelIntegration::handleMaximizedSizeChanged);
151 handleMaximizedSizeChanged();
158 if (m_toplevel ==
nullptr)
161 m_toplevel->sendMaximized(nonwindowedState.output->availableGeometry().size() / nonwindowedState.output->scaleFactor());
166 if (!m_item->view()->isPrimary())
171 if (windowedGeometry.initialWindowSize.isValid())
172 m_toplevel->sendUnmaximized(windowedGeometry.initialWindowSize);
174 m_toplevel->sendUnmaximized();
179 if (m_toplevel->maximized()) {
180 if (
auto *output = m_item->view()->output()) {
181 m_item->moveItem()->setPosition(output->position() + output->availableGeometry().topLeft());
183 qCWarning(qLcWaylandCompositor) <<
"The view does not have a corresponding output,"
184 <<
"ignoring maximized state";
187 m_item->moveItem()->setPosition(windowedGeometry.initialPosition);
193 if (!m_item->view()->isPrimary())
196 QList<QWaylandXdgToplevel::State> states = m_toplevel->states();
198 if (!states.contains(QWaylandXdgToplevel::State::FullscreenState) && !states.contains(QWaylandXdgToplevel::State::MaximizedState)) {
199 windowedGeometry.initialWindowSize = m_xdgSurface->windowGeometry().size();
200 windowedGeometry.initialPosition = m_item->moveItem()->position();
204 disconnect(nonwindowedState.sizeChangedConnection);
205 nonwindowedState.output = m_item->view()->output();
206 nonwindowedState.sizeChangedConnection = connect(nonwindowedState.output, &QWaylandOutput::geometryChanged,
this, &XdgToplevelIntegration::handleFullscreenSizeChanged);
207 handleFullscreenSizeChanged();
214 if (m_toplevel ==
nullptr)
217 m_toplevel->sendFullscreen(nonwindowedState.output->geometry().size() / nonwindowedState.output->scaleFactor());
222 if (!m_item->view()->isPrimary())
227 if (windowedGeometry.initialWindowSize.isValid())
228 m_toplevel->sendUnmaximized(windowedGeometry.initialWindowSize);
230 m_toplevel->sendUnmaximized();
235 if (m_toplevel->fullscreen()) {
236 if (
auto *output = m_item->view()->output()) {
237 m_item->moveItem()->setPosition(output->position() + output->geometry().topLeft());
239 qCWarning(qLcWaylandCompositor) <<
"The view does not have a corresponding output,"
240 <<
"ignoring fullscreen state";
243 m_item->moveItem()->setPosition(windowedGeometry.initialPosition);
249 if (m_toplevel->activated())
255 if (grabberState == GrabberState::Resize) {
258 if (resizeState.resizeEdges & Qt::TopEdge)
259 dy = resizeState.initialSurfaceSize.height() - m_item->surface()->destinationSize().height();
260 if (resizeState.resizeEdges & Qt::LeftEdge)
261 dx = resizeState.initialSurfaceSize.width() - m_item->surface()->destinationSize().width();
262 QPointF offset = m_item->mapFromSurface({dx, dy});
263 m_item->moveItem()->setPosition(resizeState.initialPosition + offset);
270 nonwindowedState.output =
nullptr;
271 disconnect(nonwindowedState.sizeChangedConnection);
276 , m_xdgSurface(qobject_cast<QWaylandXdgSurface *>(item->shellSurface()))
277 , m_popup(m_xdgSurface->popup())
281 m_item->setSurface(m_xdgSurface->surface());
282 handleGeometryChanged();
284 connect(m_popup, &QWaylandXdgPopup::configuredGeometryChanged,
this, &XdgPopupIntegration::handleGeometryChanged);
285 connect(m_xdgSurface->shell(), &QWaylandXdgShell::popupCreated,
this, [item](QWaylandXdgPopup *popup, QWaylandXdgSurface *){
286 handlePopupCreated(item, popup);
292 if (m_item->view()->output()) {
293 const QPoint windowOffset = m_popup->parentXdgSurface()->windowGeometry().topLeft();
294 const QPoint surfacePosition = m_popup->unconstrainedPosition() + windowOffset;
295 const QPoint itemPosition = m_item->mapFromSurface(surfacePosition).toPoint();
298 m_item->moveItem()->setPosition(itemPosition);
300 qWarning() <<
"XdgPopupIntegration popup item without output" << m_item;
308#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.
Combined button and popup list for selecting options.
static void handlePopupCreated(QWaylandQuickShellSurfaceItem *parentItem, QWaylandXdgPopup *popup)