mirror of
https://github.com/KDE/latte-dock.git
synced 2025-01-01 01:17:51 +03:00
94914ee578
--if the app crashed and the user trys to reopen Latte then the Containments must first return to their Original Layouts. This code provides that and informs also the user during the startup.
955 lines
29 KiB
C++
955 lines
29 KiB
C++
/*
|
|
* Copyright 2016 Smith AR <audoban@openmaibox.org>
|
|
* Michail Vourlakos <mvourlakos@gmail.com>
|
|
*
|
|
* This file is part of Latte-Dock
|
|
*
|
|
* Latte-Dock 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.
|
|
*
|
|
* Latte-Dock 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 "dockcorona.h"
|
|
#include "dockview.h"
|
|
#include "packageplugins/shell/dockpackage.h"
|
|
#include "abstractwindowinterface.h"
|
|
#include "alternativeshelper.h"
|
|
#include "screenpool.h"
|
|
//dbus adaptor
|
|
#include "lattedockadaptor.h"
|
|
|
|
#include <QAction>
|
|
#include <QApplication>
|
|
#include <QScreen>
|
|
#include <QDBusConnection>
|
|
#include <QDebug>
|
|
#include <QDesktopWidget>
|
|
#include <QFile>
|
|
#include <QFontDatabase>
|
|
#include <QQmlContext>
|
|
|
|
#include <Plasma>
|
|
#include <Plasma/Corona>
|
|
#include <Plasma/Containment>
|
|
|
|
#include <KActionCollection>
|
|
#include <KPluginMetaData>
|
|
#include <KGlobalAccel>
|
|
#include <KLocalizedString>
|
|
#include <KPackage/Package>
|
|
#include <KPackage/PackageLoader>
|
|
#include <KAboutData>
|
|
#include <KActivities/Consumer>
|
|
|
|
#include <KWindowSystem>
|
|
#include <KWayland/Client/connection_thread.h>
|
|
#include <KWayland/Client/registry.h>
|
|
#include <KWayland/Client/plasmashell.h>
|
|
|
|
namespace Latte {
|
|
|
|
DockCorona::DockCorona(bool defaultLayoutOnStartup, QString layoutNameOnStartUp, QObject *parent)
|
|
: Plasma::Corona(parent),
|
|
m_defaultLayoutOnStartup(defaultLayoutOnStartup),
|
|
m_layoutNameOnStartUp(layoutNameOnStartUp),
|
|
m_activityConsumer(new KActivities::Consumer(this)),
|
|
m_screenPool(new ScreenPool(KSharedConfig::openConfig(), this)),
|
|
m_globalShortcuts(new GlobalShortcuts(this)),
|
|
m_universalSettings(new UniversalSettings(KSharedConfig::openConfig(), this)),
|
|
m_layoutManager(new LayoutManager(this))
|
|
{
|
|
setupWaylandIntegration();
|
|
|
|
KPackage::Package package(new DockPackage(this));
|
|
m_screenPool->load();
|
|
|
|
if (!package.isValid()) {
|
|
qWarning() << staticMetaObject.className()
|
|
<< "the package" << package.metadata().rawData() << "is invalid!";
|
|
return;
|
|
} else {
|
|
qDebug() << staticMetaObject.className()
|
|
<< "the package" << package.metadata().rawData() << "is valid!";
|
|
}
|
|
|
|
setKPackage(package);
|
|
//! universal settings must be loaded after the package has been set
|
|
m_universalSettings->load();
|
|
|
|
qmlRegisterTypes();
|
|
QFontDatabase::addApplicationFont(kPackage().filePath("tangerineFont"));
|
|
|
|
if (m_activityConsumer && (m_activityConsumer->serviceStatus() == KActivities::Consumer::Running)) {
|
|
load();
|
|
}
|
|
|
|
connect(m_activityConsumer, &KActivities::Consumer::serviceStatusChanged, this, &DockCorona::load);
|
|
|
|
m_docksScreenSyncTimer.setSingleShot(true);
|
|
m_docksScreenSyncTimer.setInterval(2500);
|
|
connect(&m_docksScreenSyncTimer, &QTimer::timeout, this, &DockCorona::syncDockViewsToScreens);
|
|
|
|
//! Dbus adaptor initialization
|
|
new LatteDockAdaptor(this);
|
|
QDBusConnection dbus = QDBusConnection::sessionBus();
|
|
dbus.registerObject(QStringLiteral("/Latte"), this);
|
|
}
|
|
|
|
DockCorona::~DockCorona()
|
|
{
|
|
m_docksScreenSyncTimer.stop();
|
|
|
|
if (m_layoutManager->memoryUsage() == Dock::SingleLayout) {
|
|
cleanConfig();
|
|
}
|
|
|
|
qDebug() << "Latte Corona - unload: containments ...";
|
|
|
|
m_layoutManager->unload();
|
|
|
|
m_globalShortcuts->deleteLater();
|
|
m_layoutManager->deleteLater();
|
|
m_screenPool->deleteLater();
|
|
m_universalSettings->deleteLater();
|
|
|
|
disconnect(m_activityConsumer, &KActivities::Consumer::serviceStatusChanged, this, &DockCorona::load);
|
|
delete m_activityConsumer;
|
|
|
|
qDebug() << "Latte Corona - deleted...";
|
|
}
|
|
|
|
void DockCorona::load()
|
|
{
|
|
if (m_activityConsumer && (m_activityConsumer->serviceStatus() == KActivities::Consumer::Running) && m_activitiesStarting) {
|
|
disconnect(m_activityConsumer, &KActivities::Consumer::serviceStatusChanged, this, &DockCorona::load);
|
|
m_layoutManager->load();
|
|
|
|
m_activitiesStarting = false;
|
|
|
|
// connect(qGuiApp, &QGuiApplication::screenAdded, this, &DockCorona::addOutput, Qt::UniqueConnection);
|
|
connect(qGuiApp, &QGuiApplication::primaryScreenChanged, this, &DockCorona::primaryOutputChanged, Qt::UniqueConnection);
|
|
// connect(qGuiApp, &QGuiApplication::screenRemoved, this, &DockCorona::screenRemoved, Qt::UniqueConnection);
|
|
connect(QApplication::desktop(), &QDesktopWidget::screenCountChanged, this, &DockCorona::screenCountChanged);
|
|
connect(m_screenPool, &ScreenPool::primaryPoolChanged, this, &DockCorona::screenCountChanged);
|
|
|
|
QString assignedLayout = m_layoutManager->shouldSwitchToLayout(m_activityConsumer->currentActivity());
|
|
|
|
QString loadLayoutName = "";
|
|
|
|
if (!m_defaultLayoutOnStartup && m_layoutNameOnStartUp.isEmpty()) {
|
|
if (!assignedLayout.isEmpty() && assignedLayout != m_universalSettings->currentLayoutName()) {
|
|
loadLayoutName = assignedLayout;
|
|
} else {
|
|
loadLayoutName = m_universalSettings->currentLayoutName();
|
|
}
|
|
|
|
if (!m_layoutManager->layoutExists(loadLayoutName)) {
|
|
loadLayoutName = m_layoutManager->defaultLayoutName();
|
|
m_layoutManager->importDefaultLayout(false);
|
|
}
|
|
} else if (m_defaultLayoutOnStartup) {
|
|
loadLayoutName = m_layoutManager->importer()->uniqueLayoutName(m_layoutManager->defaultLayoutName());
|
|
m_layoutManager->importDefaultLayout(true);
|
|
} else {
|
|
loadLayoutName = m_layoutNameOnStartUp;
|
|
}
|
|
|
|
m_layoutManager->loadLayoutOnStartup(loadLayoutName);
|
|
}
|
|
}
|
|
|
|
void DockCorona::unload()
|
|
{
|
|
qDebug() << "unload: removing containments...";
|
|
|
|
while (!containments().isEmpty()) {
|
|
//deleting a containment will remove it from the list due to QObject::destroyed connect in Corona
|
|
//this form doesn't crash, while qDeleteAll(containments()) does
|
|
delete containments().first();
|
|
}
|
|
}
|
|
|
|
void DockCorona::setupWaylandIntegration()
|
|
{
|
|
using namespace KWayland::Client;
|
|
|
|
if (!KWindowSystem::isPlatformWayland()) {
|
|
return;
|
|
}
|
|
|
|
auto connection = ConnectionThread::fromApplication(this);
|
|
|
|
if (!connection)
|
|
return;
|
|
|
|
Registry *registry{new Registry(this)};
|
|
registry->create(connection);
|
|
|
|
connect(registry, &Registry::plasmaShellAnnounced, this
|
|
, [this, registry](quint32 name, quint32 version) {
|
|
m_waylandDockCorona = registry->createPlasmaShell(name, version, this);
|
|
});
|
|
|
|
connect(qApp, &QCoreApplication::aboutToQuit, this, [this, registry]() {
|
|
if (m_waylandDockCorona)
|
|
m_waylandDockCorona->release();
|
|
|
|
registry->release();
|
|
});
|
|
|
|
registry->setup();
|
|
}
|
|
|
|
KWayland::Client::PlasmaShell *DockCorona::waylandDockCoronaInterface() const
|
|
{
|
|
return m_waylandDockCorona;
|
|
}
|
|
|
|
void DockCorona::cleanConfig()
|
|
{
|
|
auto containmentsEntries = config()->group("Containments");
|
|
bool changed = false;
|
|
|
|
foreach (auto cId, containmentsEntries.groupList()) {
|
|
if (!containmentExists(cId.toUInt())) {
|
|
//cleanup obsolete containments
|
|
containmentsEntries.group(cId).deleteGroup();
|
|
changed = true;
|
|
qDebug() << "obsolete containment configuration deleted:" << cId;
|
|
} else {
|
|
//cleanup obsolete applets of running containments
|
|
auto appletsEntries = containmentsEntries.group(cId).group("Applets");
|
|
|
|
foreach (auto appletId, appletsEntries.groupList()) {
|
|
if (!appletExists(cId.toUInt(), appletId.toUInt())) {
|
|
appletsEntries.group(appletId).deleteGroup();
|
|
changed = true;
|
|
qDebug() << "obsolete applet configuration deleted:" << appletId;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (changed) {
|
|
config()->sync();
|
|
qDebug() << "configuration file cleaned...";
|
|
}
|
|
}
|
|
|
|
bool DockCorona::containmentExists(uint id) const
|
|
{
|
|
foreach (auto containment, containments()) {
|
|
if (id == containment->id()) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool DockCorona::appletExists(uint containmentId, uint appletId) const
|
|
{
|
|
Plasma::Containment *containment = nullptr;
|
|
|
|
foreach (auto cont, containments()) {
|
|
if (containmentId == cont->id()) {
|
|
containment = cont;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!containment) {
|
|
return false;
|
|
}
|
|
|
|
foreach (auto applet, containment->applets()) {
|
|
if (applet->id() == appletId) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
KActivities::Consumer *DockCorona::activitiesConsumer() const
|
|
{
|
|
return m_activityConsumer;
|
|
}
|
|
|
|
ScreenPool *DockCorona::screenPool() const
|
|
{
|
|
return m_screenPool;
|
|
}
|
|
|
|
UniversalSettings *DockCorona::universalSettings() const
|
|
{
|
|
return m_universalSettings;
|
|
}
|
|
|
|
LayoutManager *DockCorona::layoutManager() const
|
|
{
|
|
return m_layoutManager;
|
|
}
|
|
|
|
int DockCorona::numScreens() const
|
|
{
|
|
return qGuiApp->screens().count();
|
|
}
|
|
|
|
QRect DockCorona::screenGeometry(int id) const
|
|
{
|
|
const auto screens = qGuiApp->screens();
|
|
const QScreen *screen{qGuiApp->primaryScreen()};
|
|
|
|
QString screenName;
|
|
|
|
if (m_screenPool->knownIds().contains(id))
|
|
screenName = m_screenPool->connector(id);
|
|
|
|
foreach (auto scr, screens) {
|
|
if (scr->name() == screenName) {
|
|
screen = scr;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return screen->geometry();
|
|
}
|
|
|
|
QRegion DockCorona::availableScreenRegion(int id) const
|
|
{
|
|
return availableScreenRegionWithCriteria(id);
|
|
}
|
|
|
|
QRegion DockCorona::availableScreenRegionWithCriteria(int id, QString forLayout) const
|
|
{
|
|
const auto screens = qGuiApp->screens();
|
|
const QScreen *screen{qGuiApp->primaryScreen()};
|
|
|
|
QString screenName;
|
|
|
|
if (m_screenPool->knownIds().contains(id))
|
|
screenName = m_screenPool->connector(id);
|
|
|
|
foreach (auto scr, screens) {
|
|
if (scr->name() == screenName) {
|
|
screen = scr;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!screen)
|
|
return QRegion();
|
|
|
|
QHash<const Plasma::Containment *, DockView *> *views;
|
|
|
|
if (forLayout.isEmpty()) {
|
|
views = m_layoutManager->currentDockViews();
|
|
} else {
|
|
views = m_layoutManager->layoutDockViews(forLayout);
|
|
}
|
|
|
|
QRegion available(screen->geometry());
|
|
|
|
if (views) {
|
|
for (const auto *view : *views) {
|
|
if (view && view->containment() && view->screen() == screen
|
|
&& view->visibility() && (view->visibility()->mode() != Latte::Dock::AutoHide)) {
|
|
int realThickness = view->normalThickness() - view->shadow();
|
|
|
|
// Usually availableScreenRect is used by the desktop,
|
|
// but Latte dont have desktop, then here just
|
|
// need calculate available space for top and bottom location,
|
|
// because the left and right are those who dodge others docks
|
|
switch (view->location()) {
|
|
case Plasma::Types::TopEdge:
|
|
if (view->behaveAsPlasmaPanel()) {
|
|
available -= view->geometry();
|
|
} else {
|
|
QRect realGeometry;
|
|
int realWidth = view->maxLength() * view->width();
|
|
|
|
switch (view->alignment()) {
|
|
case Latte::Dock::Left:
|
|
realGeometry = QRect(view->x(), view->y(),
|
|
realWidth, realThickness);
|
|
break;
|
|
|
|
case Latte::Dock::Center:
|
|
case Latte::Dock::Justify:
|
|
realGeometry = QRect(qMax(view->geometry().x(), view->geometry().center().x() - realWidth / 2) , view->y(),
|
|
realWidth , realThickness);
|
|
break;
|
|
|
|
case Latte::Dock::Right:
|
|
realGeometry = QRect(view->geometry().right() - realWidth + 1, view->y(),
|
|
realWidth, realThickness);
|
|
break;
|
|
}
|
|
|
|
available -= realGeometry;
|
|
}
|
|
|
|
break;
|
|
|
|
case Plasma::Types::BottomEdge:
|
|
if (view->behaveAsPlasmaPanel()) {
|
|
available -= view->geometry();
|
|
} else {
|
|
QRect realGeometry;
|
|
int realWidth = view->maxLength() * view->width();
|
|
int realY = view->geometry().bottom() - realThickness + 1;
|
|
|
|
switch (view->alignment()) {
|
|
case Latte::Dock::Left:
|
|
realGeometry = QRect(view->x(), realY,
|
|
realWidth, realThickness);
|
|
break;
|
|
|
|
case Latte::Dock::Center:
|
|
case Latte::Dock::Justify:
|
|
realGeometry = QRect(qMax(view->geometry().x(), view->geometry().center().x() - realWidth / 2),
|
|
realY, realWidth, realThickness);
|
|
break;
|
|
|
|
case Latte::Dock::Right:
|
|
realGeometry = QRect(view->geometry().right() - realWidth + 1, realY,
|
|
realWidth, realThickness);
|
|
break;
|
|
}
|
|
|
|
available -= realGeometry;
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
//! bypass clang warnings
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*qDebug() << "::::: FREE AREAS :::::";
|
|
|
|
for (int i = 0; i < available.rectCount(); ++i) {
|
|
qDebug() << available.rects().at(i);
|
|
}
|
|
|
|
qDebug() << "::::: END OF FREE AREAS :::::";*/
|
|
|
|
return available;
|
|
}
|
|
|
|
QRect DockCorona::availableScreenRect(int id) const
|
|
{
|
|
return availableScreenRectWithCriteria(id);
|
|
}
|
|
|
|
QRect DockCorona::availableScreenRectWithCriteria(int id, QList<Dock::Visibility> modes, QList<Plasma::Types::Location> edges) const
|
|
{
|
|
const auto screens = qGuiApp->screens();
|
|
const QScreen *screen{qGuiApp->primaryScreen()};
|
|
|
|
if (m_screenPool->knownIds().contains(id)) {
|
|
QString scrName = m_screenPool->connector(id);
|
|
|
|
foreach (auto scr, screens) {
|
|
if (scr->name() == scrName) {
|
|
screen = scr;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!screen)
|
|
return {};
|
|
|
|
bool allModes = modes.isEmpty();
|
|
|
|
bool allEdges = edges.isEmpty();
|
|
|
|
auto available = screen->geometry();
|
|
|
|
QHash<const Plasma::Containment *, DockView *> *views = m_layoutManager->currentDockViews();
|
|
|
|
if (views) {
|
|
for (const auto *view : *views) {
|
|
if (view && view->containment() && view->screen() == screen
|
|
&& ((allEdges || edges.contains(view->location()))
|
|
&& (allModes || (view->visibility() && modes.contains(view->visibility()->mode()))))) {
|
|
|
|
auto dockRect = view->absGeometry();
|
|
|
|
// Usually availableScreenRect is used by the desktop,
|
|
// but Latte dont have desktop, then here just
|
|
// need calculate available space for top and bottom location,
|
|
// because the left and right are those who dodge others docks
|
|
switch (view->location()) {
|
|
case Plasma::Types::TopEdge:
|
|
available.setTop(dockRect.bottom() + 1);
|
|
break;
|
|
|
|
case Plasma::Types::BottomEdge:
|
|
available.setBottom(dockRect.top() - 1);
|
|
break;
|
|
|
|
case Plasma::Types::LeftEdge:
|
|
available.setLeft(dockRect.right() + 1);
|
|
|
|
break;
|
|
|
|
case Plasma::Types::RightEdge:
|
|
available.setRight(dockRect.left() - 1);
|
|
|
|
break;
|
|
|
|
default:
|
|
//! bypass clang warnings
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return available;
|
|
}
|
|
|
|
//! the number of currently running docks containing
|
|
//! tasks plasmoid
|
|
int DockCorona::noDocksWithTasks() const
|
|
{
|
|
QHash<const Plasma::Containment *, DockView *> *views = m_layoutManager->currentDockViews();
|
|
|
|
int result = 0;
|
|
|
|
if (views) {
|
|
foreach (auto view, *views) {
|
|
if (view->tasksPresent()) {
|
|
result++;
|
|
}
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
void DockCorona::addOutput(QScreen *screen)
|
|
{
|
|
Q_ASSERT(screen);
|
|
|
|
int id = m_screenPool->id(screen->name());
|
|
|
|
if (id == -1) {
|
|
int newId = m_screenPool->firstAvailableId();
|
|
m_screenPool->insertScreenMapping(newId, screen->name());
|
|
}
|
|
}
|
|
|
|
void DockCorona::primaryOutputChanged()
|
|
{
|
|
/* qDebug() << "primary changed ### "<< qGuiApp->primaryScreen()->name();
|
|
foreach(auto scr, qGuiApp->screens()){
|
|
qDebug() << "Found screen: "<<scr->name();
|
|
}*/
|
|
|
|
//if (m_dockViews.count()==1 && qGuiApp->screens().size()==1) {
|
|
// foreach(auto view, m_dockViews) {
|
|
// view->setScreenToFollow(qGuiApp->primaryScreen());
|
|
// }
|
|
// }
|
|
}
|
|
|
|
void DockCorona::screenRemoved(QScreen *screen)
|
|
{
|
|
Q_ASSERT(screen);
|
|
}
|
|
|
|
void DockCorona::screenCountChanged()
|
|
{
|
|
m_docksScreenSyncTimer.start();
|
|
}
|
|
|
|
//! the central functions that updates loading/unloading dockviews
|
|
//! concerning screen changed (for multi-screen setups mainly)
|
|
void DockCorona::syncDockViewsToScreens()
|
|
{
|
|
m_layoutManager->syncDockViewsToScreens();
|
|
}
|
|
|
|
int DockCorona::primaryScreenId() const
|
|
{
|
|
return m_screenPool->id(qGuiApp->primaryScreen()->name());
|
|
}
|
|
|
|
int DockCorona::docksCount(int screen) const
|
|
{
|
|
QScreen *scr = m_screenPool->screenForId(screen);
|
|
QHash<const Plasma::Containment *, DockView *> *views = m_layoutManager->currentDockViews();
|
|
|
|
int docks{0};
|
|
|
|
if (views) {
|
|
for (const auto &view : *views) {
|
|
if (view && view->screen() == scr && !view->containment()->destroyed()) {
|
|
++docks;
|
|
}
|
|
}
|
|
}
|
|
|
|
// qDebug() << docks << "docks on screen:" << screen;
|
|
return docks;
|
|
}
|
|
|
|
int DockCorona::docksCount() const
|
|
{
|
|
int docks{0};
|
|
|
|
QHash<const Plasma::Containment *, DockView *> *views = m_layoutManager->currentDockViews();
|
|
|
|
if (views) {
|
|
for (const auto &view : *views) {
|
|
if (view && view->containment() && !view->containment()->destroyed()) {
|
|
++docks;
|
|
}
|
|
}
|
|
}
|
|
|
|
// qDebug() << docks << "docks on screen:" << screen;
|
|
return docks;
|
|
}
|
|
|
|
int DockCorona::docksCount(QScreen *screen) const
|
|
{
|
|
int docks{0};
|
|
|
|
QHash<const Plasma::Containment *, DockView *> *views = m_layoutManager->currentDockViews();
|
|
|
|
if (views) {
|
|
for (const auto &view : *views) {
|
|
if (view && view->screen() == screen && !view->containment()->destroyed()) {
|
|
++docks;
|
|
}
|
|
}
|
|
}
|
|
|
|
// qDebug() << docks << "docks on screen:" << screen;
|
|
return docks;
|
|
}
|
|
|
|
void DockCorona::closeApplication()
|
|
{
|
|
qGuiApp->quit();
|
|
}
|
|
|
|
void DockCorona::aboutApplication()
|
|
{
|
|
if (aboutDialog) {
|
|
aboutDialog->hide();
|
|
aboutDialog->deleteLater();
|
|
}
|
|
|
|
aboutDialog = new KAboutApplicationDialog(KAboutData::applicationData());
|
|
connect(aboutDialog.data(), &QDialog::finished, aboutDialog.data(), &QObject::deleteLater);
|
|
WindowSystem::self().skipTaskBar(*aboutDialog);
|
|
|
|
aboutDialog->show();
|
|
}
|
|
|
|
int DockCorona::noOfDocks()
|
|
{
|
|
QHash<const Plasma::Containment *, DockView *> *views = m_layoutManager->currentDockViews();
|
|
|
|
if (views) {
|
|
return views->count();
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
QList<Plasma::Types::Location> DockCorona::freeEdges(QScreen *screen) const
|
|
{
|
|
using Plasma::Types;
|
|
QList<Types::Location> edges{Types::BottomEdge, Types::LeftEdge,
|
|
Types::TopEdge, Types::RightEdge};
|
|
|
|
QHash<const Plasma::Containment *, DockView *> *views = m_layoutManager->currentDockViews();
|
|
|
|
if (views) {
|
|
for (auto *view : *views) {
|
|
if (view && view->currentScreen() == screen->name()) {
|
|
edges.removeOne(view->location());
|
|
}
|
|
}
|
|
}
|
|
|
|
return edges;
|
|
}
|
|
|
|
QList<Plasma::Types::Location> DockCorona::freeEdges(int screen) const
|
|
{
|
|
using Plasma::Types;
|
|
QList<Types::Location> edges{Types::BottomEdge, Types::LeftEdge,
|
|
Types::TopEdge, Types::RightEdge};
|
|
|
|
QScreen *scr = m_screenPool->screenForId(screen);
|
|
|
|
QHash<const Plasma::Containment *, DockView *> *views = m_layoutManager->currentDockViews();
|
|
|
|
if (views) {
|
|
for (auto *view : *views) {
|
|
if (view && scr && view->currentScreen() == scr->name()) {
|
|
edges.removeOne(view->location());
|
|
}
|
|
}
|
|
}
|
|
|
|
return edges;
|
|
}
|
|
|
|
int DockCorona::screenForContainment(const Plasma::Containment *containment) const
|
|
{
|
|
//FIXME: indexOf is not a proper way to support multi-screen
|
|
// as for environment to environment the indexes change
|
|
// also there is the following issue triggered
|
|
// from dockView adaptToScreen()
|
|
//
|
|
// in a multi-screen environment that
|
|
// primary screen is not set to 0 it was
|
|
// created an endless showing loop at
|
|
// startup (catch-up race) between
|
|
// screen:0 and primaryScreen
|
|
|
|
//case in which this containment is child of an applet, hello systray :)
|
|
|
|
if (Plasma::Applet *parentApplet = qobject_cast<Plasma::Applet *>(containment->parent())) {
|
|
if (Plasma::Containment *cont = parentApplet->containment()) {
|
|
return screenForContainment(cont);
|
|
} else {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
QHash<const Plasma::Containment *, DockView *> *views = m_layoutManager->currentDockViews();
|
|
|
|
//if the panel views already exist, base upon them
|
|
|
|
DockView *view = views ? views->value(containment) : nullptr;
|
|
|
|
if (view && view->screen()) {
|
|
return m_screenPool->id(view->screen()->name());
|
|
}
|
|
|
|
//Failed? fallback on lastScreen()
|
|
//lastScreen() is the correct screen for panels
|
|
//It is also correct for desktops *that have the correct activity()*
|
|
//a containment with lastScreen() == 0 but another activity,
|
|
//won't be associated to a screen
|
|
// qDebug() << "ShellCorona screenForContainment: " << containment << " Last screen is " << containment->lastScreen();
|
|
|
|
for (auto screen : qGuiApp->screens()) {
|
|
// containment->lastScreen() == m_screenPool->id(screen->name()) to check if the lastScreen refers to a screen that exists/it's known
|
|
if (containment->lastScreen() == m_screenPool->id(screen->name()) &&
|
|
(containment->activity() == m_activityConsumer->currentActivity() ||
|
|
containment->containmentType() == Plasma::Types::PanelContainment || containment->containmentType() == Plasma::Types::CustomPanelContainment)) {
|
|
return containment->lastScreen();
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
bool DockCorona::explicitDockOccupyEdge(int screen, Plasma::Types::Location location) const
|
|
{
|
|
foreach (auto containment, containments()) {
|
|
bool onPrimary = containment->config().readEntry("onPrimary", true);
|
|
int id = containment->lastScreen();
|
|
Plasma::Types::Location contLocation = containment->location();
|
|
|
|
if (!onPrimary && id == screen && contLocation == location) {
|
|
return true;
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void DockCorona::recreateDock(Plasma::Containment *containment)
|
|
{
|
|
m_layoutManager->recreateDock(containment);
|
|
}
|
|
|
|
void DockCorona::showAlternativesForApplet(Plasma::Applet *applet)
|
|
{
|
|
const QString alternativesQML = kPackage().filePath("appletalternativesui");
|
|
|
|
if (alternativesQML.isEmpty()) {
|
|
return;
|
|
}
|
|
|
|
QHash<const Plasma::Containment *, DockView *> *views = m_layoutManager->currentDockViews();
|
|
|
|
DockView *dockView = (*views)[applet->containment()];
|
|
|
|
KDeclarative::QmlObject *qmlObj{nullptr};
|
|
|
|
if (dockView) {
|
|
dockView->setAlternativesIsShown(true);
|
|
qmlObj = new KDeclarative::QmlObject(dockView);
|
|
} else {
|
|
qmlObj = new KDeclarative::QmlObject(this);
|
|
}
|
|
|
|
qmlObj->setInitializationDelayed(true);
|
|
qmlObj->setSource(QUrl::fromLocalFile(alternativesQML));
|
|
|
|
AlternativesHelper *helper = new AlternativesHelper(applet, qmlObj);
|
|
qmlObj->rootContext()->setContextProperty(QStringLiteral("alternativesHelper"), helper);
|
|
|
|
m_alternativesObjects << qmlObj;
|
|
qmlObj->completeInitialization();
|
|
|
|
//! Alternative dialog signals
|
|
connect(helper, &QObject::destroyed, this, [dockView]() {
|
|
dockView->setAlternativesIsShown(false);
|
|
});
|
|
|
|
connect(qmlObj->rootObject(), SIGNAL(visibleChanged(bool)),
|
|
this, SLOT(alternativesVisibilityChanged(bool)));
|
|
|
|
connect(applet, &Plasma::Applet::destroyedChanged, this, [this, qmlObj](bool destroyed) {
|
|
if (!destroyed) {
|
|
return;
|
|
}
|
|
|
|
QMutableListIterator<KDeclarative::QmlObject *> it(m_alternativesObjects);
|
|
|
|
while (it.hasNext()) {
|
|
KDeclarative::QmlObject *obj = it.next();
|
|
|
|
if (obj == qmlObj) {
|
|
it.remove();
|
|
obj->deleteLater();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
void DockCorona::alternativesVisibilityChanged(bool visible)
|
|
{
|
|
if (visible) {
|
|
return;
|
|
}
|
|
|
|
QObject *root = sender();
|
|
|
|
QMutableListIterator<KDeclarative::QmlObject *> it(m_alternativesObjects);
|
|
|
|
while (it.hasNext()) {
|
|
KDeclarative::QmlObject *obj = it.next();
|
|
|
|
if (obj->rootObject() == root) {
|
|
it.remove();
|
|
obj->deleteLater();
|
|
}
|
|
}
|
|
}
|
|
|
|
void DockCorona::loadDefaultLayout()
|
|
{
|
|
qDebug() << "loading default layout";
|
|
//! Settting mutable for create a containment
|
|
setImmutability(Plasma::Types::Mutable);
|
|
QVariantList args;
|
|
auto defaultContainment = createContainmentDelayed("org.kde.latte.containment", args);
|
|
defaultContainment->setContainmentType(Plasma::Types::PanelContainment);
|
|
defaultContainment->init();
|
|
|
|
if (!defaultContainment || !defaultContainment->kPackage().isValid()) {
|
|
qWarning() << "the requested containment plugin can not be located or loaded";
|
|
return;
|
|
}
|
|
|
|
auto config = defaultContainment->config();
|
|
defaultContainment->restore(config);
|
|
QList<Plasma::Types::Location> edges = freeEdges(defaultContainment->screen());
|
|
|
|
if ((edges.count() > 0)) {
|
|
defaultContainment->setLocation(edges.at(0));
|
|
} else {
|
|
defaultContainment->setLocation(Plasma::Types::BottomEdge);
|
|
}
|
|
|
|
if (m_layoutManager->memoryUsage() == Dock::MultipleLayouts) {
|
|
config.writeEntry("layoutId", m_layoutManager->currentLayoutName());
|
|
}
|
|
|
|
defaultContainment->updateConstraints(Plasma::Types::StartupCompletedConstraint);
|
|
|
|
defaultContainment->save(config);
|
|
requestConfigSync();
|
|
|
|
defaultContainment->flushPendingConstraintsEvents();
|
|
emit containmentAdded(defaultContainment);
|
|
emit containmentCreated(defaultContainment);
|
|
|
|
//m_layoutManager->addDock(defaultContainment);
|
|
defaultContainment->createApplet(QStringLiteral("org.kde.latte.plasmoid"));
|
|
defaultContainment->createApplet(QStringLiteral("org.kde.plasma.analogclock"));
|
|
}
|
|
|
|
QStringList DockCorona::containmentsIds()
|
|
{
|
|
QStringList ids;
|
|
|
|
foreach (auto containment, containments()) {
|
|
ids << QString::number(containment->id());
|
|
}
|
|
|
|
return ids;
|
|
}
|
|
|
|
QStringList DockCorona::appletsIds()
|
|
{
|
|
QStringList ids;
|
|
|
|
foreach (auto containment, containments()) {
|
|
foreach (auto applet, containment->applets()) {
|
|
ids << QString::number(applet->id());
|
|
}
|
|
|
|
}
|
|
|
|
return ids;
|
|
}
|
|
|
|
//! Activate launcher menu through dbus interface
|
|
void DockCorona::activateLauncherMenu()
|
|
{
|
|
m_globalShortcuts->activateLauncherMenu();
|
|
}
|
|
|
|
//! update badge for specific dock item
|
|
void DockCorona::updateDockItemBadge(QString identifier, QString value)
|
|
{
|
|
m_globalShortcuts->updateDockItemBadge(identifier, value);
|
|
}
|
|
|
|
inline void DockCorona::qmlRegisterTypes() const
|
|
{
|
|
qmlRegisterType<QScreen>();
|
|
}
|
|
|
|
}
|