1
0
mirror of https://github.com/KDE/latte-dock.git synced 2024-12-24 17:33:50 +03:00
latte-dock/app/lattecorona.cpp

956 lines
29 KiB
C++
Raw Normal View History

/*
* Copyright 2016 Smith AR <audoban@openmaibox.org>
2017-01-03 01:05:30 +03:00
* 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 "lattecorona.h"
2018-12-02 03:05:52 +03:00
// local
#include "alternativeshelper.h"
#include "globalshortcuts.h"
#include "importer.h"
2018-12-02 03:05:52 +03:00
#include "lattedockadaptor.h"
#include "launcherssignals.h"
#include "layoutmanager.h"
2017-02-24 21:58:21 +03:00
#include "screenpool.h"
#include "packageplugins/shell/lattepackage.h"
#include "plasma/extended/theme.h"
#include "settings/universalsettings.h"
#include "view/view.h"
#include "wm/abstractwindowinterface.h"
#include "wm/waylandinterface.h"
#include "wm/xwindowinterface.h"
2018-12-02 03:05:52 +03:00
// Qt
#include <QAction>
#include <QApplication>
#include <QScreen>
#include <QDBusConnection>
#include <QDebug>
#include <QDesktopWidget>
#include <QFile>
#include <QFontDatabase>
#include <QQmlContext>
2018-12-02 03:05:52 +03:00
// Plasma
#include <Plasma>
#include <Plasma/Corona>
#include <Plasma/Containment>
#include <PlasmaQuick/ConfigView>
2018-12-02 03:05:52 +03:00
// KDE
2016-12-31 00:25:27 +03:00
#include <KActionCollection>
#include <KPluginMetaData>
#include <KGlobalAccel>
#include <KLocalizedString>
#include <KPackage/Package>
#include <KPackage/PackageLoader>
2017-01-29 08:16:28 +03:00
#include <KAboutData>
#include <KActivities/Consumer>
#include <KDeclarative/QmlObject>
#include <KWindowSystem>
#include <KWayland/Client/connection_thread.h>
#include <KWayland/Client/registry.h>
#include <KWayland/Client/plasmashell.h>
#include <KWayland/Client/plasmawindowmanagement.h>
2016-12-30 10:20:06 +03:00
namespace Latte {
Corona::Corona(bool defaultLayoutOnStartup, QString layoutNameOnStartUp, int userSetMemoryUsage, QObject *parent)
: Plasma::Corona(parent),
m_defaultLayoutOnStartup(defaultLayoutOnStartup),
m_userSetMemoryUsage(userSetMemoryUsage),
m_layoutNameOnStartUp(layoutNameOnStartUp),
2017-04-02 08:25:03 +03:00
m_activityConsumer(new KActivities::Consumer(this)),
2017-02-24 21:58:21 +03:00
m_screenPool(new ScreenPool(KSharedConfig::openConfig(), this)),
m_globalShortcuts(new GlobalShortcuts(this)),
m_universalSettings(new UniversalSettings(KSharedConfig::openConfig(), this)),
m_themeExtended(new PlasmaExtended::Theme(KSharedConfig::openConfig(), this)),
m_layoutManager(new LayoutManager(this))
{
//! create the window manager
if (KWindowSystem::isPlatformWayland()) {
m_wm = new WaylandInterface(this);
} else {
m_wm = new XWindowInterface(this);
}
setupWaylandIntegration();
KPackage::Package package(new Latte::Package(this));
2018-10-27 00:59:36 +03:00
m_screenPool->load();
2017-01-16 22:07:49 +03:00
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!";
}
2017-01-16 22:07:49 +03:00
setKPackage(package);
2018-10-27 00:59:36 +03:00
//! universal settings / extendedtheme must be loaded after the package has been set
m_universalSettings->load();
2018-10-27 00:59:36 +03:00
m_themeExtended->load();
qmlRegisterTypes();
if (m_activityConsumer && (m_activityConsumer->serviceStatus() == KActivities::Consumer::Running)) {
load();
}
connect(m_activityConsumer, &KActivities::Consumer::serviceStatusChanged, this, &Corona::load);
m_viewsScreenSyncTimer.setSingleShot(true);
m_viewsScreenSyncTimer.setInterval(m_universalSettings->screenTrackerInterval());
connect(&m_viewsScreenSyncTimer, &QTimer::timeout, this, &Corona::syncLatteViewsToScreens);
connect(m_universalSettings, &UniversalSettings::screenTrackerIntervalChanged, this, [this]() {
m_viewsScreenSyncTimer.setInterval(m_universalSettings->screenTrackerInterval());
});
//! Dbus adaptor initialization
new LatteDockAdaptor(this);
QDBusConnection dbus = QDBusConnection::sessionBus();
dbus.registerObject(QStringLiteral("/Latte"), this);
}
Corona::~Corona()
{
//! BEGIN: Give the time to slide-out views when closing
m_layoutManager->hideAllViews();
2018-02-03 16:44:37 +03:00
2018-08-30 08:47:53 +03:00
//! Don't delay the destruction under wayland in any case
//! because it creates a crash with kwin effects
//! https://bugs.kde.org/show_bug.cgi?id=392890
if (!KWindowSystem::isPlatformWayland()) {
QTimer::singleShot(400, [this]() {
m_quitTimedEnded = true;
});
while (!m_quitTimedEnded) {
QGuiApplication::processEvents(QEventLoop::AllEvents, 50);
}
2018-02-03 16:44:37 +03:00
}
//! END: slide-out views when closing
2018-02-03 16:44:37 +03:00
m_viewsScreenSyncTimer.stop();
2018-01-14 18:30:29 +03:00
if (m_layoutManager->memoryUsage() == Types::SingleLayout) {
2018-01-14 18:30:29 +03:00
cleanConfig();
}
2018-01-10 20:48:23 +03:00
qDebug() << "Latte Corona - unload: containments ...";
2017-06-26 22:16:42 +03:00
m_layoutManager->unload();
2017-01-16 22:07:49 +03:00
m_wm->deleteLater();
m_globalShortcuts->deleteLater();
m_layoutManager->deleteLater();
m_screenPool->deleteLater();
m_universalSettings->deleteLater();
m_themeExtended->deleteLater();
disconnect(m_activityConsumer, &KActivities::Consumer::serviceStatusChanged, this, &Corona::load);
delete m_activityConsumer;
2017-06-26 22:16:42 +03:00
qDebug() << "Latte Corona - deleted...";
}
void Corona::load()
{
if (m_activityConsumer && (m_activityConsumer->serviceStatus() == KActivities::Consumer::Running) && m_activitiesStarting) {
disconnect(m_activityConsumer, &KActivities::Consumer::serviceStatusChanged, this, &Corona::load);
m_layoutManager->load();
2017-02-24 21:58:21 +03:00
m_activitiesStarting = false;
connect(qGuiApp, &QGuiApplication::primaryScreenChanged, this, &Corona::primaryOutputChanged, Qt::UniqueConnection);
connect(QApplication::desktop(), &QDesktopWidget::screenCountChanged, this, &Corona::screenCountChanged);
2018-02-03 16:44:37 +03:00
connect(m_screenPool, &ScreenPool::primaryPoolChanged, this, &Corona::screenCountChanged);
2017-02-24 21:58:21 +03:00
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;
}
if (m_userSetMemoryUsage != -1 && !KWindowSystem::isPlatformWayland()) {
Types::LayoutsMemoryUsage usage = static_cast<Types::LayoutsMemoryUsage>(m_userSetMemoryUsage);
m_universalSettings->setLayoutsMemoryUsage(usage);
}
if (KWindowSystem::isPlatformWayland()) {
m_universalSettings->setLayoutsMemoryUsage(Types::SingleLayout);
}
m_layoutManager->loadLayoutOnStartup(loadLayoutName);
//! load screens signals such screenGeometryChanged in order to support
//! plasmoid.screenGeometry properly
for (QScreen *screen : qGuiApp->screens()) {
addOutput(screen);
}
connect(qGuiApp, &QGuiApplication::screenAdded, this, &Corona::addOutput, Qt::UniqueConnection);
}
}
void Corona::unload()
2017-06-26 22:16:42 +03:00
{
qDebug() << "unload: removing containments...";
2017-06-26 22:16:42 +03:00
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 Corona::setupWaylandIntegration()
{
if (!KWindowSystem::isPlatformWayland()) {
return;
}
2018-03-01 01:52:05 +03:00
using namespace KWayland::Client;
2017-06-24 00:27:58 +03:00
auto connection = ConnectionThread::fromApplication(this);
2018-03-01 01:52:05 +03:00
if (!connection) {
return;
2018-03-01 01:52:05 +03:00
}
Registry *registry{new Registry(this)};
registry->create(connection);
connect(registry, &Registry::plasmaShellAnnounced, this
, [this, registry](quint32 name, quint32 version) {
m_waylandCorona = registry->createPlasmaShell(name, version, this);
});
QObject::connect(registry, &KWayland::Client::Registry::plasmaWindowManagementAnnounced,
[this, registry](quint32 name, quint32 version) {
KWayland::Client::PlasmaWindowManagement *pwm = registry->createPlasmaWindowManagement(name, version, this);
WaylandInterface *wI = qobject_cast<WaylandInterface *>(m_wm);
if (wI) {
wI->initWindowManagement(pwm);
}
});
registry->setup();
2018-03-01 01:52:05 +03:00
connection->roundtrip();
}
KWayland::Client::PlasmaShell *Corona::waylandCoronaInterface() const
{
return m_waylandCorona;
}
void Corona::cleanConfig()
{
auto containmentsEntries = config()->group("Containments");
bool changed = false;
foreach (auto cId, containmentsEntries.groupList()) {
2017-02-06 04:55:41 +03:00
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()) {
2017-02-06 04:55:41 +03:00
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 Corona::containmentExists(uint id) const
{
foreach (auto containment, containments()) {
if (id == containment->id()) {
return true;
}
}
return false;
}
bool Corona::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 *Corona::activitiesConsumer() const
{
return m_activityConsumer;
}
ScreenPool *Corona::screenPool() const
2017-02-24 21:58:21 +03:00
{
return m_screenPool;
}
UniversalSettings *Corona::universalSettings() const
{
return m_universalSettings;
}
LayoutManager *Corona::layoutManager() const
{
return m_layoutManager;
}
AbstractWindowInterface *Corona::wm() const
{
return m_wm;
}
PlasmaExtended::Theme *Corona::themeExtended() const
{
return m_themeExtended;
}
int Corona::numScreens() const
{
return qGuiApp->screens().count();
}
QRect Corona::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);
2017-01-16 22:07:49 +03:00
foreach (auto scr, screens) {
if (scr->name() == screenName) {
screen = scr;
break;
}
}
2017-01-16 22:07:49 +03:00
return screen->geometry();
}
QRegion Corona::availableScreenRegion(int id) const
{
return availableScreenRegionWithCriteria(id);
}
QRegion Corona::availableScreenRegionWithCriteria(int id, QString forLayout) const
{
const auto screens = qGuiApp->screens();
const QScreen *screen{qGuiApp->primaryScreen()};
2017-03-20 20:00:28 +03:00
QString screenName;
if (m_screenPool->knownIds().contains(id))
screenName = m_screenPool->connector(id);
foreach (auto scr, screens) {
2017-03-20 20:00:28 +03:00
if (scr->name() == screenName) {
screen = scr;
break;
}
}
if (!screen)
return QRegion();
QHash<const Plasma::Containment *, Latte::View *> *views;
if (forLayout.isEmpty()) {
views = m_layoutManager->currentLatteViews();
} else {
views = m_layoutManager->layoutLatteViews(forLayout);
}
QRegion available(screen->geometry());
2018-01-14 18:30:29 +03:00
if (views) {
for (const auto *view : *views) {
if (view && view->containment() && view->screen() == screen
&& view->visibility() && (view->visibility()->mode() != Latte::Types::AutoHide)) {
int realThickness = view->normalThickness() - view->effects()->innerShadow();
2018-01-14 18:30:29 +03:00
// Usually availableScreenRect is used by the desktop,
2018-08-30 08:47:53 +03:00
// but Latte don't have desktop, then here just
2018-01-14 18:30:29 +03:00
// need calculate available space for top and bottom location,
// because the left and right are those who dodge others views
2018-01-14 18:30:29 +03:00
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::Types::Left:
2018-01-14 18:30:29 +03:00
realGeometry = QRect(view->x(), view->y(),
realWidth, realThickness);
break;
case Latte::Types::Center:
case Latte::Types::Justify:
2018-07-03 22:15:45 +03:00
realGeometry = QRect(qMax(view->geometry().x(), view->geometry().center().x() - realWidth / 2), view->y(),
realWidth, realThickness);
2018-01-14 18:30:29 +03:00
break;
case Latte::Types::Right:
2018-01-14 18:30:29 +03:00
realGeometry = QRect(view->geometry().right() - realWidth + 1, view->y(),
realWidth, realThickness);
break;
}
available -= realGeometry;
}
2018-01-14 18:30:29 +03:00
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::Types::Left:
2018-01-14 18:30:29 +03:00
realGeometry = QRect(view->x(), realY,
realWidth, realThickness);
break;
case Latte::Types::Center:
case Latte::Types::Justify:
2018-01-14 18:30:29 +03:00
realGeometry = QRect(qMax(view->geometry().x(), view->geometry().center().x() - realWidth / 2),
realY, realWidth, realThickness);
break;
case Latte::Types::Right:
2018-01-14 18:30:29 +03:00
realGeometry = QRect(view->geometry().right() - realWidth + 1, realY,
realWidth, realThickness);
break;
}
available -= realGeometry;
}
2018-01-14 18:30:29 +03:00
break;
2018-01-14 18:30:29 +03:00
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 Corona::availableScreenRect(int id) const
{
return availableScreenRectWithCriteria(id);
}
QRect Corona::availableScreenRectWithCriteria(int id, QList<Types::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;
}
}
}
2017-02-11 06:12:17 +03:00
if (!screen)
return {};
2017-01-16 22:07:49 +03:00
bool allModes = modes.isEmpty();
bool allEdges = edges.isEmpty();
2017-02-11 06:12:17 +03:00
auto available = screen->geometry();
QHash<const Plasma::Containment *, Latte::View *> *views = m_layoutManager->currentLatteViews();
2018-01-14 18:30:29 +03:00
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()))))) {
2018-01-14 18:30:29 +03:00
auto dockRect = view->absGeometry();
2018-01-14 18:30:29 +03:00
// Usually availableScreenRect is used by the desktop,
2018-08-30 08:47:53 +03:00
// but Latte don't have desktop, then here just
2018-01-14 18:30:29 +03:00
// 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;
2018-01-14 18:30:29 +03:00
case Plasma::Types::BottomEdge:
available.setBottom(dockRect.top() - 1);
break;
2018-01-14 18:30:29 +03:00
case Plasma::Types::LeftEdge:
available.setLeft(dockRect.right() + 1);
2018-01-14 18:30:29 +03:00
break;
2018-01-14 18:30:29 +03:00
case Plasma::Types::RightEdge:
available.setRight(dockRect.left() - 1);
2018-01-14 18:30:29 +03:00
break;
2018-01-14 18:30:29 +03:00
default:
//! bypass clang warnings
break;
}
}
}
}
2017-02-11 06:12:17 +03:00
return available;
}
void Corona::addOutput(QScreen *screen)
2017-02-24 21:58:21 +03:00
{
Q_ASSERT(screen);
int id = m_screenPool->id(screen->name());
if (id == -1) {
int newId = m_screenPool->firstAvailableId();
m_screenPool->insertScreenMapping(newId, screen->name());
}
connect(screen, &QScreen::geometryChanged, this, [ = ]() {
const int id = m_screenPool->id(screen->name());
if (id >= 0) {
emit screenGeometryChanged(id);
emit availableScreenRegionChanged();
emit availableScreenRectChanged();
}
});
emit availableScreenRectChanged();
emit screenAdded(m_screenPool->id(screen->name()));
2017-02-24 21:58:21 +03:00
}
void Corona::primaryOutputChanged()
2017-02-24 21:58:21 +03:00
{
m_viewsScreenSyncTimer.start();
2017-02-24 21:58:21 +03:00
}
void Corona::screenRemoved(QScreen *screen)
2017-02-24 21:58:21 +03:00
{
Q_ASSERT(screen);
}
void Corona::screenCountChanged()
{
m_viewsScreenSyncTimer.start();
}
//! the central functions that updates loading/unloading latteviews
//! concerning screen changed (for multi-screen setups mainly)
void Corona::syncLatteViewsToScreens()
{
m_layoutManager->syncLatteViewsToScreens();
2017-02-24 21:58:21 +03:00
}
int Corona::primaryScreenId() const
{
2017-02-24 21:58:21 +03:00
return m_screenPool->id(qGuiApp->primaryScreen()->name());
}
void Corona::closeApplication()
{
//! this code must be called asynchronously because it is called
//! also from qml (Settings window).
QTimer::singleShot(5, [this]() {
m_layoutManager->hideLatteSettingsDialog();
m_layoutManager->hideAllViews();
});
//! give the time for the views to hide themselves
QTimer::singleShot(500, [this]() {
qGuiApp->quit();
});
}
void Corona::aboutApplication()
2017-01-29 08:16:28 +03:00
{
if (aboutDialog) {
aboutDialog->hide();
aboutDialog->deleteLater();
}
aboutDialog = new KAboutApplicationDialog(KAboutData::applicationData());
connect(aboutDialog.data(), &QDialog::finished, aboutDialog.data(), &QObject::deleteLater);
m_wm->skipTaskBar(*aboutDialog);
m_wm->setKeepAbove(*aboutDialog, true);
2017-01-29 08:16:28 +03:00
aboutDialog->show();
}
int Corona::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 latteView 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
2017-02-24 21:58:21 +03:00
//case in which this containment is child of an applet, hello systray :)
2018-01-14 18:30:29 +03:00
2017-02-24 21:58:21 +03:00
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 *, Latte::View *> *views = m_layoutManager->currentLatteViews();
2017-02-24 21:58:21 +03:00
//if the panel views already exist, base upon them
Latte::View *view = views ? views->value(containment) : nullptr;
2017-02-24 21:58:21 +03:00
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();
2017-02-24 21:58:21 +03:00
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)) {
2017-02-24 21:58:21 +03:00
return containment->lastScreen();
}
}
return -1;
}
void Corona::showAlternativesForApplet(Plasma::Applet *applet)
{
const QString alternativesQML = kPackage().filePath("appletalternativesui");
2017-02-24 21:58:21 +03:00
if (alternativesQML.isEmpty()) {
return;
}
QHash<const Plasma::Containment *, Latte::View *> *views = m_layoutManager->currentLatteViews();
Latte::View *latteView = (*views)[applet->containment()];
KDeclarative::QmlObject *qmlObj{nullptr};
if (latteView) {
latteView->setAlternativesIsShown(true);
qmlObj = new KDeclarative::QmlObject(latteView);
} 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, [latteView]() {
latteView->setAlternativesIsShown(false);
});
connect(qmlObj->rootObject(), SIGNAL(visibleChanged(bool)),
this, SLOT(alternativesVisibilityChanged(bool)));
2017-02-24 21:58:21 +03:00
connect(applet, &Plasma::Applet::destroyedChanged, this, [this, qmlObj](bool destroyed) {
if (!destroyed) {
return;
}
2017-02-24 21:58:21 +03:00
QMutableListIterator<KDeclarative::QmlObject *> it(m_alternativesObjects);
2017-02-24 21:58:21 +03:00
while (it.hasNext()) {
KDeclarative::QmlObject *obj = it.next();
2017-02-24 21:58:21 +03:00
if (obj == qmlObj) {
it.remove();
obj->deleteLater();
}
}
});
}
void Corona::alternativesVisibilityChanged(bool visible)
{
if (visible) {
return;
}
QObject *root = sender();
QMutableListIterator<KDeclarative::QmlObject *> it(m_alternativesObjects);
2017-02-24 21:58:21 +03:00
while (it.hasNext()) {
KDeclarative::QmlObject *obj = it.next();
2017-02-24 21:58:21 +03:00
if (obj->rootObject() == root) {
it.remove();
obj->deleteLater();
}
}
}
void Corona::loadDefaultLayout()
{
qDebug() << "loading default layout";
//! Settting mutable for create a containment
setImmutability(Plasma::Types::Mutable);
QVariantList args;
2016-12-25 16:58:14 +03:00
auto defaultContainment = createContainmentDelayed("org.kde.latte.containment", args);
defaultContainment->setContainmentType(Plasma::Types::PanelContainment);
defaultContainment->init();
2017-01-16 22:07:49 +03:00
if (!defaultContainment || !defaultContainment->kPackage().isValid()) {
qWarning() << "the requested containment plugin can not be located or loaded";
return;
}
2017-01-16 22:07:49 +03:00
auto config = defaultContainment->config();
2016-12-28 16:17:36 +03:00
defaultContainment->restore(config);
using Plasma::Types;
QList<Types::Location> edges{Types::BottomEdge, Types::LeftEdge,
Types::TopEdge, Types::RightEdge};
Layout *currentLayout = m_layoutManager->activeLayout(m_layoutManager->currentLayoutName());
if (currentLayout) {
edges = currentLayout->freeEdges(defaultContainment->screen());
}
2017-01-16 22:07:49 +03:00
if ((edges.count() > 0)) {
defaultContainment->setLocation(edges.at(0));
} else {
defaultContainment->setLocation(Plasma::Types::BottomEdge);
}
2017-01-16 22:07:49 +03:00
if (m_layoutManager->memoryUsage() == Latte::Types::MultipleLayouts) {
config.writeEntry("layoutId", m_layoutManager->currentLayoutName());
}
2016-12-28 16:17:36 +03:00
defaultContainment->updateConstraints(Plasma::Types::StartupCompletedConstraint);
2016-12-28 16:17:36 +03:00
defaultContainment->save(config);
requestConfigSync();
2016-12-28 16:17:36 +03:00
defaultContainment->flushPendingConstraintsEvents();
emit containmentAdded(defaultContainment);
emit containmentCreated(defaultContainment);
2016-12-29 00:07:17 +03:00
defaultContainment->createApplet(QStringLiteral("org.kde.latte.plasmoid"));
defaultContainment->createApplet(QStringLiteral("org.kde.plasma.analogclock"));
}
QStringList Corona::containmentsIds()
{
2018-01-14 18:30:29 +03:00
QStringList ids;
2018-01-14 18:30:29 +03:00
foreach (auto containment, containments()) {
ids << QString::number(containment->id());
}
return ids;
}
QStringList Corona::appletsIds()
{
QStringList ids;
2018-01-14 18:30:29 +03:00
foreach (auto containment, containments()) {
auto applets = containment->config().group("Applets");
ids << applets.groupList();
}
return ids;
}
//! Activate launcher menu through dbus interface
void Corona::activateLauncherMenu()
{
m_globalShortcuts->activateLauncherMenu();
}
void Corona::windowColorScheme(QString windowIdAndScheme)
{
int firstSlash = windowIdAndScheme.indexOf("-");
QString windowIdStr = windowIdAndScheme.mid(0, firstSlash);
QString schemeStr = windowIdAndScheme.mid(firstSlash + 1);
m_wm->setColorSchemeForWindow(windowIdStr, schemeStr);
}
//! update badge for specific view item
void Corona::updateDockItemBadge(QString identifier, QString value)
{
m_globalShortcuts->updateDockItemBadge(identifier, value);
}
void Corona::switchToLayout(QString layout)
{
m_layoutManager->switchToLayout(layout);
}
void Corona::showSettingsWindow(int page)
{
Types::LatteConfigPage p = Types::LayoutPage;
if (page >= Types::LayoutPage && page <= Types::PreferencesPage) {
p = static_cast<Types::LatteConfigPage>(page);
}
m_layoutManager->showLatteSettingsDialog(p);
}
QStringList Corona::contextMenuData()
2018-02-07 00:39:45 +03:00
{
QStringList data;
data << QString::number((int)m_layoutManager->memoryUsage());
data << m_layoutManager->currentLayoutName();
foreach (auto layoutName, m_layoutManager->menuLayouts()) {
if (m_layoutManager->activeLayout(layoutName)) {
data << QString("1," + layoutName);
} else {
data << QString("0," + layoutName);
}
}
return data;
}
inline void Corona::qmlRegisterTypes() const
{
qmlRegisterType<QScreen>();
}
2016-12-30 10:20:06 +03:00
}