/* * Copyright 2013 Marco Martin * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "desktopview.h" #include "containmentconfigview.h" #include "shellcorona.h" #include "shellmanager.h" #include "krunner_interface.h" #include #include #include #include #include #include #include #include #include DesktopView::DesktopView(Plasma::Corona *corona, QScreen *targetScreen) : PlasmaQuick::ContainmentView(corona, 0), m_windowType(Desktop), m_shellSurface(nullptr) { if (targetScreen) { setScreen(targetScreen); } setTitle(corona->kPackage().metadata().name()); setIcon(QIcon::fromTheme(corona->kPackage().metadata().iconName())); engine()->rootContext()->setContextProperty("desktop", this); setSource(QUrl::fromLocalFile(corona->kPackage().filePath("views", "Desktop.qml"))); ensureWindowType(); adaptToScreen(); //For some reason, if I connect the method directly it doesn't get called, I think it's for the lack of argument connect(this, &QWindow::screenChanged, this, [=](QScreen*) { adaptToScreen(); ensureWindowType(); }); QObject::connect(corona, &Plasma::Corona::kPackageChanged, this, &DesktopView::coronaPackageChanged); connect(this, &DesktopView::sceneGraphInitialized, this, [this, corona]() { // check whether the GL Context supports OpenGL // Note: hasOpenGLShaderPrograms is broken, see QTBUG--39730 if (!QOpenGLShaderProgram::hasOpenGLShaderPrograms(openglContext())) { qWarning() << "GLSL not available, Plasma won't be functional"; QMetaObject::invokeMethod(corona, "showOpenGLNotCompatibleWarning", Qt::QueuedConnection); } }, Qt::DirectConnection); } DesktopView::~DesktopView() { } void DesktopView::showEvent(QShowEvent* e) { QQuickWindow::showEvent(e); ensureWindowType(); adaptToScreen(); } void DesktopView::adaptToScreen() { //This happens sometimes, when shutting down the process if (!screen() || m_oldScreen==screen()) { return; } // qDebug() << "adapting to screen" << screen()->name() << this; if ((m_windowType == Desktop || m_windowType == WindowedDesktop) && !ShellManager::s_forceWindowed) { setGeometry(screen()->geometry()); setMinimumSize(screen()->geometry().size()); setMaximumSize(screen()->geometry().size()); if(m_oldScreen) { disconnect(m_oldScreen.data(), &QScreen::geometryChanged, this, &DesktopView::screenGeometryChanged); } connect(screen(), &QScreen::geometryChanged, this, &DesktopView::screenGeometryChanged, Qt::UniqueConnection); } else { disconnect(screen(), &QScreen::geometryChanged, this, &DesktopView::screenGeometryChanged); } m_oldScreen = screen(); } DesktopView::WindowType DesktopView::windowType() const { return m_windowType; } void DesktopView::setWindowType(DesktopView::WindowType type) { if (m_windowType == type) { return; } m_windowType = type; ensureWindowType(); adaptToScreen(); emit windowTypeChanged(); } void DesktopView::ensureWindowType() { //This happens sometimes, when shutting down the process if (!screen()) { return; } if (m_windowType == Window || ShellManager::s_forceWindowed) { setFlags(Qt::Window); KWindowSystem::setType(winId(), NET::Normal); KWindowSystem::clearState(winId(), NET::FullScreen); setupWaylandIntegration(); if (m_shellSurface) { m_shellSurface->setRole(KWayland::Client::PlasmaShellSurface::Role::Normal); } } else if (m_windowType == Desktop) { setFlags(Qt::Window | Qt::FramelessWindowHint); KWindowSystem::setType(winId(), NET::Desktop); KWindowSystem::setState(winId(), NET::KeepBelow); setupWaylandIntegration(); if (m_shellSurface) { m_shellSurface->setRole(KWayland::Client::PlasmaShellSurface::Role::Desktop); } } else if (m_windowType == WindowedDesktop) { KWindowSystem::setType(winId(), NET::Normal); KWindowSystem::clearState(winId(), NET::FullScreen); setFlags(Qt::FramelessWindowHint | flags()); setupWaylandIntegration(); if (m_shellSurface) { m_shellSurface->setRole(KWayland::Client::PlasmaShellSurface::Role::Normal); } } else if (m_windowType == FullScreen) { setFlags(Qt::Window); KWindowSystem::setType(winId(), NET::Normal); KWindowSystem::setState(winId(), NET::FullScreen); setupWaylandIntegration(); if (m_shellSurface) { m_shellSurface->setRole(KWayland::Client::PlasmaShellSurface::Role::Normal); } } } DesktopView::SessionType DesktopView::sessionType() const { if (qobject_cast(corona())) { return ShellSession; } else { return ApplicationSession; } } bool DesktopView::event(QEvent *e) { if (e->type() == QEvent::KeyRelease) { QKeyEvent *ke = static_cast(e); if (KWindowSystem::showingDesktop() && ke->key() == Qt::Key_Escape) { ShellCorona *c = qobject_cast(corona()); if (c) { KWindowSystem::setShowingDesktop(false); } } } else if (e->type() == QEvent::FocusIn) { //FIXME: this should *not* be needed ensureWindowType(); } else if (e->type() == QEvent::FocusOut) { QObject *graphicObject = containment()->property("_plasma_graphicObject").value(); if (graphicObject) { graphicObject->setProperty("focus", false); } } return PlasmaQuick::ContainmentView::event(e); } void DesktopView::keyPressEvent(QKeyEvent *e) { // When a key is pressed on desktop when nothing else is active forward the key to krunner if (!e->modifiers() && activeFocusItem() == contentItem()) { const QString text = e->text().trimmed(); if (!text.isEmpty() && text[0].isPrint()) { const QString interface("org.kde.krunner"); org::kde::krunner::App krunner(interface, "/App", QDBusConnection::sessionBus()); krunner.query(text); e->accept(); } } ContainmentView::keyPressEvent(e); } void DesktopView::showConfigurationInterface(Plasma::Applet *applet) { if (m_configView) { if (m_configView->applet() != applet) { m_configView->hide(); m_configView->deleteLater(); } else { m_configView->requestActivate(); return; } } if (!applet || !applet->containment()) { return; } Plasma::Containment *cont = qobject_cast(applet); if (cont && cont->isContainment()) { m_configView = new ContainmentConfigView(cont); } else { m_configView = new PlasmaQuick::ConfigView(applet); } m_configView.data()->init(); m_configView.data()->show(); } void DesktopView::screenGeometryChanged(const QRect &geom) { setGeometry(screen()->geometry()); setMinimumSize(screen()->geometry().size()); setMaximumSize(screen()->geometry().size()); } void DesktopView::coronaPackageChanged(const KPackage::Package &package) { setContainment(0); setSource(QUrl::fromLocalFile(package.filePath("views", "Desktop.qml"))); } void DesktopView::setupWaylandIntegration() { if (m_shellSurface) { // already setup return; } if (ShellCorona *c = qobject_cast(corona())) { using namespace KWayland::Client; PlasmaShell *interface = c->waylandPlasmaShellInterface(); if (!interface) { return; } Surface *s = Surface::fromWindow(this); if (!s) { return; } m_shellSurface = interface->createSurface(s, this); } }