/**************************************************************************** ** ** Copyright (C) 2014-2015 Pier Luigi Fiorini ** Copyright (C) 2015 The Qt Company Ltd. ** Contact: http://www.qt.io/licensing/ ** ** This file is part of the Qt Compositor. ** ** $QT_BEGIN_LICENSE:BSD$ ** You may use this file under the terms of the BSD license as follows: ** ** "Redistribution and use in source and binary forms, with or without ** modification, are permitted provided that the following conditions are ** met: ** * Redistributions of source code must retain the above copyright ** notice, this list of conditions and the following disclaimer. ** * Redistributions in binary form must reproduce the above copyright ** notice, this list of conditions and the following disclaimer in ** the documentation and/or other materials provided with the ** distribution. ** * Neither the name of The Qt Company Ltd nor the names of its ** contributors may be used to endorse or promote products derived ** from this software without specific prior written permission. ** ** ** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS ** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT ** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR ** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT ** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, ** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT ** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, ** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY ** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT ** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE ** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "qwaylandsurface.h" #include #include "wayland_wrapper/qwlsurface_p.h" #include "wayland_wrapper/qwlextendedsurface_p.h" #include "wayland_wrapper/qwlsubsurface_p.h" #include "wayland_wrapper/qwlcompositor_p.h" #include "wayland_wrapper/qwlshellsurface_p.h" #include "wayland_wrapper/qwlinputdevice_p.h" #include "wayland_wrapper/qwldatadevice_p.h" #include "wayland_wrapper/qwldatadevicemanager_p.h" #include "qwaylandcompositor.h" #include "qwaylandclient.h" #include "qwaylandsurface_p.h" #include "qwaylandbufferref.h" #include "qwaylandsurfaceinterface.h" #include #include QT_BEGIN_NAMESPACE const QEvent::Type QWaylandSurfaceEnterEvent::WaylandSurfaceEnter = (QEvent::Type)QEvent::registerEventType(); const QEvent::Type QWaylandSurfaceLeaveEvent::WaylandSurfaceLeave = (QEvent::Type)QEvent::registerEventType(); QWaylandSurfacePrivate::QWaylandSurfacePrivate(wl_client *wlClient, quint32 id, int version, QWaylandCompositor *compositor, QWaylandSurface *surface) : QtWayland::Surface(wlClient, id, version, compositor, surface) , closing(false) , refCount(1) , client(QWaylandClient::fromWlClient(wlClient)) , windowType(QWaylandSurface::WindowType::None) {} class QWaylandSurfaceEnterEventPrivate { public: QWaylandSurfaceEnterEventPrivate(QWaylandOutput *_output) : output(_output) { } QWaylandOutput *output; }; QWaylandSurfaceEnterEvent::QWaylandSurfaceEnterEvent(QWaylandOutput *output) : QEvent(WaylandSurfaceEnter) , d(new QWaylandSurfaceEnterEventPrivate(output)) { } QWaylandSurfaceEnterEvent::~QWaylandSurfaceEnterEvent() { delete d; } QWaylandOutput *QWaylandSurfaceEnterEvent::output() const { return d->output; } class QWaylandSurfaceLeaveEventPrivate { public: QWaylandSurfaceLeaveEventPrivate(QWaylandOutput *_output) : output(_output) { } QWaylandOutput *output; }; QWaylandSurfaceLeaveEvent::QWaylandSurfaceLeaveEvent(QWaylandOutput *output) : QEvent(WaylandSurfaceLeave) , d(new QWaylandSurfaceLeaveEventPrivate(output)) { } QWaylandSurfaceLeaveEvent::~QWaylandSurfaceLeaveEvent() { delete d; } QWaylandOutput *QWaylandSurfaceLeaveEvent::output() const { return d->output; } QWaylandSurface::QWaylandSurface(wl_client *client, quint32 id, int version, QWaylandCompositor *compositor) : QObject(*new QWaylandSurfacePrivate(client, id, version, compositor, this)) { } QWaylandSurface::QWaylandSurface(QWaylandSurfacePrivate *dptr) : QObject(*dptr) { } QWaylandSurface::~QWaylandSurface() { Q_D(QWaylandSurface); qDeleteAll(d->interfaces); delete d->m_attacher; } QWaylandClient *QWaylandSurface::client() const { Q_D(const QWaylandSurface); if (d->isDestroyed() || !d->compositor()->clients().contains(d->client)) return Q_NULLPTR; return d->client; } QWaylandSurface *QWaylandSurface::parentSurface() const { Q_D(const QWaylandSurface); if (d->subSurface() && d->subSurface()->parent()) { return d->subSurface()->parent()->waylandSurface(); } return 0; } QLinkedList QWaylandSurface::subSurfaces() const { Q_D(const QWaylandSurface); if (d->subSurface()) { return d->subSurface()->subSurfaces(); } return QLinkedList(); } void QWaylandSurface::addInterface(QWaylandSurfaceInterface *iface) { Q_D(QWaylandSurface); d->interfaces.prepend(iface); } void QWaylandSurface::removeInterface(QWaylandSurfaceInterface *iface) { Q_D(QWaylandSurface); d->interfaces.removeOne(iface); } QWaylandSurface::Type QWaylandSurface::type() const { Q_D(const QWaylandSurface); return d->type(); } bool QWaylandSurface::isYInverted() const { Q_D(const QWaylandSurface); return d->isYInverted(); } bool QWaylandSurface::visible() const { return isMapped(); } bool QWaylandSurface::isMapped() const { Q_D(const QWaylandSurface); return d->mapped(); } QSize QWaylandSurface::size() const { Q_D(const QWaylandSurface); return d->size(); } void QWaylandSurface::requestSize(const QSize &size) { Q_D(QWaylandSurface); QWaylandSurfaceResizeOp op(size); if (!sendInterfaceOp(op)) { int id = wl_resource_get_id(d->resource()->handle); qWarning("No surface interface forwarded the resize request for this surface (wl_surface@%d).", id); } } Qt::ScreenOrientations QWaylandSurface::orientationUpdateMask() const { Q_D(const QWaylandSurface); if (!d->extendedSurface()) return Qt::PrimaryOrientation; return d->extendedSurface()->contentOrientationMask(); } Qt::ScreenOrientation QWaylandSurface::contentOrientation() const { Q_D(const QWaylandSurface); return d->contentOrientation(); } QWaylandSurface::WindowFlags QWaylandSurface::windowFlags() const { Q_D(const QWaylandSurface); if (!d->extendedSurface()) return QWaylandSurface::WindowFlags(0); return d->extendedSurface()->windowFlags(); } QWaylandSurface::WindowType QWaylandSurface::windowType() const { Q_D(const QWaylandSurface); return d->windowType; } QWaylandSurface *QWaylandSurface::transientParent() const { Q_D(const QWaylandSurface); return d->transientParent() ? d->transientParent()->waylandSurface() : 0; } QPointF QWaylandSurface::transientOffset() const { Q_D(const QWaylandSurface); return d->m_transientOffset; } QtWayland::Surface * QWaylandSurface::handle() { Q_D(QWaylandSurface); return d; } QVariantMap QWaylandSurface::windowProperties() const { Q_D(const QWaylandSurface); if (!d->extendedSurface()) return QVariantMap(); return d->extendedSurface()->windowProperties(); } void QWaylandSurface::setWindowProperty(const QString &name, const QVariant &value) { Q_D(QWaylandSurface); if (!d->extendedSurface()) return; d->extendedSurface()->setWindowProperty(name, value); } QWaylandCompositor *QWaylandSurface::compositor() const { Q_D(const QWaylandSurface); return d->compositor()->waylandCompositor(); } QWaylandOutput *QWaylandSurface::mainOutput() const { Q_D(const QWaylandSurface); // Returns the output that contains the most if not all // the surface (window managers will take care of setting // this, defaults to the first output) return d->mainOutput()->waylandOutput(); } void QWaylandSurface::setMainOutput(QWaylandOutput *mainOutput) { Q_D(QWaylandSurface); if (mainOutput) d->setMainOutput(mainOutput->handle()); } QList QWaylandSurface::outputs() const { Q_D(const QWaylandSurface); QList list; Q_FOREACH (QtWayland::Output *output, d->outputs()) list.append(output->waylandOutput()); return list; } QWindow::Visibility QWaylandSurface::visibility() const { Q_D(const QWaylandSurface); return d->m_visibility; } void QWaylandSurface::setVisibility(QWindow::Visibility v) { Q_D(QWaylandSurface); if (v == visibility()) return; d->m_visibility = v; QWaylandSurfaceSetVisibilityOp op(v); sendInterfaceOp(op); emit visibilityChanged(); } bool QWaylandSurface::sendInterfaceOp(QWaylandSurfaceOp &op) { Q_D(QWaylandSurface); foreach (QWaylandSurfaceInterface *iface, d->interfaces) { if (iface->runOperation(&op)) return true; } return false; } void QWaylandSurface::ping() { Q_D(QWaylandSurface); uint32_t serial = wl_display_next_serial(compositor()->waylandDisplay()); QWaylandSurfacePingOp op(serial); if (!sendInterfaceOp(op)) { int id = wl_resource_get_id(d->resource()->handle); qWarning("No surface interface forwarded the ping for this surface (wl_surface@%d).", id); } } void QWaylandSurface::sendOnScreenVisibilityChange(bool visible) { setVisibility(visible ? QWindow::AutomaticVisibility : QWindow::Hidden); } QString QWaylandSurface::className() const { Q_D(const QWaylandSurface); return d->className(); } QString QWaylandSurface::title() const { Q_D(const QWaylandSurface); return d->title(); } bool QWaylandSurface::hasInputPanelSurface() const { Q_D(const QWaylandSurface); return d->inputPanelSurface() != 0; } /*! * \return True if WL_SHELL_SURFACE_TRANSIENT_INACTIVE was set for this surface, meaning it should not receive keyboard focus. */ bool QWaylandSurface::transientInactive() const { Q_D(const QWaylandSurface); return d->transientInactive(); } bool QWaylandSurface::inputRegionContains(const QPoint &p) const { Q_D(const QWaylandSurface); return d->inputRegion().contains(p); } void QWaylandSurface::destroy() { Q_D(QWaylandSurface); if (--d->refCount == 0) compositor()->handle()->destroySurface(d); } void QWaylandSurface::destroySurface() { QWaylandSurfaceOp op(QWaylandSurfaceOp::Close); if (!sendInterfaceOp(op)) emit surfaceDestroyed(); } /*! Updates the surface with the compositor's retained clipboard selection. While this is done automatically when the surface receives keyboard focus, this function is useful for updating clients which do not have keyboard focus. */ void QWaylandSurface::updateSelection() { Q_D(QWaylandSurface); const QtWayland::InputDevice *inputDevice = d->compositor()->defaultInputDevice(); if (inputDevice) { const QtWayland::DataDevice *dataDevice = inputDevice->dataDevice(); if (dataDevice) { d->compositor()->dataDeviceManager()->offerRetainedSelection( dataDevice->resourceMap().value(d->resource()->client())->handle); } } } void QWaylandSurface::ref() { Q_D(QWaylandSurface); ++d->refCount; } void QWaylandSurface::setMapped(bool mapped) { Q_D(QWaylandSurface); d->setMapped(mapped); } void QWaylandSurface::setBufferAttacher(QWaylandBufferAttacher *attacher) { Q_D(QWaylandSurface); d->m_attacher = attacher; } QWaylandBufferAttacher *QWaylandSurface::bufferAttacher() const { Q_D(const QWaylandSurface); return d->m_attacher; } QList QWaylandSurface::views() const { Q_D(const QWaylandSurface); return d->views; } QList QWaylandSurface::interfaces() const { Q_D(const QWaylandSurface); return d->interfaces; } QWaylandSurface *QWaylandSurface::fromResource(::wl_resource *res) { QtWayland::Surface *s = QtWayland::Surface::fromResource(res); if (s) return s->waylandSurface(); return Q_NULLPTR; } void QWaylandSurfacePrivate::setTitle(const QString &title) { Q_Q(QWaylandSurface); if (m_title != title) { m_title = title; emit q->titleChanged(); } } void QWaylandSurfacePrivate::setClassName(const QString &className) { Q_Q(QWaylandSurface); if (m_className != className) { m_className = className; emit q->classNameChanged(); } } void QWaylandSurfacePrivate::setType(QWaylandSurface::WindowType type) { Q_Q(QWaylandSurface); if (windowType != type) { windowType = type; emit q->windowTypeChanged(type); } } QT_END_NAMESPACE