mirror of
https://github.com/linuxdeepin/dde-dock.git
synced 2025-06-03 00:15:21 +00:00
556 lines
18 KiB
C++
556 lines
18 KiB
C++
/*
|
|
* Copyright (C) 2011 ~ 2018 Deepin Technology Co., Ltd.
|
|
*
|
|
* Author: sbw <sbw@sbw.so>
|
|
*
|
|
* Maintainer: sbw <sbw@sbw.so>
|
|
*
|
|
* 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 3 of the License, or
|
|
* 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, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "docksettings.h"
|
|
#include "panel/mainpanel.h"
|
|
#include "item/appitem.h"
|
|
|
|
#include <QDebug>
|
|
#include <QX11Info>
|
|
|
|
#include <DApplication>
|
|
#include <QScreen>
|
|
|
|
#define ICON_SIZE_LARGE 48
|
|
#define ICON_SIZE_MEDIUM 36
|
|
#define ICON_SIZE_SMALL 30
|
|
#define FASHION_MODE_PADDING 30
|
|
|
|
DWIDGET_USE_NAMESPACE
|
|
|
|
extern const QPoint rawXPosition(const QPoint &scaledPos);
|
|
|
|
DockSettings::DockSettings(QWidget *parent)
|
|
: QObject(parent),
|
|
|
|
m_autoHide(true),
|
|
|
|
m_fashionModeAct(tr("Fashion Mode"), this),
|
|
m_efficientModeAct(tr("Efficient Mode"), this),
|
|
m_topPosAct(tr("Top"), this),
|
|
m_bottomPosAct(tr("Bottom"), this),
|
|
m_leftPosAct(tr("Left"), this),
|
|
m_rightPosAct(tr("Right"), this),
|
|
m_largeSizeAct(tr("Large"), this),
|
|
m_mediumSizeAct(tr("Medium"), this),
|
|
m_smallSizeAct(tr("Small"), this),
|
|
m_keepShownAct(tr("Keep Shown"), this),
|
|
m_keepHiddenAct(tr("Keep Hidden"), this),
|
|
m_smartHideAct(tr("Smart Hide"), this),
|
|
|
|
m_displayInter(new DBusDisplay(this)),
|
|
m_dockInter(new DBusDock(this)),
|
|
m_itemController(DockItemController::instance(this))
|
|
{
|
|
m_primaryRect = m_displayInter->primaryRect();
|
|
m_primaryRawRect = m_displayInter->primaryRawRect();
|
|
m_screenRawHeight = m_displayInter->screenRawHeight();
|
|
m_screenRawWidth = m_displayInter->screenRawWidth();
|
|
m_position = Dock::Position(m_dockInter->position());
|
|
m_displayMode = Dock::DisplayMode(m_dockInter->displayMode());
|
|
m_hideMode = Dock::HideMode(m_dockInter->hideMode());
|
|
m_hideState = Dock::HideState(m_dockInter->hideState());
|
|
m_iconSize = m_dockInter->iconSize();
|
|
AppItem::setIconBaseSize(m_iconSize * qApp->devicePixelRatio());
|
|
DockItem::setDockPosition(m_position);
|
|
qApp->setProperty(PROP_POSITION, QVariant::fromValue(m_position));
|
|
DockItem::setDockDisplayMode(m_displayMode);
|
|
qApp->setProperty(PROP_DISPLAY_MODE, QVariant::fromValue(m_displayMode));
|
|
|
|
m_fashionModeAct.setCheckable(true);
|
|
m_efficientModeAct.setCheckable(true);
|
|
m_topPosAct.setCheckable(true);
|
|
m_bottomPosAct.setCheckable(true);
|
|
m_leftPosAct.setCheckable(true);
|
|
m_rightPosAct.setCheckable(true);
|
|
m_largeSizeAct.setCheckable(true);
|
|
m_mediumSizeAct.setCheckable(true);
|
|
m_smallSizeAct.setCheckable(true);
|
|
m_keepShownAct.setCheckable(true);
|
|
m_keepHiddenAct.setCheckable(true);
|
|
m_smartHideAct.setCheckable(true);
|
|
|
|
WhiteMenu *modeSubMenu = new WhiteMenu(&m_settingsMenu);
|
|
modeSubMenu->addAction(&m_fashionModeAct);
|
|
modeSubMenu->addAction(&m_efficientModeAct);
|
|
QAction *modeSubMenuAct = new QAction(tr("Mode"), this);
|
|
modeSubMenuAct->setMenu(modeSubMenu);
|
|
|
|
WhiteMenu *locationSubMenu = new WhiteMenu(&m_settingsMenu);
|
|
locationSubMenu->addAction(&m_topPosAct);
|
|
locationSubMenu->addAction(&m_bottomPosAct);
|
|
locationSubMenu->addAction(&m_leftPosAct);
|
|
locationSubMenu->addAction(&m_rightPosAct);
|
|
QAction *locationSubMenuAct = new QAction(tr("Location"), this);
|
|
locationSubMenuAct->setMenu(locationSubMenu);
|
|
|
|
WhiteMenu *sizeSubMenu = new WhiteMenu(&m_settingsMenu);
|
|
sizeSubMenu->addAction(&m_largeSizeAct);
|
|
sizeSubMenu->addAction(&m_mediumSizeAct);
|
|
sizeSubMenu->addAction(&m_smallSizeAct);
|
|
QAction *sizeSubMenuAct = new QAction(tr("Size"), this);
|
|
sizeSubMenuAct->setMenu(sizeSubMenu);
|
|
|
|
WhiteMenu *statusSubMenu = new WhiteMenu(&m_settingsMenu);
|
|
statusSubMenu->addAction(&m_keepShownAct);
|
|
statusSubMenu->addAction(&m_keepHiddenAct);
|
|
statusSubMenu->addAction(&m_smartHideAct);
|
|
QAction *statusSubMenuAct = new QAction(tr("Status"), this);
|
|
statusSubMenuAct->setMenu(statusSubMenu);
|
|
|
|
m_hideSubMenu = new WhiteMenu(&m_settingsMenu);
|
|
QAction *hideSubMenuAct = new QAction(tr("Plugins"), this);
|
|
hideSubMenuAct->setMenu(m_hideSubMenu);
|
|
|
|
m_settingsMenu.addAction(modeSubMenuAct);
|
|
m_settingsMenu.addAction(locationSubMenuAct);
|
|
m_settingsMenu.addAction(sizeSubMenuAct);
|
|
m_settingsMenu.addAction(statusSubMenuAct);
|
|
m_settingsMenu.addAction(hideSubMenuAct);
|
|
m_settingsMenu.setTitle("Settings Menu");
|
|
|
|
connect(&m_settingsMenu, &WhiteMenu::triggered, this, &DockSettings::menuActionClicked);
|
|
connect(m_dockInter, &DBusDock::PositionChanged, this, &DockSettings::onPositionChanged);
|
|
connect(m_dockInter, &DBusDock::IconSizeChanged, this, &DockSettings::iconSizeChanged);
|
|
connect(m_dockInter, &DBusDock::DisplayModeChanged, this, &DockSettings::onDisplayModeChanged);
|
|
connect(m_dockInter, &DBusDock::HideModeChanged, this, &DockSettings::hideModeChanged, Qt::QueuedConnection);
|
|
connect(m_dockInter, &DBusDock::HideStateChanged, this, &DockSettings::hideStateChanged);
|
|
connect(m_dockInter, &DBusDock::ServiceRestarted, this, &DockSettings::resetFrontendGeometry);
|
|
|
|
connect(m_itemController, &DockItemController::itemInserted, this, &DockSettings::dockItemCountChanged, Qt::QueuedConnection);
|
|
connect(m_itemController, &DockItemController::itemRemoved, this, &DockSettings::dockItemCountChanged, Qt::QueuedConnection);
|
|
|
|
connect(m_displayInter, &DBusDisplay::PrimaryRectChanged, this, &DockSettings::primaryScreenChanged, Qt::QueuedConnection);
|
|
connect(m_displayInter, &DBusDisplay::ScreenHeightChanged, this, &DockSettings::primaryScreenChanged, Qt::QueuedConnection);
|
|
connect(m_displayInter, &DBusDisplay::ScreenWidthChanged, this, &DockSettings::primaryScreenChanged, Qt::QueuedConnection);
|
|
|
|
DApplication *app = qobject_cast<DApplication*>(qApp);
|
|
if (app) {
|
|
connect(app, &DApplication::iconThemeChanged, this, &DockSettings::gtkIconThemeChanged);
|
|
}
|
|
|
|
calculateWindowConfig();
|
|
updateForbidPostions();
|
|
resetFrontendGeometry();
|
|
}
|
|
|
|
const QSize DockSettings::panelSize() const
|
|
{
|
|
switch (m_position)
|
|
{
|
|
case Top:
|
|
case Bottom:
|
|
return m_mainWindowSize - QSize(0, WINDOW_OVERFLOW);
|
|
case Left:
|
|
case Right:
|
|
return m_mainWindowSize - QSize(WINDOW_OVERFLOW, 0);
|
|
}
|
|
|
|
Q_UNREACHABLE();
|
|
}
|
|
|
|
const QRect DockSettings::windowRect(const Position position, const bool hide) const
|
|
{
|
|
QSize size = m_mainWindowSize;
|
|
if (hide)
|
|
{
|
|
switch (position)
|
|
{
|
|
case Top:
|
|
case Bottom: size.setHeight(2); break;
|
|
case Left:
|
|
case Right: size.setWidth(2); break;
|
|
}
|
|
}
|
|
|
|
const QRect primaryRect = m_primaryRect;
|
|
const int offsetX = (primaryRect.width() - size.width()) / 2;
|
|
const int offsetY = (primaryRect.height() - size.height()) / 2;
|
|
|
|
QPoint p(0, 0);
|
|
switch (position)
|
|
{
|
|
case Top:
|
|
p = QPoint(offsetX, -WINDOW_OVERFLOW); break;
|
|
case Left:
|
|
p = QPoint(-WINDOW_OVERFLOW, offsetY); break;
|
|
case Right:
|
|
p = QPoint(primaryRect.width() - size.width() + WINDOW_OVERFLOW, offsetY); break;
|
|
case Bottom:
|
|
p = QPoint(offsetX, primaryRect.height() - size.height() + WINDOW_OVERFLOW); break;
|
|
default:Q_UNREACHABLE();
|
|
}
|
|
|
|
return QRect(primaryRect.topLeft() + p, size);
|
|
}
|
|
|
|
void DockSettings::showDockSettingsMenu()
|
|
{
|
|
m_autoHide = false;
|
|
|
|
qDeleteAll(m_hideSubMenu->actions());
|
|
for (auto *p : m_itemController->pluginList())
|
|
{
|
|
if (!p->pluginIsAllowDisable())
|
|
continue;
|
|
|
|
const bool enable = !p->pluginIsDisable();
|
|
const QString &name = p->pluginName();
|
|
const QString &display = p->pluginDisplayName();
|
|
|
|
QAction *act = new QAction(display, this);
|
|
act->setCheckable(true);
|
|
act->setChecked(enable);
|
|
act->setData(name);
|
|
|
|
m_hideSubMenu->addAction(act);
|
|
}
|
|
|
|
m_fashionModeAct.setChecked(m_displayMode == Fashion);
|
|
m_efficientModeAct.setChecked(m_displayMode == Efficient);
|
|
m_topPosAct.setChecked(m_position == Top);
|
|
m_topPosAct.setEnabled(!m_forbidPositions.contains(Top));
|
|
m_bottomPosAct.setChecked(m_position == Bottom);
|
|
m_bottomPosAct.setEnabled(!m_forbidPositions.contains(Bottom));
|
|
m_leftPosAct.setChecked(m_position == Left);
|
|
m_leftPosAct.setEnabled(!m_forbidPositions.contains(Left));
|
|
m_rightPosAct.setChecked(m_position == Right);
|
|
m_rightPosAct.setEnabled(!m_forbidPositions.contains(Right));
|
|
m_largeSizeAct.setChecked(m_iconSize == ICON_SIZE_LARGE);
|
|
m_mediumSizeAct.setChecked(m_iconSize == ICON_SIZE_MEDIUM);
|
|
m_smallSizeAct.setChecked(m_iconSize == ICON_SIZE_SMALL);
|
|
m_keepShownAct.setChecked(m_hideMode == KeepShowing);
|
|
m_keepHiddenAct.setChecked(m_hideMode == KeepHidden);
|
|
m_smartHideAct.setChecked(m_hideMode == SmartHide);
|
|
|
|
|
|
m_settingsMenu.exec(QCursor::pos());
|
|
|
|
setAutoHide(true);
|
|
}
|
|
|
|
void DockSettings::updateGeometry()
|
|
{
|
|
|
|
}
|
|
|
|
void DockSettings::setAutoHide(const bool autoHide)
|
|
{
|
|
if (m_autoHide == autoHide)
|
|
return;
|
|
|
|
m_autoHide = autoHide;
|
|
emit autoHideChanged(m_autoHide);
|
|
}
|
|
|
|
void DockSettings::menuActionClicked(QAction *action)
|
|
{
|
|
Q_ASSERT(action);
|
|
|
|
if (action == &m_fashionModeAct)
|
|
return m_dockInter->setDisplayMode(Fashion);
|
|
if (action == &m_efficientModeAct)
|
|
return m_dockInter->setDisplayMode(Efficient);
|
|
|
|
if (action == &m_topPosAct)
|
|
return m_dockInter->setPosition(Top);
|
|
if (action == &m_bottomPosAct)
|
|
return m_dockInter->setPosition(Bottom);
|
|
if (action == &m_leftPosAct)
|
|
return m_dockInter->setPosition(Left);
|
|
if (action == &m_rightPosAct)
|
|
return m_dockInter->setPosition(Right);
|
|
|
|
if (action == &m_largeSizeAct)
|
|
return m_dockInter->setIconSize(ICON_SIZE_LARGE);
|
|
if (action == &m_mediumSizeAct)
|
|
return m_dockInter->setIconSize(ICON_SIZE_MEDIUM);
|
|
if (action == &m_smallSizeAct)
|
|
return m_dockInter->setIconSize(ICON_SIZE_SMALL);
|
|
|
|
if (action == &m_keepShownAct)
|
|
return m_dockInter->setHideMode(KeepShowing);
|
|
if (action == &m_keepHiddenAct)
|
|
return m_dockInter->setHideMode(KeepHidden);
|
|
if (action == &m_smartHideAct)
|
|
return m_dockInter->setHideMode(SmartHide);
|
|
|
|
// check plugin hide menu.
|
|
const QString &data = action->data().toString();
|
|
if (data.isEmpty())
|
|
return;
|
|
for (auto *p : m_itemController->pluginList())
|
|
{
|
|
if (p->pluginName() == data)
|
|
return p->pluginStateSwitched();
|
|
}
|
|
}
|
|
|
|
void DockSettings::onPositionChanged()
|
|
{
|
|
const Position prevPos = m_position;
|
|
const Position nextPos = Dock::Position(m_dockInter->position());
|
|
|
|
if (prevPos == nextPos)
|
|
return;
|
|
|
|
emit positionChanged(prevPos);
|
|
|
|
QTimer::singleShot(200, this, [this, nextPos] {
|
|
m_position = nextPos;
|
|
DockItem::setDockPosition(nextPos);
|
|
qApp->setProperty(PROP_POSITION, QVariant::fromValue(nextPos));
|
|
|
|
calculateWindowConfig();
|
|
|
|
m_itemController->refershItemsIcon();
|
|
});
|
|
}
|
|
|
|
void DockSettings::iconSizeChanged()
|
|
{
|
|
// qDebug() << Q_FUNC_INFO;
|
|
m_iconSize = m_dockInter->iconSize();
|
|
AppItem::setIconBaseSize(m_iconSize * qApp->devicePixelRatio());
|
|
|
|
calculateWindowConfig();
|
|
|
|
emit dataChanged();
|
|
}
|
|
|
|
void DockSettings::onDisplayModeChanged()
|
|
{
|
|
// qDebug() << Q_FUNC_INFO;
|
|
m_displayMode = Dock::DisplayMode(m_dockInter->displayMode());
|
|
DockItem::setDockDisplayMode(m_displayMode);
|
|
qApp->setProperty(PROP_DISPLAY_MODE, QVariant::fromValue(m_displayMode));
|
|
|
|
calculateWindowConfig();
|
|
|
|
emit dataChanged();
|
|
emit displayModeChanegd();
|
|
}
|
|
|
|
void DockSettings::hideModeChanged()
|
|
{
|
|
// qDebug() << Q_FUNC_INFO;
|
|
m_hideMode = Dock::HideMode(m_dockInter->hideMode());
|
|
|
|
emit windowHideModeChanged();
|
|
}
|
|
|
|
void DockSettings::hideStateChanged()
|
|
{
|
|
// qDebug() << Q_FUNC_INFO;
|
|
const Dock::HideState state = Dock::HideState(m_dockInter->hideState());
|
|
|
|
if (state == Dock::Unknown)
|
|
return;
|
|
|
|
m_hideState = state;
|
|
|
|
emit windowVisibleChanged();
|
|
}
|
|
|
|
void DockSettings::dockItemCountChanged()
|
|
{
|
|
if (m_displayMode == Dock::Efficient)
|
|
return;
|
|
|
|
calculateWindowConfig();
|
|
|
|
emit windowGeometryChanged();
|
|
}
|
|
|
|
void DockSettings::primaryScreenChanged()
|
|
{
|
|
// qDebug() << Q_FUNC_INFO;
|
|
m_primaryRect = m_displayInter->primaryRect();
|
|
m_primaryRawRect = m_displayInter->primaryRawRect();
|
|
m_screenRawHeight = m_displayInter->screenRawHeight();
|
|
m_screenRawWidth = m_displayInter->screenRawWidth();
|
|
|
|
calculateWindowConfig();
|
|
updateForbidPostions();
|
|
|
|
emit dataChanged();
|
|
}
|
|
|
|
void DockSettings::resetFrontendGeometry()
|
|
{
|
|
const QRect r = windowRect(m_position);
|
|
const qreal ratio = qApp->devicePixelRatio();
|
|
const QPoint p = rawXPosition(r.topLeft());
|
|
const uint w = r.width() * ratio;
|
|
const uint h = r.height() * ratio;
|
|
|
|
m_frontendRect = QRect(p.x(), p.y(), w, h);
|
|
m_dockInter->SetFrontendWindowRect(p.x(), p.y(), w, h);
|
|
}
|
|
|
|
bool DockSettings::test(const Position pos, const QList<QRect> &otherScreens) const
|
|
{
|
|
QRect maxStrut(0, 0, m_screenRawWidth, m_screenRawHeight);
|
|
switch (pos)
|
|
{
|
|
case Top:
|
|
maxStrut.setBottom(m_primaryRawRect.top() - 1);
|
|
maxStrut.setLeft(m_primaryRawRect.left());
|
|
maxStrut.setRight(m_primaryRawRect.right());
|
|
break;
|
|
case Bottom:
|
|
maxStrut.setTop(m_primaryRawRect.bottom() + 1);
|
|
maxStrut.setLeft(m_primaryRawRect.left());
|
|
maxStrut.setRight(m_primaryRawRect.right());
|
|
break;
|
|
case Left:
|
|
maxStrut.setRight(m_primaryRawRect.left() - 1);
|
|
maxStrut.setTop(m_primaryRawRect.top());
|
|
maxStrut.setBottom(m_primaryRawRect.bottom());
|
|
break;
|
|
case Right:
|
|
maxStrut.setLeft(m_primaryRawRect.right() + 1);
|
|
maxStrut.setTop(m_primaryRawRect.top());
|
|
maxStrut.setBottom(m_primaryRawRect.bottom());
|
|
break;
|
|
default:;
|
|
}
|
|
|
|
if (maxStrut.width() == 0 || maxStrut.height() == 0)
|
|
return true;
|
|
|
|
for (const auto &r : otherScreens)
|
|
if (maxStrut.intersects(r))
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
void DockSettings::updateForbidPostions()
|
|
{
|
|
qDebug() << Q_FUNC_INFO;
|
|
|
|
const auto &screens = qApp->screens();
|
|
if (screens.size() < 2)
|
|
return m_forbidPositions.clear();
|
|
|
|
QSet<Position> forbids;
|
|
QList<QRect> rawScreenRects;
|
|
for (auto *s : screens)
|
|
{
|
|
if (s == qApp->primaryScreen())
|
|
continue;
|
|
|
|
const QRect &g = s->geometry();
|
|
rawScreenRects << QRect(g.topLeft(), g.size() * s->devicePixelRatio());
|
|
}
|
|
|
|
if (!test(Top, rawScreenRects))
|
|
forbids << Top;
|
|
if (!test(Bottom, rawScreenRects))
|
|
forbids << Bottom;
|
|
if (!test(Left, rawScreenRects))
|
|
forbids << Left;
|
|
if (!test(Right, rawScreenRects))
|
|
forbids << Right;
|
|
|
|
m_forbidPositions = std::move(forbids);
|
|
}
|
|
|
|
void DockSettings::calculateWindowConfig()
|
|
{
|
|
// qDebug() << Q_FUNC_INFO;
|
|
const auto ratio = qApp->devicePixelRatio();
|
|
const int defaultHeight = std::round(AppItem::itemBaseHeight() / ratio);
|
|
const int defaultWidth = std::round(AppItem::itemBaseWidth() / ratio);
|
|
|
|
if (m_displayMode == Dock::Efficient)
|
|
{
|
|
switch (m_position)
|
|
{
|
|
case Top:
|
|
case Bottom:
|
|
m_mainWindowSize.setHeight(defaultHeight + PANEL_BORDER + WINDOW_OVERFLOW);
|
|
m_mainWindowSize.setWidth(m_primaryRect.width());
|
|
break;
|
|
|
|
case Left:
|
|
case Right:
|
|
m_mainWindowSize.setHeight(m_primaryRect.height());
|
|
m_mainWindowSize.setWidth(defaultWidth + PANEL_BORDER + WINDOW_OVERFLOW);
|
|
break;
|
|
|
|
default:
|
|
Q_ASSERT(false);
|
|
}
|
|
}
|
|
else if (m_displayMode == Dock::Fashion)
|
|
{
|
|
int visibleItemCount = 0;
|
|
const QList<DockItem *> itemList = m_itemController->itemList();
|
|
for (auto item : itemList)
|
|
{
|
|
switch (item->itemType())
|
|
{
|
|
case DockItem::Launcher:
|
|
case DockItem::App:
|
|
case DockItem::Plugins:
|
|
case DockItem::Placeholder:
|
|
++visibleItemCount;
|
|
break;
|
|
default:;
|
|
}
|
|
}
|
|
|
|
const int perfectWidth = visibleItemCount * defaultWidth + PANEL_BORDER * 2 + PANEL_PADDING * 2;
|
|
const int perfectHeight = visibleItemCount * defaultHeight + PANEL_BORDER * 2 + PANEL_PADDING * 2;
|
|
const int calcWidth = qMin(m_primaryRect.width() - FASHION_MODE_PADDING * 2, perfectWidth);
|
|
const int calcHeight = qMin(m_primaryRect.height() - FASHION_MODE_PADDING * 2, perfectHeight);
|
|
switch (m_position)
|
|
{
|
|
case Top:
|
|
case Bottom:
|
|
m_mainWindowSize.setHeight(defaultHeight + PANEL_BORDER);
|
|
m_mainWindowSize.setWidth(calcWidth);
|
|
break;
|
|
|
|
case Left:
|
|
case Right:
|
|
m_mainWindowSize.setHeight(calcHeight);
|
|
m_mainWindowSize.setWidth(defaultWidth + PANEL_BORDER);
|
|
break;
|
|
|
|
default:
|
|
Q_ASSERT(false);
|
|
}
|
|
} else {
|
|
Q_ASSERT(false);
|
|
}
|
|
|
|
resetFrontendGeometry();
|
|
}
|
|
|
|
void DockSettings::gtkIconThemeChanged()
|
|
{
|
|
qDebug() << Q_FUNC_INFO;
|
|
m_itemController->refershItemsIcon();
|
|
}
|