You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
lxqt-panel-packaging/panel/lxqtpanel.cpp

1359 lines
42 KiB

/* BEGIN_COMMON_COPYRIGHT_HEADER
* (c)LGPL2+
*
* LXDE-Qt - a lightweight, Qt based, desktop toolset
* http://razor-qt.org
*
* Copyright: 2010-2011 Razor team
* Authors:
* Alexander Sokoloff <sokoloff.a@gmail.com>
*
* This program or library is free software; you can redistribute it
* and/or modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library 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
* Lesser General Public License for more details.
* You should have received a copy of the GNU Lesser General
* Public License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301 USA
*
* END_COMMON_COPYRIGHT_HEADER */
#include "lxqtpanel.h"
#include "lxqtpanellimits.h"
#include "ilxqtpanelplugin.h"
#include "lxqtpanelapplication.h"
#include "lxqtpanellayout.h"
#include "config/configpaneldialog.h"
#include "popupmenu.h"
#include "plugin.h"
#include "panelpluginsmodel.h"
#include "windownotifier.h"
#include <LXQt/PluginInfo>
#include <QScreen>
#include <QWindow>
#include <QX11Info>
#include <QDebug>
#include <QString>
#include <QDesktopWidget>
#include <QMenu>
#include <QMessageBox>
#include <QDropEvent>
#include <XdgIcon>
#include <XdgDirs>
#include <KWindowSystem/KWindowSystem>
#include <KWindowSystem/NETWM>
// Turn on this to show the time required to load each plugin during startup
// #define DEBUG_PLUGIN_LOADTIME
#ifdef DEBUG_PLUGIN_LOADTIME
#include <QElapsedTimer>
#endif
// Config keys and groups
#define CFG_KEY_SCREENNUM "desktop"
#define CFG_KEY_POSITION "position"
#define CFG_KEY_PANELSIZE "panelSize"
#define CFG_KEY_ICONSIZE "iconSize"
#define CFG_KEY_LINECNT "lineCount"
#define CFG_KEY_LENGTH "width"
#define CFG_KEY_PERCENT "width-percent"
#define CFG_KEY_ALIGNMENT "alignment"
#define CFG_KEY_FONTCOLOR "font-color"
#define CFG_KEY_BACKGROUNDCOLOR "background-color"
#define CFG_KEY_BACKGROUNDIMAGE "background-image"
#define CFG_KEY_OPACITY "opacity"
#define CFG_KEY_RESERVESPACE "reserve-space"
#define CFG_KEY_PLUGINS "plugins"
#define CFG_KEY_HIDABLE "hidable"
#define CFG_KEY_ANIMATION "animation-duration"
#define CFG_KEY_SHOW_DELAY "show-delay"
#define CFG_KEY_LOCKPANEL "lockPanel"
/************************************************
Returns the Position by the string.
String is one of "Top", "Left", "Bottom", "Right", string is not case sensitive.
If the string is not correct, returns defaultValue.
************************************************/
ILXQtPanel::Position LXQtPanel::strToPosition(const QString& str, ILXQtPanel::Position defaultValue)
{
if (str.toUpper() == "TOP") return LXQtPanel::PositionTop;
if (str.toUpper() == "LEFT") return LXQtPanel::PositionLeft;
if (str.toUpper() == "RIGHT") return LXQtPanel::PositionRight;
if (str.toUpper() == "BOTTOM") return LXQtPanel::PositionBottom;
return defaultValue;
}
/************************************************
Return string representation of the position
************************************************/
QString LXQtPanel::positionToStr(ILXQtPanel::Position position)
{
switch (position)
{
case LXQtPanel::PositionTop:
return QString("Top");
case LXQtPanel::PositionLeft:
return QString("Left");
case LXQtPanel::PositionRight:
return QString("Right");
case LXQtPanel::PositionBottom:
return QString("Bottom");
}
return QString();
}
/************************************************
************************************************/
LXQtPanel::LXQtPanel(const QString &configGroup, LXQt::Settings *settings, QWidget *parent) :
QFrame(parent),
mSettings(settings),
mConfigGroup(configGroup),
mPlugins{nullptr},
mStandaloneWindows{new WindowNotifier},
mPanelSize(0),
mIconSize(0),
mLineCount(0),
mLength(0),
mAlignment(AlignmentLeft),
mPosition(ILXQtPanel::PositionBottom),
mScreenNum(0), //whatever (avoid conditional on uninitialized value)
mActualScreenNum(0),
mHidable(false),
mHidden(false),
mAnimationTime(0),
mReserveSpace(true),
mAnimation(nullptr),
mLockPanel(false)
{
//You can find information about the flags and widget attributes in your
//Qt documentation or at http://doc.qt.io/qt-5/qt.html
//Qt::FramelessWindowHint = Produces a borderless window. The user cannot
//move or resize a borderless window via the window system. On X11, ...
//Qt::WindowStaysOnTopHint = Informs the window system that the window
//should stay on top of all other windows. Note that on ...
Qt::WindowFlags flags = Qt::FramelessWindowHint | Qt::WindowStaysOnTopHint;
// NOTE: by PCMan:
// In Qt 4, the window is not activated if it has Qt::WA_X11NetWmWindowTypeDock.
// Since Qt 5, the default behaviour is changed. A window is always activated on mouse click.
// Please see the source code of Qt5: src/plugins/platforms/xcb/qxcbwindow.cpp.
// void QXcbWindow::handleButtonPressEvent(const xcb_button_press_event_t *event)
// This new behaviour caused lxqt bug #161 - Cannot minimize windows from panel 1 when two task managers are open
// Besides, this breaks minimizing or restoring windows when clicking on the taskbar buttons.
// To workaround this regression bug, we need to add this window flag here.
// However, since the panel gets no keyboard focus, this may decrease accessibility since
// it's not possible to use the panel with keyboards. We need to find a better solution later.
flags |= Qt::WindowDoesNotAcceptFocus;
setWindowFlags(flags);
//Adds _NET_WM_WINDOW_TYPE_DOCK to the window's _NET_WM_WINDOW_TYPE X11 window property. See http://standards.freedesktop.org/wm-spec/ for more details.
setAttribute(Qt::WA_X11NetWmWindowTypeDock);
//Enables tooltips for inactive windows.
setAttribute(Qt::WA_AlwaysShowToolTips);
//Indicates that the widget should have a translucent background, i.e., any non-opaque regions of the widgets will be translucent because the widget will have an alpha channel. Setting this ...
setAttribute(Qt::WA_TranslucentBackground);
//Allows data from drag and drop operations to be dropped onto the widget (see QWidget::setAcceptDrops()).
setAttribute(Qt::WA_AcceptDrops);
setWindowTitle("LXQt Panel");
setObjectName(QString("LXQtPanel %1").arg(configGroup));
//LXQtPanel (inherits QFrame) -> lav (QGridLayout) -> LXQtPanelWidget (QFrame) -> LXQtPanelLayout
LXQtPanelWidget = new QFrame(this);
LXQtPanelWidget->setObjectName("BackgroundWidget");
QGridLayout* lav = new QGridLayout();
lav->setContentsMargins(0, 0, 0, 0);
setLayout(lav);
this->layout()->addWidget(LXQtPanelWidget);
mLayout = new LXQtPanelLayout(LXQtPanelWidget);
connect(mLayout, &LXQtPanelLayout::pluginMoved, this, &LXQtPanel::pluginMoved);
LXQtPanelWidget->setLayout(mLayout);
mLayout->setLineCount(mLineCount);
mDelaySave.setSingleShot(true);
mDelaySave.setInterval(SETTINGS_SAVE_DELAY);
connect(&mDelaySave, SIGNAL(timeout()), this, SLOT(saveSettings()));
mHideTimer.setSingleShot(true);
mHideTimer.setInterval(PANEL_HIDE_DELAY);
connect(&mHideTimer, SIGNAL(timeout()), this, SLOT(hidePanelWork()));
mShowDelayTimer.setSingleShot(true);
mShowDelayTimer.setInterval(PANEL_SHOW_DELAY);
connect(&mShowDelayTimer, &QTimer::timeout, [this] { showPanel(mAnimationTime > 0); });
connect(QApplication::desktop(), &QDesktopWidget::resized, this, &LXQtPanel::ensureVisible);
connect(QApplication::desktop(), &QDesktopWidget::screenCountChanged, this, &LXQtPanel::ensureVisible);
// connecting to QDesktopWidget::workAreaResized shouldn't be necessary,
// as we've already connceted to QDesktopWidget::resized, but it actually
// is. Read mode on https://github.com/lxde/lxqt-panel/pull/310
connect(QApplication::desktop(), &QDesktopWidget::workAreaResized,
this, &LXQtPanel::ensureVisible);
connect(LXQt::Settings::globalSettings(), SIGNAL(settingsChanged()), this, SLOT(update()));
connect(lxqtApp, SIGNAL(themeChanged()), this, SLOT(realign()));
connect(mStandaloneWindows.data(), &WindowNotifier::firstShown, [this] { showPanel(true); });
connect(mStandaloneWindows.data(), &WindowNotifier::lastHidden, this, &LXQtPanel::hidePanel);
readSettings();
ensureVisible();
loadPlugins();
show();
// show it the first time, despite setting
if (mHidable)
{
showPanel(false);
QTimer::singleShot(PANEL_HIDE_FIRST_TIME, this, SLOT(hidePanel()));
}
}
/************************************************
************************************************/
void LXQtPanel::readSettings()
{
// Read settings ......................................
mSettings->beginGroup(mConfigGroup);
// Let Hidability be the first thing we read
// so that every call to realign() is without side-effect
mHidable = mSettings->value(CFG_KEY_HIDABLE, mHidable).toBool();
mHidden = mHidable;
mAnimationTime = mSettings->value(CFG_KEY_ANIMATION, mAnimationTime).toInt();
mShowDelayTimer.setInterval(mSettings->value(CFG_KEY_SHOW_DELAY, mShowDelayTimer.interval()).toInt());
// By default we are using size & count from theme.
setPanelSize(mSettings->value(CFG_KEY_PANELSIZE, PANEL_DEFAULT_SIZE).toInt(), false);
setIconSize(mSettings->value(CFG_KEY_ICONSIZE, PANEL_DEFAULT_ICON_SIZE).toInt(), false);
setLineCount(mSettings->value(CFG_KEY_LINECNT, PANEL_DEFAULT_LINE_COUNT).toInt(), false);
setLength(mSettings->value(CFG_KEY_LENGTH, 100).toInt(),
mSettings->value(CFG_KEY_PERCENT, true).toBool(),
false);
mScreenNum = mSettings->value(CFG_KEY_SCREENNUM, QApplication::desktop()->primaryScreen()).toInt();
setPosition(mScreenNum,
strToPosition(mSettings->value(CFG_KEY_POSITION).toString(), PositionBottom),
false);
setAlignment(Alignment(mSettings->value(CFG_KEY_ALIGNMENT, mAlignment).toInt()), false);
QColor color = mSettings->value(CFG_KEY_FONTCOLOR, "").value<QColor>();
if (color.isValid())
setFontColor(color, true);
setOpacity(mSettings->value(CFG_KEY_OPACITY, 100).toInt(), true);
mReserveSpace = mSettings->value(CFG_KEY_RESERVESPACE, true).toBool();
color = mSettings->value(CFG_KEY_BACKGROUNDCOLOR, "").value<QColor>();
if (color.isValid())
setBackgroundColor(color, true);
QString image = mSettings->value(CFG_KEY_BACKGROUNDIMAGE, "").toString();
if (!image.isEmpty())
setBackgroundImage(image, false);
mLockPanel = mSettings->value(CFG_KEY_LOCKPANEL, false).toBool();
mSettings->endGroup();
}
/************************************************
************************************************/
void LXQtPanel::saveSettings(bool later)
{
mDelaySave.stop();
if (later)
{
mDelaySave.start();
return;
}
mSettings->beginGroup(mConfigGroup);
//Note: save/load of plugin names is completely handled by mPlugins object
//mSettings->setValue(CFG_KEY_PLUGINS, mPlugins->pluginNames());
mSettings->setValue(CFG_KEY_PANELSIZE, mPanelSize);
mSettings->setValue(CFG_KEY_ICONSIZE, mIconSize);
mSettings->setValue(CFG_KEY_LINECNT, mLineCount);
mSettings->setValue(CFG_KEY_LENGTH, mLength);
mSettings->setValue(CFG_KEY_PERCENT, mLengthInPercents);
mSettings->setValue(CFG_KEY_SCREENNUM, mScreenNum);
mSettings->setValue(CFG_KEY_POSITION, positionToStr(mPosition));
mSettings->setValue(CFG_KEY_ALIGNMENT, mAlignment);
mSettings->setValue(CFG_KEY_FONTCOLOR, mFontColor.isValid() ? mFontColor : QColor());
mSettings->setValue(CFG_KEY_BACKGROUNDCOLOR, mBackgroundColor.isValid() ? mBackgroundColor : QColor());
mSettings->setValue(CFG_KEY_BACKGROUNDIMAGE, QFileInfo(mBackgroundImage).exists() ? mBackgroundImage : QString());
mSettings->setValue(CFG_KEY_OPACITY, mOpacity);
mSettings->setValue(CFG_KEY_RESERVESPACE, mReserveSpace);
mSettings->setValue(CFG_KEY_HIDABLE, mHidable);
mSettings->setValue(CFG_KEY_ANIMATION, mAnimationTime);
mSettings->setValue(CFG_KEY_SHOW_DELAY, mShowDelayTimer.interval());
mSettings->setValue(CFG_KEY_LOCKPANEL, mLockPanel);
mSettings->endGroup();
}
/************************************************
************************************************/
void LXQtPanel::ensureVisible()
{
if (!canPlacedOn(mScreenNum, mPosition))
setPosition(findAvailableScreen(mPosition), mPosition, false);
else
mActualScreenNum = mScreenNum;
// the screen size might be changed
realign();
}
/************************************************
************************************************/
LXQtPanel::~LXQtPanel()
{
mLayout->setEnabled(false);
delete mAnimation;
delete mConfigDialog.data();
// do not save settings because of "user deleted panel" functionality saveSettings();
}
/************************************************
************************************************/
void LXQtPanel::show()
{
QWidget::show();
KWindowSystem::setOnDesktop(effectiveWinId(), NET::OnAllDesktops);
}
/************************************************
************************************************/
QStringList pluginDesktopDirs()
{
QStringList dirs;
dirs << QString(getenv("LXQT_PANEL_PLUGINS_DIR")).split(':', QString::SkipEmptyParts);
dirs << QString("%1/%2").arg(XdgDirs::dataHome(), "/lxqt/lxqt-panel");
dirs << PLUGIN_DESKTOPS_DIR;
return dirs;
}
/************************************************
************************************************/
void LXQtPanel::loadPlugins()
{
QString names_key(mConfigGroup);
names_key += '/';
names_key += QLatin1String(CFG_KEY_PLUGINS);
mPlugins.reset(new PanelPluginsModel(this, names_key, pluginDesktopDirs()));
connect(mPlugins.data(), &PanelPluginsModel::pluginAdded, mLayout, &LXQtPanelLayout::addPlugin);
connect(mPlugins.data(), &PanelPluginsModel::pluginMovedUp, mLayout, &LXQtPanelLayout::moveUpPlugin);
//reemit signals
connect(mPlugins.data(), &PanelPluginsModel::pluginAdded, this, &LXQtPanel::pluginAdded);
connect(mPlugins.data(), &PanelPluginsModel::pluginRemoved, this, &LXQtPanel::pluginRemoved);
for (auto const & plugin : mPlugins->plugins())
{
mLayout->addPlugin(plugin);
connect(plugin, &Plugin::dragLeft, [this] { mShowDelayTimer.stop(); hidePanel(); });
}
}
/************************************************
************************************************/
int LXQtPanel::getReserveDimension()
{
return mHidable ? PANEL_HIDE_SIZE : qMax(PANEL_MINIMUM_SIZE, mPanelSize);
}
void LXQtPanel::setPanelGeometry(bool animate)
{
const QRect currentScreen = QApplication::desktop()->screenGeometry(mActualScreenNum);
QRect rect;
if (isHorizontal())
{
// Horiz panel ***************************
rect.setHeight(qMax(PANEL_MINIMUM_SIZE, mPanelSize));
if (mLengthInPercents)
rect.setWidth(currentScreen.width() * mLength / 100.0);
else
{
if (mLength <= 0)
rect.setWidth(currentScreen.width() + mLength);
else
rect.setWidth(mLength);
}
rect.setWidth(qMax(rect.size().width(), mLayout->minimumSize().width()));
// Horiz ......................
switch (mAlignment)
{
case LXQtPanel::AlignmentLeft:
rect.moveLeft(currentScreen.left());
break;
case LXQtPanel::AlignmentCenter:
rect.moveCenter(currentScreen.center());
break;
case LXQtPanel::AlignmentRight:
rect.moveRight(currentScreen.right());
break;
}
// Vert .......................
if (mPosition == ILXQtPanel::PositionTop)
{
if (mHidden)
rect.moveBottom(currentScreen.top() + PANEL_HIDE_SIZE - 1);
else
rect.moveTop(currentScreen.top());
}
else
{
if (mHidden)
rect.moveTop(currentScreen.bottom() - PANEL_HIDE_SIZE + 1);
else
rect.moveBottom(currentScreen.bottom());
}
}
else
{
// Vert panel ***************************
rect.setWidth(qMax(PANEL_MINIMUM_SIZE, mPanelSize));
if (mLengthInPercents)
rect.setHeight(currentScreen.height() * mLength / 100.0);
else
{
if (mLength <= 0)
rect.setHeight(currentScreen.height() + mLength);
else
rect.setHeight(mLength);
}
rect.setHeight(qMax(rect.size().height(), mLayout->minimumSize().height()));
// Vert .......................
switch (mAlignment)
{
case LXQtPanel::AlignmentLeft:
rect.moveTop(currentScreen.top());
break;
case LXQtPanel::AlignmentCenter:
rect.moveCenter(currentScreen.center());
break;
case LXQtPanel::AlignmentRight:
rect.moveBottom(currentScreen.bottom());
break;
}
// Horiz ......................
if (mPosition == ILXQtPanel::PositionLeft)
{
if (mHidden)
rect.moveRight(currentScreen.left() + PANEL_HIDE_SIZE - 1);
else
rect.moveLeft(currentScreen.left());
}
else
{
if (mHidden)
rect.moveLeft(currentScreen.right() - PANEL_HIDE_SIZE + 1);
else
rect.moveRight(currentScreen.right());
}
}
if (rect != geometry())
{
setFixedSize(rect.size());
if (animate)
{
if (mAnimation == nullptr)
{
mAnimation = new QPropertyAnimation(this, "geometry");
mAnimation->setEasingCurve(QEasingCurve::Linear);
//Note: for hiding, the margins are set after animation is finished
connect(mAnimation, &QAbstractAnimation::finished, [this] { if (mHidden) setMargins(); });
}
mAnimation->setDuration(mAnimationTime);
mAnimation->setStartValue(geometry());
mAnimation->setEndValue(rect);
//Note: for showing-up, the margins are removed instantly
if (!mHidden)
setMargins();
mAnimation->start();
}
else
{
setMargins();
setGeometry(rect);
}
}
}
void LXQtPanel::setMargins()
{
if (mHidden)
{
if (isHorizontal())
{
if (mPosition == ILXQtPanel::PositionTop)
mLayout->setContentsMargins(0, 0, 0, PANEL_HIDE_SIZE);
else
mLayout->setContentsMargins(0, PANEL_HIDE_SIZE, 0, 0);
}
else
{
if (mPosition == ILXQtPanel::PositionLeft)
mLayout->setContentsMargins(0, 0, PANEL_HIDE_SIZE, 0);
else
mLayout->setContentsMargins(PANEL_HIDE_SIZE, 0, 0, 0);
}
}
else
mLayout->setContentsMargins(0, 0, 0, 0);
}
void LXQtPanel::realign()
{
if (!isVisible())
return;
#if 0
qDebug() << "** Realign *********************";
qDebug() << "PanelSize: " << mPanelSize;
qDebug() << "IconSize: " << mIconSize;
qDebug() << "LineCount: " << mLineCount;
qDebug() << "Length: " << mLength << (mLengthInPercents ? "%" : "px");
qDebug() << "Alignment: " << (mAlignment == 0 ? "center" : (mAlignment < 0 ? "left" : "right"));
qDebug() << "Position: " << positionToStr(mPosition) << "on" << mScreenNum;
qDebug() << "Plugins count: " << mPlugins.count();
#endif
setPanelGeometry();
// Reserve our space on the screen ..........
// It's possible that our geometry is not changed, but screen resolution is changed,
// so resetting WM_STRUT is still needed. To make it simple, we always do it.
updateWmStrut();
}
// Update the _NET_WM_PARTIAL_STRUT and _NET_WM_STRUT properties for the window
void LXQtPanel::updateWmStrut()
{
WId wid = effectiveWinId();
if(wid == 0 || !isVisible())
return;
if (mReserveSpace)
{
const QRect wholeScreen = QApplication::desktop()->geometry();
const QRect rect = geometry();
// NOTE: http://standards.freedesktop.org/wm-spec/wm-spec-latest.html
// Quote from the EWMH spec: " Note that the strut is relative to the screen edge, and not the edge of the xinerama monitor."
// So, we use the geometry of the whole screen to calculate the strut rather than using the geometry of individual monitors.
// Though the spec only mention Xinerama and did not mention XRandR, the rule should still be applied.
// At least openbox is implemented like this.
switch (mPosition)
{
case LXQtPanel::PositionTop:
KWindowSystem::setExtendedStrut(wid,
/* Left */ 0, 0, 0,
/* Right */ 0, 0, 0,
/* Top */ rect.top() + getReserveDimension(), rect.left(), rect.right(),
/* Bottom */ 0, 0, 0
);
break;
case LXQtPanel::PositionBottom:
KWindowSystem::setExtendedStrut(wid,
/* Left */ 0, 0, 0,
/* Right */ 0, 0, 0,
/* Top */ 0, 0, 0,
/* Bottom */ wholeScreen.bottom() - rect.bottom() + getReserveDimension(), rect.left(), rect.right()
);
break;
case LXQtPanel::PositionLeft:
KWindowSystem::setExtendedStrut(wid,
/* Left */ rect.left() + getReserveDimension(), rect.top(), rect.bottom(),
/* Right */ 0, 0, 0,
/* Top */ 0, 0, 0,
/* Bottom */ 0, 0, 0
);
break;
case LXQtPanel::PositionRight:
KWindowSystem::setExtendedStrut(wid,
/* Left */ 0, 0, 0,
/* Right */ wholeScreen.right() - rect.right() + getReserveDimension(), rect.top(), rect.bottom(),
/* Top */ 0, 0, 0,
/* Bottom */ 0, 0, 0
);
break;
}
} else
{
KWindowSystem::setExtendedStrut(wid,
/* Left */ 0, 0, 0,
/* Right */ 0, 0, 0,
/* Top */ 0, 0, 0,
/* Bottom */ 0, 0, 0
);
}
}
/************************************************
The panel can't be placed on boundary of two displays.
This function checks if the panel can be placed on the display
@screenNum on @position.
************************************************/
bool LXQtPanel::canPlacedOn(int screenNum, LXQtPanel::Position position)
{
QDesktopWidget* dw = QApplication::desktop();
switch (position)
{
case LXQtPanel::PositionTop:
for (int i = 0; i < dw->screenCount(); ++i)
if (dw->screenGeometry(i).bottom() < dw->screenGeometry(screenNum).top())
return false;
return true;
case LXQtPanel::PositionBottom:
for (int i = 0; i < dw->screenCount(); ++i)
if (dw->screenGeometry(i).top() > dw->screenGeometry(screenNum).bottom())
return false;
return true;
case LXQtPanel::PositionLeft:
for (int i = 0; i < dw->screenCount(); ++i)
if (dw->screenGeometry(i).right() < dw->screenGeometry(screenNum).left())
return false;
return true;
case LXQtPanel::PositionRight:
for (int i = 0; i < dw->screenCount(); ++i)
if (dw->screenGeometry(i).left() > dw->screenGeometry(screenNum).right())
return false;
return true;
}
return false;
}
/************************************************
************************************************/
int LXQtPanel::findAvailableScreen(LXQtPanel::Position position)
{
int current = mScreenNum;
for (int i = current; i < QApplication::desktop()->screenCount(); ++i)
if (canPlacedOn(i, position))
return i;
for (int i = 0; i < current; ++i)
if (canPlacedOn(i, position))
return i;
return 0;
}
/************************************************
************************************************/
void LXQtPanel::showConfigDialog()
{
if (mConfigDialog.isNull())
mConfigDialog = new ConfigPanelDialog(this, nullptr /*make it top level window*/);
mConfigDialog->showConfigPanelPage();
mStandaloneWindows->observeWindow(mConfigDialog.data());
mConfigDialog->show();
mConfigDialog->raise();
mConfigDialog->activateWindow();
WId wid = mConfigDialog->windowHandle()->winId();
KWindowSystem::activateWindow(wid);
KWindowSystem::setOnDesktop(wid, KWindowSystem::currentDesktop());
}
/************************************************
************************************************/
void LXQtPanel::showAddPluginDialog()
{
if (mConfigDialog.isNull())
mConfigDialog = new ConfigPanelDialog(this, nullptr /*make it top level window*/);
mConfigDialog->showConfigPluginsPage();
mStandaloneWindows->observeWindow(mConfigDialog.data());
mConfigDialog->show();
mConfigDialog->raise();
mConfigDialog->activateWindow();
WId wid = mConfigDialog->windowHandle()->winId();
KWindowSystem::activateWindow(wid);
KWindowSystem::setOnDesktop(wid, KWindowSystem::currentDesktop());
}
/************************************************
************************************************/
void LXQtPanel::updateStyleSheet()
{
QStringList sheet;
sheet << QString("Plugin > QAbstractButton, LXQtTray { qproperty-iconSize: %1px %1px; }").arg(mIconSize);
sheet << QString("Plugin > * > QAbstractButton, TrayIcon { qproperty-iconSize: %1px %1px; }").arg(mIconSize);
if (mFontColor.isValid())
sheet << QString("Plugin * { color: " + mFontColor.name() + "; }");
QString object = LXQtPanelWidget->objectName();
if (mBackgroundColor.isValid())
{
QString color = QString("%1, %2, %3, %4")
.arg(mBackgroundColor.red())
.arg(mBackgroundColor.green())
.arg(mBackgroundColor.blue())
.arg((float) mOpacity / 100);
sheet << QString("LXQtPanel #BackgroundWidget { background-color: rgba(" + color + "); }");
}
if (QFileInfo(mBackgroundImage).exists())
sheet << QString("LXQtPanel #BackgroundWidget { background-image: url('" + mBackgroundImage + "');}");
setStyleSheet(sheet.join("\n"));
}
/************************************************
************************************************/
void LXQtPanel::setPanelSize(int value, bool save)
{
if (mPanelSize != value)
{
mPanelSize = value;
realign();
if (save)
saveSettings(true);
}
}
/************************************************
************************************************/
void LXQtPanel::setIconSize(int value, bool save)
{
if (mIconSize != value)
{
mIconSize = value;
updateStyleSheet();
mLayout->setLineSize(mIconSize);
if (save)
saveSettings(true);
realign();
}
}
/************************************************
************************************************/
void LXQtPanel::setLineCount(int value, bool save)
{
if (mLineCount != value)
{
mLineCount = value;
mLayout->setEnabled(false);
mLayout->setLineCount(mLineCount);
mLayout->setEnabled(true);
if (save)
saveSettings(true);
realign();
}
}
/************************************************
************************************************/
void LXQtPanel::setLength(int length, bool inPercents, bool save)
{
if (mLength == length &&
mLengthInPercents == inPercents)
return;
mLength = length;
mLengthInPercents = inPercents;
if (save)
saveSettings(true);
realign();
}
/************************************************
************************************************/
void LXQtPanel::setPosition(int screen, ILXQtPanel::Position position, bool save)
{
if (mScreenNum == screen &&
mPosition == position)
return;
mActualScreenNum = screen;
mPosition = position;
mLayout->setPosition(mPosition);
if (save)
{
mScreenNum = screen;
saveSettings(true);
}
// Qt 5 adds a new class QScreen and add API for setting the screen of a QWindow.
// so we had better use it. However, without this, our program should still work
// as long as XRandR is used. Since XRandR combined all screens into a large virtual desktop
// every screen and their virtual siblings are actually on the same virtual desktop.
// So things still work if we don't set the screen correctly, but this is not the case
// for other backends, such as the upcoming wayland support. Hence it's better to set it.
if(windowHandle())
{
// QScreen* newScreen = qApp->screens().at(screen);
// QScreen* oldScreen = windowHandle()->screen();
// const bool shouldRecreate = windowHandle()->handle() && !(oldScreen && oldScreen->virtualSiblings().contains(newScreen));
// Q_ASSERT(shouldRecreate == false);
// NOTE: When you move a window to another screen, Qt 5 might recreate the window as needed
// But luckily, this never happen in XRandR, so Qt bug #40681 is not triggered here.
// (The only exception is when the old screen is destroyed, Qt always re-create the window and
// this corner case triggers #40681.)
// When using other kind of multihead settings, such as Xinerama, this might be different and
// unless Qt developers can fix their bug, we have no way to workaround that.
windowHandle()->setScreen(qApp->screens().at(screen));
}
realign();
}
/************************************************
*
************************************************/
void LXQtPanel::setAlignment(Alignment value, bool save)
{
if (mAlignment == value)
return;
mAlignment = value;
if (save)
saveSettings(true);
realign();
}
/************************************************
*
************************************************/
void LXQtPanel::setFontColor(QColor color, bool save)
{
mFontColor = color;
updateStyleSheet();
if (save)
saveSettings(true);
}
/************************************************
************************************************/
void LXQtPanel::setBackgroundColor(QColor color, bool save)
{
mBackgroundColor = color;
updateStyleSheet();
if (save)
saveSettings(true);
}
/************************************************
************************************************/
void LXQtPanel::setBackgroundImage(QString path, bool save)
{
mBackgroundImage = path;
updateStyleSheet();
if (save)
saveSettings(true);
}
/************************************************
*
************************************************/
void LXQtPanel::setOpacity(int opacity, bool save)
{
mOpacity = opacity;
updateStyleSheet();
if (save)
saveSettings(true);
}
/************************************************
*
************************************************/
void LXQtPanel::setReserveSpace(bool reserveSpace, bool save)
{
if (mReserveSpace == reserveSpace)
return;
mReserveSpace = reserveSpace;
if (save)
saveSettings(true);
updateWmStrut();
}
/************************************************
************************************************/
QRect LXQtPanel::globalGometry() const
{
return QRect(mapToGlobal(QPoint(0, 0)), this->size());
}
/************************************************
************************************************/
bool LXQtPanel::event(QEvent *event)
{
switch (event->type())
{
case QEvent::ContextMenu:
showPopupMenu();
break;
case QEvent::LayoutRequest:
emit realigned();
break;
case QEvent::WinIdChange:
{
// qDebug() << "WinIdChange" << hex << effectiveWinId();
if(effectiveWinId() == 0)
break;
// Sometimes Qt needs to re-create the underlying window of the widget and
// the winId() may be changed at runtime. So we need to reset all X11 properties
// when this happens.
qDebug() << "WinIdChange" << hex << effectiveWinId() << "handle" << windowHandle() << windowHandle()->screen();
// Qt::WA_X11NetWmWindowTypeDock becomes ineffective in Qt 5
// See QTBUG-39887: https://bugreports.qt-project.org/browse/QTBUG-39887
// Let's use KWindowSystem for that
KWindowSystem::setType(effectiveWinId(), NET::Dock);
updateWmStrut(); // reserve screen space for the panel
KWindowSystem::setOnAllDesktops(effectiveWinId(), true);
break;
}
case QEvent::DragEnter:
dynamic_cast<QDropEvent *>(event)->setDropAction(Qt::IgnoreAction);
event->accept();
//no break intentionally
case QEvent::Enter:
mShowDelayTimer.start();
break;
case QEvent::Leave:
case QEvent::DragLeave:
mShowDelayTimer.stop();
hidePanel();
break;
default:
break;
}
return QFrame::event(event);
}
/************************************************
************************************************/
void LXQtPanel::showEvent(QShowEvent *event)
{
QFrame::showEvent(event);
realign();
}
/************************************************
************************************************/
void LXQtPanel::showPopupMenu(Plugin *plugin)
{
PopupMenu * menu = new PopupMenu(tr("Panel"), this);
menu->setAttribute(Qt::WA_DeleteOnClose);
menu->setIcon(XdgIcon::fromTheme("configure-toolbars"));
// Plugin Menu ..............................
if (plugin)
{
QMenu *m = plugin->popupMenu();
if (m)
{
menu->addTitle(plugin->windowTitle());
for (auto const & action : m->actions())
{
action->setParent(menu);
action->setDisabled(mLockPanel);
menu->addAction(action);
}
delete m;
}
}
// Panel menu ...............................
menu->addTitle(QIcon(), tr("Panel"));
menu->addAction(XdgIcon::fromTheme(QLatin1String("configure")),
tr("Configure Panel"),
this, SLOT(showConfigDialog())
)->setDisabled(mLockPanel);
menu->addAction(XdgIcon::fromTheme("preferences-plugin"),
tr("Manage Widgets"),
this, SLOT(showAddPluginDialog())
)->setDisabled(mLockPanel);
LXQtPanelApplication *a = reinterpret_cast<LXQtPanelApplication*>(qApp);
menu->addAction(XdgIcon::fromTheme(QLatin1String("list-add")),
tr("Add New Panel"),
a, SLOT(addNewPanel())
);
if (a->count() > 1)
{
menu->addAction(XdgIcon::fromTheme(QLatin1String("list-remove")),
tr("Remove Panel", "Menu Item"),
this, SLOT(userRequestForDeletion())
)->setDisabled(mLockPanel);
}
QAction * act_lock = menu->addAction(tr("Lock This Panel"));
act_lock->setCheckable(true);
act_lock->setChecked(mLockPanel);
connect(act_lock, &QAction::triggered, [this] { mLockPanel = !mLockPanel; saveSettings(false); });
#ifdef DEBUG
menu->addSeparator();
menu->addAction("Exit (debug only)", qApp, SLOT(quit()));
#endif
/* Note: in multihead & multipanel setup the QMenu::popup/exec places the window
* sometimes wrongly (it seems that this bug is somehow connected to misinterpretation
* of QDesktopWidget::availableGeometry)
*/
menu->setGeometry(calculatePopupWindowPos(QCursor::pos(), menu->sizeHint()));
willShowWindow(menu);
menu->show();
}
Plugin* LXQtPanel::findPlugin(const ILXQtPanelPlugin* iPlugin) const
{
for (auto const & plug : mPlugins->plugins())
if (plug->iPlugin() == iPlugin)
return plug;
return nullptr;
}
/************************************************
************************************************/
QRect LXQtPanel::calculatePopupWindowPos(QPoint const & absolutePos, QSize const & windowSize) const
{
int x = absolutePos.x(), y = absolutePos.y();
switch (position())
{
case ILXQtPanel::PositionTop:
y = globalGometry().bottom();
break;
case ILXQtPanel::PositionBottom:
y = globalGometry().top() - windowSize.height();
break;
case ILXQtPanel::PositionLeft:
x = globalGometry().right();
break;
case ILXQtPanel::PositionRight:
x = globalGometry().left() - windowSize.width();
break;
}
QRect res(QPoint(x, y), windowSize);
QRect screen = QApplication::desktop()->screenGeometry(this);
// NOTE: We cannot use AvailableGeometry() which returns the work area here because when in a
// multihead setup with different resolutions. In this case, the size of the work area is limited
// by the smallest monitor and may be much smaller than the current screen and we will place the
// menu at the wrong place. This is very bad for UX. So let's use the full size of the screen.
if (res.right() > screen.right())
res.moveRight(screen.right());
if (res.bottom() > screen.bottom())
res.moveBottom(screen.bottom());
if (res.left() < screen.left())
res.moveLeft(screen.left());
if (res.top() < screen.top())
res.moveTop(screen.top());
return res;
}
/************************************************
************************************************/
QRect LXQtPanel::calculatePopupWindowPos(const ILXQtPanelPlugin *plugin, const QSize &windowSize) const
{
Plugin *panel_plugin = findPlugin(plugin);
if (nullptr == panel_plugin)
{
qWarning() << Q_FUNC_INFO << "Wrong logic? Unable to find Plugin* for" << plugin << "known plugins follow...";
for (auto const & plug : mPlugins->plugins())
qWarning() << plug->iPlugin() << plug;
return QRect();
}
return calculatePopupWindowPos(panel_plugin->mapToGlobal(QPoint(0, 0)), windowSize);
}
/************************************************
************************************************/
void LXQtPanel::willShowWindow(QWidget * w)
{
mStandaloneWindows->observeWindow(w);
}
/************************************************
************************************************/
QString LXQtPanel::qssPosition() const
{
return positionToStr(position());
}
/************************************************
************************************************/
void LXQtPanel::pluginMoved(Plugin * plug)
{
//get new position of the moved plugin
bool found{false};
QString plug_is_before;
for (int i=0; i<mLayout->count(); ++i)
{
Plugin *plugin = qobject_cast<Plugin*>(mLayout->itemAt(i)->widget());
if (plugin)
{
if (found)
{
//we found our plugin in previous cycle -> is before this (or empty as last)
plug_is_before = plugin->settingsGroup();
break;
} else
found = (plug == plugin);
}
}
mPlugins->movePlugin(plug, plug_is_before);
}
/************************************************
************************************************/
void LXQtPanel::userRequestForDeletion()
{
const QMessageBox::StandardButton ret
= QMessageBox::warning(this, tr("Remove Panel", "Dialog Title") ,
tr("Removing a panel can not be undone.\nDo you want to remove this panel?"),
QMessageBox::Yes | QMessageBox::No);
if (ret != QMessageBox::Yes) {
return;
}
mSettings->beginGroup(mConfigGroup);
QStringList plugins = mSettings->value("plugins").toStringList();
mSettings->endGroup();
Q_FOREACH(QString i, plugins)
if (!i.isEmpty())
mSettings->remove(i);
mSettings->remove(mConfigGroup);
emit deletedByUser(this);
}
void LXQtPanel::showPanel(bool animate)
{
if (mHidable)
{
mHideTimer.stop();
if (mHidden)
{
mHidden = false;
setPanelGeometry(mAnimationTime > 0 && animate);
}
}
}
void LXQtPanel::hidePanel()
{
if (mHidable && !mHidden
&& !mStandaloneWindows->isAnyWindowShown()
)
mHideTimer.start();
}
void LXQtPanel::hidePanelWork()
{
if (!geometry().contains(QCursor::pos()))
{
if (!mStandaloneWindows->isAnyWindowShown())
{
mHidden = true;
setPanelGeometry(mAnimationTime > 0);
} else
{
mHideTimer.start();
}
}
}
void LXQtPanel::setHidable(bool hidable, bool save)
{
if (mHidable == hidable)
return;
mHidable = hidable;
if (save)
saveSettings(true);
realign();
}
void LXQtPanel::setAnimationTime(int animationTime, bool save)
{
if (mAnimationTime == animationTime)
return;
mAnimationTime = animationTime;
if (save)
saveSettings(true);
}
void LXQtPanel::setShowDelay(int showDelay, bool save)
{
if (mShowDelayTimer.interval() == showDelay)
return;
mShowDelayTimer.setInterval(showDelay);
if (save)
saveSettings(true);
}
bool LXQtPanel::isPluginSingletonAndRunnig(QString const & pluginId) const
{
Plugin const * plugin = mPlugins->pluginByID(pluginId);
if (nullptr == plugin)
return false;
else
return plugin->iPlugin()->flags().testFlag(ILXQtPanelPlugin::SingleInstance);
}