/* * 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 #include "panelview.h" #include "shellcorona.h" #include "panelshadows_p.h" #include "panelconfigview.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #if HAVE_X11 #include #include #endif static const int MINSIZE = 10; PanelView::PanelView(ShellCorona *corona, QScreen *targetScreen, QWindow *parent) : PlasmaQuick::View(corona, parent), m_offset(0), m_maxLength(0), m_minLength(0), m_distance(0), m_thickness(30), m_alignment(Qt::AlignLeft), m_corona(corona), m_visibilityMode(NormalPanel), m_background(0) { if (targetScreen) { setScreen(targetScreen); } setResizeMode(QQuickView::SizeRootObjectToView); setClearBeforeRendering(true); setColor(QColor(Qt::transparent)); setFlags(Qt::FramelessWindowHint|Qt::WindowDoesNotAcceptFocus); themeChanged(); connect(&m_theme, &Plasma::Theme::themeChanged, this, &PanelView::themeChanged); m_positionPaneltimer.setSingleShot(true); m_positionPaneltimer.setInterval(150); connect(&m_positionPaneltimer, &QTimer::timeout, this, [=] () { restore(); positionPanel(); }); m_unhideTimer.setSingleShot(true); m_unhideTimer.setInterval(500); connect(&m_unhideTimer, &QTimer::timeout, this, &PanelView::restoreAutoHide); connect(screen(), SIGNAL(geometryChanged(QRect)), &m_positionPaneltimer, SLOT(start())); connect(this, SIGNAL(locationChanged(Plasma::Types::Location)), &m_positionPaneltimer, SLOT(start())); connect(this, SIGNAL(containmentChanged()), &m_positionPaneltimer, SLOT(start())); connect(this, SIGNAL(containmentChanged()), this, SLOT(containmentChanged())); connect(this, &View::locationChanged, [=] () { emit m_corona->availableScreenRectChanged(); emit m_corona->availableScreenRegionChanged(); }); if (!m_corona->package().isValid()) { qWarning() << "Invalid home screen package"; } m_strutsTimer.setSingleShot(true); connect(&m_strutsTimer, &QTimer::timeout, this, &PanelView::updateStruts); connect(m_corona->screensConfiguration(), &KScreen::Config::outputAdded, this, &PanelView::updateStruts); connect(m_corona->screensConfiguration(), &KScreen::Config::outputRemoved, this, &PanelView::updateStruts); qmlRegisterType(); engine()->rootContext()->setContextProperty("panel", this); setSource(QUrl::fromLocalFile(m_corona->package().filePath("views", "Panel.qml"))); PanelShadows::self()->addWindow(this); } PanelView::~PanelView() { if (containment()) { m_corona->requestApplicationConfigSync(); } PanelShadows::self()->removeWindow(this); } KConfigGroup PanelView::config() const { if (!containment()) { return KConfigGroup(); } KConfigGroup views(m_corona->applicationConfig(), "PlasmaViews"); views = KConfigGroup(&views, QString("Panel %1").arg(containment()->id())); if (formFactor() == Plasma::Types::Vertical) { return KConfigGroup(&views, "Vertical" + QString::number(screen()->size().height())); //treat everything else as horizontal } else { return KConfigGroup(&views, "Horizontal" + QString::number(screen()->size().width())); } } void PanelView::maximize() { int length; if (containment()->formFactor() == Plasma::Types::Vertical) { length = screen()->size().height(); } else { length = screen()->size().width(); } setOffset(0); setMinimumLength(length); setMaximumLength(length); } Qt::Alignment PanelView::alignment() const { return m_alignment; } void PanelView::setAlignment(Qt::Alignment alignment) { if (m_alignment == alignment) { return; } m_alignment = alignment; config().writeEntry("alignment", (int)m_alignment); emit alignmentChanged(); positionPanel(); } int PanelView::offset() const { return m_offset; } void PanelView::setOffset(int offset) { if (m_offset == offset) { return; } if (formFactor() == Plasma::Types::Vertical) { if (offset + m_maxLength > screen()->size().height()) { setMaximumLength( -m_offset + screen()->size().height() ); } } else { if (offset + m_maxLength > screen()->size().width()) { setMaximumLength( -m_offset + screen()->size().width() ); } } m_offset = offset; config().writeEntry("offset", m_offset); positionPanel(); emit offsetChanged(); m_corona->requestApplicationConfigSync(); m_strutsTimer.start(STRUTSTIMERDELAY); emit m_corona->availableScreenRegionChanged(); } int PanelView::thickness() const { return m_thickness; } void PanelView::setThickness(int value) { if (value == thickness()) { return; } m_thickness = value; if (formFactor() == Plasma::Types::Vertical) { setMinimumWidth(value); setMaximumWidth(value); } else { setMinimumHeight(value); setMaximumHeight(value); } config().writeEntry("thickness", value); m_corona->requestApplicationConfigSync(); positionPanel(); emit m_corona->availableScreenRectChanged(); emit m_corona->availableScreenRegionChanged(); } int PanelView::length() const { int defaultLength = formFactor() == Plasma::Types::Vertical ? screen()->size().height() : screen()->size().width(); return config().isValid() ? config().readEntry("length", defaultLength) : defaultLength; } void PanelView::setLength(int value) { if (value == length()) { return; } config().writeEntry("length", value); m_corona->requestApplicationConfigSync(); positionPanel(); const int maxSize = screen()->size().width() - m_offset; if (containment()->formFactor() == Plasma::Types::Vertical) { resize(thickness(), qBound(MINSIZE, value, maxSize)); //Horizontal } else { resize(qBound(MINSIZE, value, maxSize), thickness()); } } int PanelView::maximumLength() const { return m_maxLength; } void PanelView::setMaximumLength(int length) { if (length == m_maxLength) { return; } if (m_minLength > length) { setMinimumLength(length); } if (formFactor() == Plasma::Types::Vertical) { setMaximumHeight(length); } else { setMaximumWidth(length); } config().writeEntry("maxLength", length); m_maxLength = length; emit maximumLengthChanged(); positionPanel(); m_corona->requestApplicationConfigSync(); } int PanelView::minimumLength() const { return m_minLength; } void PanelView::setMinimumLength(int length) { if (length == m_minLength) { return; } if (m_maxLength < length) { setMaximumLength(length); } if (formFactor() == Plasma::Types::Vertical) { setMinimumHeight(length); } else { setMinimumWidth(length); } config().writeEntry("minLength", length); m_minLength = length; positionPanel(); emit minimumLengthChanged(); m_corona->requestApplicationConfigSync(); } int PanelView::distance() const { return m_distance; } void PanelView::setDistance(int dist) { if (m_distance == dist) { return; } m_distance = dist; emit distanceChanged(); positionPanel(); } void PanelView::setVisibilityMode(PanelView::VisibilityMode mode) { m_visibilityMode = mode; if (mode == LetWindowsCover) { KWindowSystem::setState(winId(), NET::KeepBelow); } else { KWindowSystem::clearState(winId(), NET::KeepBelow); } disconnect(containment(), &Plasma::Applet::activated, this, &PanelView::showTemporarily); if (!(mode == NormalPanel || mode == WindowsGoBelow || mode == AutoHide)) { connect(containment(), &Plasma::Applet::activated, this, &PanelView::showTemporarily); } if (config().isValid()) config().writeEntry("panelVisibility", (int)mode); updateStruts(); emit visibilityModeChanged(); restoreAutoHide(); } PanelView::VisibilityMode PanelView::visibilityMode() const { return m_visibilityMode; } void PanelView::positionPanel() { if (!containment()) { return; } KWindowEffects::SlideFromLocation slideLocation = KWindowEffects::NoEdge; switch (containment()->location()) { case Plasma::Types::TopEdge: containment()->setFormFactor(Plasma::Types::Horizontal); slideLocation = KWindowEffects::TopEdge; break; case Plasma::Types::LeftEdge: containment()->setFormFactor(Plasma::Types::Vertical); slideLocation = KWindowEffects::LeftEdge; break; case Plasma::Types::RightEdge: containment()->setFormFactor(Plasma::Types::Vertical); slideLocation = KWindowEffects::RightEdge; break; case Plasma::Types::BottomEdge: default: containment()->setFormFactor(Plasma::Types::Horizontal); slideLocation = KWindowEffects::BottomEdge; break; } m_strutsTimer.stop(); m_strutsTimer.start(STRUTSTIMERDELAY); setMinimumSize(QSize(0, 0)); setMaximumSize(screen()->size()); setGeometry(geometryByDistance(m_distance)); if (formFactor() == Plasma::Types::Vertical) { setMinimumSize(QSize(thickness(), m_minLength)); setMaximumSize(QSize(thickness(), m_maxLength)); emit thicknessChanged(); emit lengthChanged(); } else { setMinimumSize(QSize(m_minLength, thickness())); setMaximumSize(QSize(m_maxLength, thickness())); emit thicknessChanged(); emit lengthChanged(); } KWindowEffects::slideWindow(winId(), slideLocation, -1); } QRect PanelView::geometryByDistance(int distance) const { QScreen *s = screen(); QPoint position; switch (containment()->location()) { case Plasma::Types::TopEdge: switch (m_alignment) { case Qt::AlignCenter: position = QPoint(QPoint(s->geometry().center().x(), s->geometry().top()) + QPoint(m_offset - size().width()/2, distance)); break; case Qt::AlignRight: position = QPoint(s->geometry().topRight() - QPoint(m_offset + size().width(), distance)); break; case Qt::AlignLeft: default: position = QPoint(s->geometry().topLeft() + QPoint(m_offset, distance)); } break; case Plasma::Types::LeftEdge: switch (m_alignment) { case Qt::AlignCenter: position = QPoint(QPoint(s->geometry().left(), s->geometry().center().y()) + QPoint(distance, m_offset)); break; case Qt::AlignRight: position = QPoint(s->geometry().bottomLeft() - QPoint(distance, m_offset + size().height())); break; case Qt::AlignLeft: default: position = QPoint(s->geometry().topLeft() + QPoint(distance, m_offset)); } break; case Plasma::Types::RightEdge: switch (m_alignment) { case Qt::AlignCenter: position = QPoint(QPoint(s->geometry().right(), s->geometry().center().y()) - QPoint(width() + distance, 0) + QPoint(0, m_offset - size().height()/2)); break; case Qt::AlignRight: position = QPoint(s->geometry().bottomRight() - QPoint(width() + distance, 0) - QPoint(0, m_offset + size().height())); break; case Qt::AlignLeft: default: position = QPoint(s->geometry().topRight() - QPoint(width() + distance, 0) + QPoint(0, m_offset)); } break; case Plasma::Types::BottomEdge: default: switch (m_alignment) { case Qt::AlignCenter: position = QPoint(QPoint(s->geometry().center().x(), s->geometry().bottom() - height() - distance) + QPoint(m_offset - size().width()/2, 1)); break; case Qt::AlignRight: position = QPoint(s->geometry().bottomRight() - QPoint(0, height() + distance) - QPoint(m_offset + size().width(), 1)); break; case Qt::AlignLeft: default: position = QPoint(s->geometry().bottomLeft() - QPoint(0, height() + distance) + QPoint(m_offset, 1)); } } return formFactor() == Plasma::Types::Vertical ? QRect(position, QSize(thickness(), length())) : QRect(position, QSize(length(), thickness())); } void PanelView::restore() { if (!containment()) { return; } m_offset = config().readEntry("offset", 0); if (m_alignment != Qt::AlignCenter) { m_offset = qMax(0, m_offset); } setAlignment((Qt::Alignment)config().readEntry("alignment", Qt::AlignLeft)); setThickness(config().readEntry("thickness", 30)); setMinimumSize(QSize(-1, -1)); //FIXME: an invalid size doesn't work with QWindows setMaximumSize(screen()->size()); if (containment()->formFactor() == Plasma::Types::Vertical) { m_maxLength = config().readEntry("maxLength", screen()->size().height()); m_minLength = config().readEntry("minLength", screen()->size().height()); const int maxSize = screen()->size().height() - m_offset; m_maxLength = qBound(MINSIZE, m_maxLength, maxSize); m_minLength = qBound(MINSIZE, m_minLength, maxSize); resize(thickness(), qBound(MINSIZE, length(), maxSize)); setMinimumHeight(m_minLength); setMaximumHeight(m_maxLength); //Horizontal } else { m_maxLength = config().readEntry("maxLength", screen()->size().width()); m_minLength = config().readEntry("minLength", screen()->size().width()); const int maxSize = screen()->size().width() - m_offset; m_maxLength = qBound(MINSIZE, m_maxLength, maxSize); m_minLength = qBound(MINSIZE, m_minLength, maxSize); resize(qBound(MINSIZE, length(), maxSize), thickness()); setMinimumWidth(m_minLength); setMaximumWidth(m_maxLength); } setVisibilityMode((VisibilityMode)config().readEntry("panelVisibility", (int)NormalPanel)); emit maximumLengthChanged(); emit minimumLengthChanged(); emit offsetChanged(); emit alignmentChanged(); } void PanelView::showConfigurationInterface(Plasma::Applet *applet) { if (!applet || !applet->containment()) { return; } Plasma::Containment *cont = qobject_cast(applet); if (m_panelConfigView && cont && cont->isContainment()) { if (m_panelConfigView.data()->isVisible()) { m_panelConfigView.data()->hide(); } else { m_panelConfigView.data()->show(); KWindowSystem::setState(m_panelConfigView.data()->winId(), NET::SkipTaskbar | NET::SkipPager); } return; } else if (m_panelConfigView) { if (m_panelConfigView->applet() == applet) { m_panelConfigView->show(); return; } else { m_panelConfigView->hide(); m_panelConfigView->deleteLater(); } } if (cont && cont->isContainment()) { m_panelConfigView = new PanelConfigView(cont, this); } else { m_panelConfigView = new PlasmaQuick::ConfigView(applet); } m_panelConfigView.data()->init(); m_panelConfigView.data()->show(); KWindowSystem::setState(m_panelConfigView.data()->winId(), NET::SkipTaskbar | NET::SkipPager); } void PanelView::restoreAutoHide() { setAutoHideEnabled(m_visibilityMode == AutoHide && (!containment() || containment()->status() < Plasma::Types::RequiresAttentionStatus)); } void PanelView::setAutoHideEnabled(bool enabled) { #if HAVE_X11 xcb_connection_t *c = QX11Info::connection(); if (!c) { return; } const QByteArray effectName = QByteArrayLiteral("_KDE_NET_WM_SCREEN_EDGE_SHOW"); xcb_intern_atom_cookie_t atomCookie = xcb_intern_atom_unchecked(c, false, effectName.length(), effectName.constData()); QScopedPointer atom(xcb_intern_atom_reply(c, atomCookie, NULL)); if (!atom) { return; } if (!enabled) { xcb_delete_property(c, winId(), atom->atom); return; } KWindowEffects::SlideFromLocation slideLocation = KWindowEffects::NoEdge; uint32_t value = 0; switch (location()) { case Plasma::Types::TopEdge: value = 0; slideLocation = KWindowEffects::TopEdge; break; case Plasma::Types::RightEdge: value = 1; slideLocation = KWindowEffects::RightEdge; break; case Plasma::Types::BottomEdge: value = 2; slideLocation = KWindowEffects::BottomEdge; break; case Plasma::Types::LeftEdge: value = 3; slideLocation = KWindowEffects::LeftEdge; break; case Plasma::Types::Floating: default: value = 4; break; } xcb_change_property(c, XCB_PROP_MODE_REPLACE, winId(), atom->atom, XCB_ATOM_CARDINAL, 32, 1, &value); KWindowEffects::slideWindow(winId(), slideLocation, -1); #endif } void PanelView::resizeEvent(QResizeEvent *ev) { updateMask(); //don't setGeometry() to meke really sure we aren't doing a resize loop setPosition(geometryByDistance(m_distance).topLeft()); PlasmaQuick::View::resizeEvent(ev); } void PanelView::moveEvent(QMoveEvent *ev) { updateMask(); PlasmaQuick::View::moveEvent(ev); } void PanelView::integrateScreen() { KWindowSystem::setOnAllDesktops(winId(), true); KWindowSystem::setType(winId(), NET::Dock); setVisibilityMode(m_visibilityMode); containment()->reactToScreenChange(); } void PanelView::showEvent(QShowEvent *event) { PanelShadows::self()->addWindow(this); PlasmaQuick::View::showEvent(event); integrateScreen(); //When the screen is set, the screen is recreated internally, so we need to //set anything that depends on the winId() connect(this, &QWindow::screenChanged, this, [=](QScreen* screen) { emit screenChangedProxy(screen); if (!screen) return; integrateScreen(); showTemporarily(); m_positionPaneltimer.start(); }); } bool PanelView::event(QEvent *e) { if (m_visibilityMode == AutoHide) { if (e->type() == QEvent::Enter) { m_unhideTimer.stop(); } else if (e->type() == QEvent::Leave) { m_unhideTimer.start(); } } return View::event(e); } void PanelView::updateMask() { if (KWindowSystem::compositingActive()) { setMask(QRegion()); } else { if (!m_background) { m_background = new Plasma::FrameSvg(this); m_background->setImagePath("widgets/panel-background"); } Plasma::FrameSvg::EnabledBorders borders = Plasma::FrameSvg::AllBorders; switch (location()) { case Plasma::Types::TopEdge: borders &= ~Plasma::FrameSvg::TopBorder; break; case Plasma::Types::LeftEdge: borders &= ~Plasma::FrameSvg::LeftBorder; break; case Plasma::Types::RightEdge: borders &= ~Plasma::FrameSvg::RightBorder; break; case Plasma::Types::BottomEdge: borders &= ~Plasma::FrameSvg::BottomBorder; break; default: break; } if (x() <= screen()->geometry().x()) { borders &= ~Plasma::FrameSvg::LeftBorder; } if (x() + width() >= screen()->geometry().x() + screen()->geometry().width()) { borders &= ~Plasma::FrameSvg::RightBorder; } if (y() <= screen()->geometry().y()) { borders &= ~Plasma::FrameSvg::TopBorder; } if (y() + height() >= screen()->geometry().y() + screen()->geometry().height()) { borders &= ~Plasma::FrameSvg::BottomBorder; } m_background->setEnabledBorders(borders); m_background->resizeFrame(size()); setMask(m_background->mask()); } } void PanelView::updateStruts() { if (!containment() || !screen()) { return; } NETExtendedStrut strut; if (m_visibilityMode == NormalPanel) { const QRect thisScreen = screen()->geometry(); const QRect wholeScreen = screen()->virtualGeometry(); //Extended struts against a screen edge near to another screen are really harmful, so windows maximized under the panel is a lesser pain //TODO: force "windows can cover" in those cases? const int numScreens = corona()->numScreens(); for (int i = 0; i < numScreens; ++i) { if (i == containment()->screen()) { continue; } const QRect otherScreen = corona()->screenGeometry(i); switch (location()) { case Plasma::Types::TopEdge: if (otherScreen.bottom() <= thisScreen.top()) { KWindowSystem::setExtendedStrut(winId(), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0); return; } break; case Plasma::Types::BottomEdge: if (otherScreen.top() >= thisScreen.bottom()) { KWindowSystem::setExtendedStrut(winId(), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0); return; } break; case Plasma::Types::RightEdge: if (otherScreen.left() >= thisScreen.right()) { KWindowSystem::setExtendedStrut(winId(), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0); return; } break; case Plasma::Types::LeftEdge: if (otherScreen.right() <= thisScreen.left()) { KWindowSystem::setExtendedStrut(winId(), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0); return; } break; default: return; } } // extended struts are to the combined screen geoms, not the single screen int leftOffset = thisScreen.x(); int rightOffset = wholeScreen.right() - thisScreen.right(); int bottomOffset = wholeScreen.bottom() - thisScreen.bottom(); // qDebug() << "screen l/r/b/t offsets are:" << leftOffset << rightOffset << bottomOffset << topOffset << location(); int topOffset = thisScreen.top(); switch (location()) { case Plasma::Types::TopEdge: strut.top_width = thickness() + topOffset; strut.top_start = x(); strut.top_end = x() + width() - 1; // qDebug() << "setting top edge to" << strut.top_width << strut.top_start << strut.top_end; break; case Plasma::Types::BottomEdge: strut.bottom_width = thickness() + bottomOffset; strut.bottom_start = x(); strut.bottom_end = x() + width() - 1; // qDebug() << "setting bottom edge to" << strut.bottom_width << strut.bottom_start << strut.bottom_end; break; case Plasma::Types::RightEdge: strut.right_width = thickness() + rightOffset; strut.right_start = y(); strut.right_end = y() + height() - 1; // qDebug() << "setting right edge to" << strut.right_width << strut.right_start << strut.right_end; break; case Plasma::Types::LeftEdge: strut.left_width = thickness() + leftOffset; strut.left_start = y(); strut.left_end = y() + height() - 1; // qDebug() << "setting left edge to" << strut.left_width << strut.left_start << strut.left_end; break; default: //qDebug() << "where are we?"; break; } } KWindowSystem::setExtendedStrut(winId(), strut.left_width, strut.left_start, strut.left_end, strut.right_width, strut.right_start, strut.right_end, strut.top_width, strut.top_start, strut.top_end, strut.bottom_width, strut.bottom_start, strut.bottom_end); } void PanelView::themeChanged() { KWindowEffects::enableBlurBehind(winId(), true); KWindowEffects::enableBackgroundContrast(winId(), m_theme.backgroundContrastEnabled(), m_theme.backgroundContrast(), m_theme.backgroundIntensity(), m_theme.backgroundSaturation()); updateMask(); } void PanelView::containmentChanged() { connect(containment(), SIGNAL(statusChanged(Plasma::Types::ItemStatus)), SLOT(statusChanged(Plasma::Types::ItemStatus))); connect(containment(), &QObject::destroyed, this, [=] (QObject *obj) { Q_UNUSED(obj) //containment()->destroyed() is true only when the user deleted it //so the config is to be thrown away, not during shutdown if (containment()->destroyed()) { KConfigGroup views(m_corona->applicationConfig(), "PlasmaViews"); for (auto grp : views.groupList()) { if (grp.contains(QRegExp("Panel " + QString::number(containment()->id()) + "$"))) { qDebug() << "Panel" << containment()->id() << "removed by user"; views.deleteGroup(grp); } views.sync(); } deleteLater(); } }, Qt::QueuedConnection); } void PanelView::statusChanged(Plasma::Types::ItemStatus status) { if (status == Plasma::Types::NeedsAttentionStatus) { showTemporarily(); } else { restoreAutoHide(); } } void PanelView::showTemporarily() { setAutoHideEnabled(false); QTimer * t = new QTimer(this); t->setSingleShot(true); t->setInterval(3000); connect(t, SIGNAL(timeout()), SLOT(restoreAutoHide())); connect(t, SIGNAL(timeout()), t, SLOT(deleteLater())); t->start(); } void PanelView::screenDestroyed(QObject* ) { // NOTE: this is overriding the screen destroyed slot, we need to do this because // otherwise Qt goes mental and starts moving our panels. See: // https://codereview.qt-project.org/#/c/88351/ // if(screen == this->screen()) { // DO NOTHING, panels are moved by ::removeScreen // } } #include "moc_panelview.cpp"