1
0
mirror of https://github.com/KDE/latte-dock.git synced 2025-01-24 18:03:53 +03:00
latte-dock/app/layout/activelayout.cpp

455 lines
12 KiB
C++
Raw Normal View History

/*
* Copyright 2017 Smith AR <audoban@openmailbox.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 "activelayout.h"
2018-12-02 02:05:52 +02:00
// local
#include "toplayout.h"
2019-01-22 18:17:55 +02:00
#include "../lattecorona.h"
#include "../layoutmanager.h"
#include "../screenpool.h"
#include "../settings/universalsettings.h"
#include "../view/view.h"
#include "../../liblatte2/types.h"
2018-01-07 19:59:18 +02:00
2018-12-02 02:05:52 +02:00
// Qt
#include <QtDBus/QtDBus>
2018-12-02 02:05:52 +02:00
// KDE
#include <KConfigGroup>
2018-12-02 02:05:52 +02:00
#include <KActivities/Consumer>
namespace Latte {
ActiveLayout::ActiveLayout(QObject *parent, QString layoutFile, QString assignedName)
: Layout::GenericLayout(parent, layoutFile, assignedName)
{
if (m_loadedCorrectly) {
loadConfig();
init();
}
}
ActiveLayout::~ActiveLayout()
{
if (!m_layoutFile.isEmpty()) {
m_layoutGroup.sync();
}
2018-01-10 19:48:23 +02:00
}
void ActiveLayout::unloadContainments()
{
Layout::GenericLayout::unloadContainments();
if (m_topLayout) {
disconnect(m_topLayout, &Layout::GenericLayout::viewsCountChanged, this, &Layout::GenericLayout::viewsCountChanged);
m_topLayout->removeActiveLayout(this);
}
}
void ActiveLayout::init()
{
connect(this, &ActiveLayout::activitiesChanged, this, &ActiveLayout::saveConfig);
connect(this, &ActiveLayout::disableBordersForMaximizedWindowsChanged, this, &ActiveLayout::saveConfig);
connect(this, &ActiveLayout::showInMenuChanged, this, &ActiveLayout::saveConfig);
connect(this, &ActiveLayout::topLayoutNameChanged, this, &ActiveLayout::saveConfig);
}
void ActiveLayout::initToCorona(Latte::Corona *corona)
{
if (GenericLayout::initToCorona(corona)) {
connect(m_corona->universalSettings(), &UniversalSettings::canDisableBordersChanged, this, [&]() {
if (m_corona->universalSettings()->canDisableBorders()) {
kwin_setDisabledMaximizedBorders(disableBordersForMaximizedWindows());
} else {
kwin_setDisabledMaximizedBorders(false);
}
});
if (m_corona->layoutManager()->memoryUsage() == Types::SingleLayout && m_corona->universalSettings()->canDisableBorders()) {
kwin_setDisabledMaximizedBorders(disableBordersForMaximizedWindows());
} else if (m_corona->layoutManager()->memoryUsage() == Types::MultipleLayouts) {
connect(m_corona->layoutManager(), &LayoutManager::currentLayoutNameChanged, this, [&]() {
if (m_corona->universalSettings()->canDisableBorders()
&& m_corona->layoutManager()->currentLayoutName() == name()) {
kwin_setDisabledMaximizedBorders(disableBordersForMaximizedWindows());
}
});
}
2018-12-01 18:20:55 +02:00
//! Request the TopLayout in case there is one and Latte is functioning in MultipleLayouts mode
if (m_corona->layoutManager()->memoryUsage() == Types::MultipleLayouts && !m_topLayoutName.isEmpty()) {
if (m_corona->layoutManager()->assignActiveToTopLayout(this, m_topLayoutName)) {
setTopLayout(m_corona->layoutManager()->topLayout(m_topLayoutName));
}
}
}
}
bool ActiveLayout::disableBordersForMaximizedWindows() const
{
return m_disableBordersForMaximizedWindows;
}
void ActiveLayout::setDisableBordersForMaximizedWindows(bool disable)
{
if (m_disableBordersForMaximizedWindows == disable) {
return;
}
m_disableBordersForMaximizedWindows = disable;
kwin_setDisabledMaximizedBorders(disable);
emit disableBordersForMaximizedWindowsChanged();
}
bool ActiveLayout::kwin_disabledMaximizedBorders() const
{
//! Identify Plasma Desktop version
QProcess process;
process.start("kreadconfig5 --file kwinrc --group Windows --key BorderlessMaximizedWindows");
process.waitForFinished();
QString output(process.readAllStandardOutput());
output = output.remove("\n");
return (output == "true");
}
void ActiveLayout::kwin_setDisabledMaximizedBorders(bool disable)
{
if (kwin_disabledMaximizedBorders() == disable) {
return;
}
QString disableText = disable ? "true" : "false";
QProcess process;
QString commandStr = "kwriteconfig5 --file kwinrc --group Windows --key BorderlessMaximizedWindows --type bool " + disableText;
process.start(commandStr);
process.waitForFinished();
QDBusInterface iface("org.kde.KWin", "/KWin", "", QDBusConnection::sessionBus());
if (iface.isValid()) {
iface.call("reconfigure");
}
}
bool ActiveLayout::showInMenu() const
{
return m_showInMenu;
}
void ActiveLayout::setShowInMenu(bool show)
{
if (m_showInMenu == show) {
return;
}
m_showInMenu = show;
emit showInMenuChanged();
}
QStringList ActiveLayout::activities() const
{
return m_activities;
}
void ActiveLayout::setActivities(QStringList activities)
{
if (m_activities == activities) {
return;
}
m_activities = activities;
emit activitiesChanged();
}
QString ActiveLayout::topLayoutName() const
{
return m_topLayoutName;
}
void ActiveLayout::setTopLayoutName(QString name)
{
if (m_topLayoutName == name) {
return;
}
m_topLayoutName = name;
emit topLayoutNameChanged();
}
void ActiveLayout::setTopLayout(TopLayout *layout)
{
if (m_topLayout == layout) {
return;
}
disconnect(m_topLayout, &Layout::GenericLayout::viewsCountChanged, this, &Layout::GenericLayout::viewsCountChanged);
m_topLayout = layout;
connect(m_topLayout, &Layout::GenericLayout::viewsCountChanged, this, &Layout::GenericLayout::viewsCountChanged);
emit viewsCountChanged();
2018-01-14 17:30:29 +02:00
}
bool ActiveLayout::isActiveLayout() const
{
if (!m_corona) {
return false;
}
ActiveLayout *activeLayout = m_corona->layoutManager()->activeLayout(m_layoutName);
if (activeLayout) {
return true;
} else {
return false;
}
}
bool ActiveLayout::isOriginalLayout() const
{
return m_layoutName != MultipleLayoutsName;
}
void ActiveLayout::loadConfig()
{
m_disableBordersForMaximizedWindows = m_layoutGroup.readEntry("disableBordersForMaximizedWindows", false);
m_showInMenu = m_layoutGroup.readEntry("showInMenu", false);
m_topLayoutName = m_layoutGroup.readEntry("topLayoutName", QString());
m_activities = m_layoutGroup.readEntry("activities", QStringList());
emit activitiesChanged();
}
//! OVERRIDES
void ActiveLayout::saveConfig()
{
qDebug() << "active layout is saving... for layout:" << m_layoutName;
m_layoutGroup.writeEntry("showInMenu", m_showInMenu);
m_layoutGroup.writeEntry("disableBordersForMaximizedWindows", m_disableBordersForMaximizedWindows);
m_layoutGroup.writeEntry("topLayoutName", m_topLayoutName);
m_layoutGroup.writeEntry("activities", m_activities);
m_layoutGroup.sync();
}
const QStringList ActiveLayout::appliedActivities()
2018-01-10 19:48:23 +02:00
{
if (!m_corona) {
return {};
2018-01-10 19:48:23 +02:00
}
if (m_corona->layoutManager()->memoryUsage() == Types::SingleLayout) {
return {"0"};
} else if (m_corona->layoutManager()->memoryUsage() == Types::MultipleLayouts) {
if (m_activities.isEmpty()) {
return m_corona->layoutManager()->orphanedActivities();
} else {
return m_activities;
}
2018-05-30 22:18:19 +03:00
} else {
return {"0"};
}
}
QList<Latte::View *> ActiveLayout::latteViews()
{
if (m_topLayout) {
QList<Latte::View *> views = Layout::GenericLayout::latteViews();
views << m_topLayout->latteViews();
return views;
}
return Layout::GenericLayout::latteViews();
}
int ActiveLayout::viewsCount(int screen) const
{
if (!m_corona) {
return 0;
}
int views = Layout::GenericLayout::viewsCount(screen);
if (m_topLayout) {
QScreen *scr = m_corona->screenPool()->screenForId(screen);
for (const auto view : m_topLayout->latteViews()) {
if (view && view->screen() == scr && !view->containment()->destroyed()) {
++views;
}
}
}
return views;
}
int ActiveLayout::viewsCount(QScreen *screen) const
{
if (!m_corona) {
return 0;
}
int views = Layout::GenericLayout::viewsCount(screen);
if (m_topLayout) {
for (const auto view : m_topLayout->latteViews()) {
if (view && view->screen() == screen && !view->containment()->destroyed()) {
++views;
}
}
}
return views;
}
int ActiveLayout::viewsCount() const
{
if (!m_corona) {
return 0;
}
int views = Layout::GenericLayout::viewsCount();
if (m_topLayout) {
for (const auto view : m_topLayout->latteViews()) {
if (view && view->containment() && !view->containment()->destroyed()) {
++views;
}
}
}
return views;
}
QList<Plasma::Types::Location> ActiveLayout::availableEdgesForView(QScreen *scr, Latte::View *forView) const
{
using Plasma::Types;
QList<Types::Location> edges{Types::BottomEdge, Types::LeftEdge,
Types::TopEdge, Types::RightEdge};
if (!m_corona) {
return edges;
}
edges = Layout::GenericLayout::availableEdgesForView(scr, forView);
if (m_topLayout) {
for (const auto view : m_topLayout->latteViews()) {
//! make sure that availabe edges takes into account only views that should be excluded,
//! this is why the forView should not be excluded
if (view && view != forView && view->positioner()->currentScreenName() == scr->name()) {
edges.removeOne(view->location());
}
}
}
return edges;
}
QList<Plasma::Types::Location> ActiveLayout::freeEdges(QScreen *scr) const
{
using Plasma::Types;
QList<Types::Location> edges{Types::BottomEdge, Types::LeftEdge,
Types::TopEdge, Types::RightEdge};
if (!m_corona) {
return edges;
}
edges = Layout::GenericLayout::freeEdges(scr);
if (m_topLayout) {
for (const auto view : m_topLayout->latteViews()) {
if (view && view->positioner()->currentScreenName() == scr->name()) {
edges.removeOne(view->location());
}
}
}
return edges;
}
QList<Plasma::Types::Location> ActiveLayout::freeEdges(int screen) const
{
using Plasma::Types;
QList<Types::Location> edges{Types::BottomEdge, Types::LeftEdge,
Types::TopEdge, Types::RightEdge};
if (!m_corona) {
return edges;
}
edges = Layout::GenericLayout::freeEdges(screen);
QScreen *scr = m_corona->screenPool()->screenForId(screen);
if (m_topLayout) {
for (const auto view : m_topLayout->latteViews()) {
if (view && scr && view->positioner()->currentScreenName() == scr->name()) {
edges.removeOne(view->location());
}
}
}
return edges;
}
QList<Latte::View *> ActiveLayout::sortedLatteViews(QList<Latte::View *> views)
{
QList<Latte::View *> combined = latteViews();
if (m_topLayout) {
combined << m_topLayout->latteViews();
}
return Layout::GenericLayout::sortedLatteViews(combined);
}
QList<Latte::View *> ActiveLayout::viewsWithPlasmaShortcuts()
{
QList<Latte::View *> combined = Layout::GenericLayout::viewsWithPlasmaShortcuts();
if (m_topLayout) {
combined << m_topLayout->viewsWithPlasmaShortcuts();
}
return combined;
}
void ActiveLayout::syncLatteViewsToScreens(Layout::ViewsMap *occupiedMap)
{
if (m_topLayout) {
Layout::ViewsMap map = m_topLayout->validViewsMap();
Layout::GenericLayout::syncLatteViewsToScreens(&map);
} else {
Layout::GenericLayout::syncLatteViewsToScreens();
}
}
}