1
0
mirror of https://github.com/KDE/latte-dock.git synced 2025-01-15 03:24:49 +03:00
latte-dock/app/layouts/storage.cpp
2020-08-20 17:28:29 +03:00

1047 lines
34 KiB
C++

/*
* Copyright 2020 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 "storage.h"
// local
#include "importer.h"
#include "manager.h"
#include "../lattecorona.h"
#include "../screenpool.h"
#include "../layout/abstractlayout.h"
#include "../view/view.h"
// Qt
#include <QDebug>
#include <QDir>
#include <QFile>
#include <QFileInfo>
// KDE
#include <KConfigGroup>
#include <KPluginMetaData>
#include <KSharedConfig>
#include <KPackage/Package>
#include <KPackage/PackageLoader>
// Plasma
#include <Plasma>
#include <Plasma/Applet>
#include <Plasma/Containment>
namespace Latte {
namespace Layouts {
const int Storage::IDNULL = -1;
const int Storage::IDBASE = 0;
Storage::Storage()
{
qDebug() << " >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> LAYOUTS::STORAGE, TEMP DIR ::: " << m_storageTmpDir.path();
SubContaimentIdentityData data;
//! Systray
m_subIdentities << SubContaimentIdentityData{.cfgGroup="Configuration", .cfgProperty="SystrayContainmentId"};
//! Group applet
m_subIdentities << SubContaimentIdentityData{.cfgGroup="Configuration", .cfgProperty="ContainmentId"};
}
Storage::~Storage()
{
}
Storage *Storage::self()
{
static Storage store;
return &store;
}
bool Storage::isWritable(const Layout::GenericLayout *layout) const
{
QFileInfo layoutFileInfo(layout->file());
if (layoutFileInfo.exists() && !layoutFileInfo.isWritable()) {
return false;
} else {
return true;
}
}
bool Storage::isLatteContainment(Plasma::Containment *containment) const
{
if (!containment) {
return false;
}
if (containment->pluginMetaData().pluginId() == "org.kde.latte.containment") {
return true;
}
return false;
}
bool Storage::isLatteContainment(const KConfigGroup &group) const
{
QString pluginId = group.readEntry("plugin", "");
return pluginId == "org.kde.latte.containment";
}
bool Storage::isSubContainment(const Layout::GenericLayout *layout, const Plasma::Applet *applet) const
{
if (!layout || !applet) {
return false;
}
for (const auto containment : *layout->containments()) {
Plasma::Applet *parentApplet = qobject_cast<Plasma::Applet *>(containment->parent());
if (parentApplet && parentApplet == applet) {
return true;
}
}
return false;
}
bool Storage::isSubContainment(const KConfigGroup &appletGroup) const
{
return isValid(subContainmentId(appletGroup));
}
bool Storage::isValid(const int &id)
{
return id >= IDBASE;
}
int Storage::subContainmentId(const KConfigGroup &appletGroup) const
{
//! cycle through subcontainments identities
for (auto subidentity : m_subIdentities) {
KConfigGroup appletConfigGroup = appletGroup;
if (!subidentity.cfgGroup.isEmpty()) {
//! if identity provides specific configuration group
if (appletConfigGroup.hasGroup(subidentity.cfgGroup)) {
appletConfigGroup = appletGroup.group(subidentity.cfgGroup);
}
}
if (!subidentity.cfgProperty.isEmpty()) {
//! if identity provides specific property for configuration group
if (appletConfigGroup.hasKey(subidentity.cfgProperty)) {
return appletConfigGroup.readEntry(subidentity.cfgProperty, IDNULL);
}
}
}
return IDNULL;
}
int Storage::subIdentityIndex(const KConfigGroup &appletGroup) const
{
if (!isSubContainment(appletGroup)) {
return IDNULL;
}
//! cycle through subcontainments identities
for (int i=0; i<m_subIdentities.count(); ++i) {
KConfigGroup appletConfigGroup = appletGroup;
if (!m_subIdentities[i].cfgGroup.isEmpty()) {
//! if identity provides specific configuration group
if (appletConfigGroup.hasGroup(m_subIdentities[i].cfgGroup)) {
appletConfigGroup = appletGroup.group(m_subIdentities[i].cfgGroup);
}
}
if (!m_subIdentities[i].cfgProperty.isEmpty()) {
//! if identity provides specific property for configuration group
if (appletConfigGroup.hasKey(m_subIdentities[i].cfgProperty)) {
int subId = appletConfigGroup.readEntry(m_subIdentities[i].cfgProperty, IDNULL);
return isValid(subId) ? i : IDNULL;
}
}
}
return IDNULL;
}
Plasma::Containment *Storage::subContainmentOf(const Layout::GenericLayout *layout, const Plasma::Applet *applet)
{
if (!layout || !applet) {
return nullptr;
}
if (isSubContainment(layout, applet)) {
for (const auto containment : *layout->containments()) {
Plasma::Applet *parentApplet = qobject_cast<Plasma::Applet *>(containment->parent());
if (parentApplet && parentApplet == applet) {
return containment;
}
}
}
return nullptr;
}
void Storage::lock(const Layout::GenericLayout *layout)
{
QFileInfo layoutFileInfo(layout->file());
if (layoutFileInfo.exists() && layoutFileInfo.isWritable()) {
QFile(layout->file()).setPermissions(QFileDevice::ReadUser | QFileDevice::ReadGroup | QFileDevice::ReadOther);
}
}
void Storage::unlock(const Layout::GenericLayout *layout)
{
QFileInfo layoutFileInfo(layout->file());
if (layoutFileInfo.exists() && !layoutFileInfo.isWritable()) {
QFile(layout->file()).setPermissions(QFileDevice::ReadUser | QFileDevice::WriteUser | QFileDevice::ReadGroup | QFileDevice::ReadOther);
}
}
void Storage::importToCorona(const Layout::GenericLayout *layout)
{
if (!layout->corona()) {
return;
}
//! Setting mutable for create a containment
layout->corona()->setImmutability(Plasma::Types::Mutable);
QString temp1FilePath = m_storageTmpDir.path() + "/" + layout->name() + ".multiple.views";
//! we need to copy first the layout file because the kde cache
//! may not have yet been updated (KSharedConfigPtr)
//! this way we make sure at the latest changes stored in the layout file
//! will be also available when changing to Multiple Layouts
QString tempLayoutFilePath = m_storageTmpDir.path() + "/" + layout->name() + ".multiple.tmplayout";
//! WE NEED A WAY TO COPY A CONTAINMENT!!!!
QFile tempLayoutFile(tempLayoutFilePath);
QFile copyFile(temp1FilePath);
QFile layoutOriginalFile(layout->file());
if (tempLayoutFile.exists()) {
tempLayoutFile.remove();
}
if (copyFile.exists())
copyFile.remove();
layoutOriginalFile.copy(tempLayoutFilePath);
KSharedConfigPtr filePtr = KSharedConfig::openConfig(tempLayoutFilePath);
KSharedConfigPtr newFile = KSharedConfig::openConfig(temp1FilePath);
KConfigGroup copyGroup = KConfigGroup(newFile, "Containments");
KConfigGroup current_containments = KConfigGroup(filePtr, "Containments");
current_containments.copyTo(&copyGroup);
copyGroup.sync();
//! update ids to unique ones
QString temp2File = newUniqueIdsLayoutFromFile(layout, temp1FilePath);
//! Finally import the configuration
importLayoutFile(layout, temp2File);
}
QString Storage::availableId(QStringList all, QStringList assigned, int base)
{
bool found = false;
int i = base;
while (!found && i < 32000) {
QString iStr = QString::number(i);
if (!all.contains(iStr) && !assigned.contains(iStr)) {
return iStr;
}
i++;
}
return QString("");
}
bool Storage::appletGroupIsValid(const KConfigGroup &appletGroup)
{
return !( appletGroup.keyList().count() == 0
&& appletGroup.groupList().count() == 1
&& appletGroup.groupList().at(0) == "Configuration"
&& appletGroup.group("Configuration").keyList().count() == 1
&& appletGroup.group("Configuration").hasKey("PreloadWeight") );
}
QString Storage::newUniqueIdsLayoutFromFile(const Layout::GenericLayout *layout, QString file)
{
if (!layout->corona()) {
return QString();
}
QString tempFile = m_storageTmpDir.path() + "/" + layout->name() + ".views.newids";
QFile copyFile(tempFile);
if (copyFile.exists()) {
copyFile.remove();
}
//! BEGIN updating the ids in the temp file
QStringList allIds;
allIds << layout->corona()->containmentsIds();
allIds << layout->corona()->appletsIds();
QStringList toInvestigateContainmentIds;
QStringList toInvestigateAppletIds;
QStringList toInvestigateSubContIds;
//! first is the subcontainment id
QHash<QString, QString> subParentContainmentIds;
QHash<QString, QString> subAppletIds;
//qDebug() << "Ids:" << allIds;
//qDebug() << "to copy containments: " << toCopyContainmentIds;
//qDebug() << "to copy applets: " << toCopyAppletIds;
QStringList assignedIds;
QHash<QString, QString> assigned;
KSharedConfigPtr filePtr = KSharedConfig::openConfig(file);
KConfigGroup investigate_conts = KConfigGroup(filePtr, "Containments");
//! Record the containment and applet ids
for (const auto &cId : investigate_conts.groupList()) {
toInvestigateContainmentIds << cId;
auto appletsEntries = investigate_conts.group(cId).group("Applets");
toInvestigateAppletIds << appletsEntries.groupList();
//! investigate for subcontainments
for (const auto &appletId : appletsEntries.groupList()) {
int subId = subContainmentId(appletsEntries.group(appletId));
//! It is a subcontainment !!!
if (isValid(subId)) {
QString tSubIdStr = QString::number(subId);
toInvestigateSubContIds << tSubIdStr;
subParentContainmentIds[tSubIdStr] = cId;
subAppletIds[tSubIdStr] = appletId;
qDebug() << "subcontainment was found in the containment...";
}
}
}
//! Reassign containment and applet ids to unique ones
for (const auto &contId : toInvestigateContainmentIds) {
QString newId = availableId(allIds, assignedIds, 12);
assignedIds << newId;
assigned[contId] = newId;
}
for (const auto &appId : toInvestigateAppletIds) {
QString newId = availableId(allIds, assignedIds, 40);
assignedIds << newId;
assigned[appId] = newId;
}
qDebug() << "ALL CORONA IDS ::: " << allIds;
qDebug() << "FULL ASSIGNMENTS ::: " << assigned;
for (const auto &cId : toInvestigateContainmentIds) {
QString value = assigned[cId];
if (assigned.contains(value)) {
QString value2 = assigned[value];
if (cId != assigned[cId] && !value2.isEmpty() && cId == value2) {
qDebug() << "PROBLEM APPEARED !!!! FOR :::: " << cId << " .. fixed ..";
assigned[cId] = cId;
assigned[value] = value;
}
}
}
for (const auto &aId : toInvestigateAppletIds) {
QString value = assigned[aId];
if (assigned.contains(value)) {
QString value2 = assigned[value];
if (aId != assigned[aId] && !value2.isEmpty() && aId == value2) {
qDebug() << "PROBLEM APPEARED !!!! FOR :::: " << aId << " .. fixed ..";
assigned[aId] = aId;
assigned[value] = value;
}
}
}
qDebug() << "FIXED FULL ASSIGNMENTS ::: " << assigned;
//! update applet ids in their containment order and in MultipleLayouts update also the layoutId
for (const auto &cId : investigate_conts.groupList()) {
//! Update options that contain applet ids
//! (appletOrder) and (lockedZoomApplets) and (userBlocksColorizingApplets)
QStringList options;
options << "appletOrder" << "lockedZoomApplets" << "userBlocksColorizingApplets";
for (const auto &settingStr : options) {
QString order1 = investigate_conts.group(cId).group("General").readEntry(settingStr, QString());
if (!order1.isEmpty()) {
QStringList order1Ids = order1.split(";");
QStringList fixedOrder1Ids;
for (int i = 0; i < order1Ids.count(); ++i) {
fixedOrder1Ids.append(assigned[order1Ids[i]]);
}
QString fixedOrder1 = fixedOrder1Ids.join(";");
investigate_conts.group(cId).group("General").writeEntry(settingStr, fixedOrder1);
}
}
if (layout->corona()->layoutsManager()->memoryUsage() == MemoryUsage::MultipleLayouts) {
investigate_conts.group(cId).writeEntry("layoutId", layout->name());
}
}
//! must update also the sub id in its applet
for (const auto &subId : toInvestigateSubContIds) {
KConfigGroup subParentContainment = investigate_conts.group(subParentContainmentIds[subId]);
KConfigGroup subAppletConfig = subParentContainment.group("Applets").group(subAppletIds[subId]);
int entityIndex = subIdentityIndex(subAppletConfig);
if (entityIndex >= 0) {
if (!m_subIdentities[entityIndex].cfgGroup.isEmpty()) {
subAppletConfig = subAppletConfig.group(m_subIdentities[entityIndex].cfgGroup);
}
if (!m_subIdentities[entityIndex].cfgProperty.isEmpty()) {
subAppletConfig.writeEntry(m_subIdentities[entityIndex].cfgProperty, assigned[subId]);
subParentContainment.sync();
}
}
}
investigate_conts.sync();
//! Copy To Temp 2 File And Update Correctly The Ids
KSharedConfigPtr file2Ptr = KSharedConfig::openConfig(tempFile);
KConfigGroup fixedNewContainmets = KConfigGroup(file2Ptr, "Containments");
for (const auto &contId : investigate_conts.groupList()) {
QString pluginId = investigate_conts.group(contId).readEntry("plugin", "");
if (pluginId != "org.kde.desktopcontainment") { //!don't add ghost containments
KConfigGroup newContainmentGroup = fixedNewContainmets.group(assigned[contId]);
investigate_conts.group(contId).copyTo(&newContainmentGroup);
newContainmentGroup.group("Applets").deleteGroup();
for (const auto &appId : investigate_conts.group(contId).group("Applets").groupList()) {
KConfigGroup appletGroup = investigate_conts.group(contId).group("Applets").group(appId);
KConfigGroup newAppletGroup = fixedNewContainmets.group(assigned[contId]).group("Applets").group(assigned[appId]);
appletGroup.copyTo(&newAppletGroup);
}
}
}
fixedNewContainmets.sync();
return tempFile;
}
void Storage::syncToLayoutFile(const Layout::GenericLayout *layout, bool removeLayoutId)
{
if (!layout->corona() || !isWritable(layout)) {
return;
}
KSharedConfigPtr filePtr = KSharedConfig::openConfig(layout->file());
KConfigGroup oldContainments = KConfigGroup(filePtr, "Containments");
oldContainments.deleteGroup();
qDebug() << " LAYOUT :: " << layout->name() << " is syncing its original file.";
for (const auto containment : *layout->containments()) {
if (removeLayoutId) {
containment->config().writeEntry("layoutId", "");
}
KConfigGroup newGroup = oldContainments.group(QString::number(containment->id()));
containment->config().copyTo(&newGroup);
if (!removeLayoutId) {
newGroup.writeEntry("layoutId", "");
newGroup.sync();
}
}
oldContainments.sync();
}
QList<Plasma::Containment *> Storage::importLayoutFile(const Layout::GenericLayout *layout, QString file)
{
KSharedConfigPtr filePtr = KSharedConfig::openConfig(file);
auto newContainments = layout->corona()->importLayout(KConfigGroup(filePtr, ""));
qDebug() << " imported containments ::: " << newContainments.length();
QList<Plasma::Containment *> importedDocks;
for (const auto containment : newContainments) {
if (isLatteContainment(containment)) {
qDebug() << "new latte containment id: " << containment->id();
importedDocks << containment;
}
}
return importedDocks;
}
ViewDelayedCreationData Storage::copyView(const Layout::GenericLayout *layout, Plasma::Containment *containment)
{
if (!containment || !layout->corona()) {
return ViewDelayedCreationData();
}
qDebug() << "copying containment layout";
//! Setting mutable for create a containment
layout->corona()->setImmutability(Plasma::Types::Mutable);
QString temp1File = m_storageTmpDir.path() + "/" + layout->name() + ".copy.view";
//! WE NEED A WAY TO COPY A CONTAINMENT!!!!
QFile copyFile(temp1File);
if (copyFile.exists())
copyFile.remove();
KSharedConfigPtr newFile = KSharedConfig::openConfig(temp1File);
KConfigGroup copied_conts = KConfigGroup(newFile, "Containments");
KConfigGroup copied_c1 = KConfigGroup(&copied_conts, QString::number(containment->id()));
containment->config().copyTo(&copied_c1);
//!investigate if there are subcontainments in the containment to copy also
//! subId, subAppletId
QHash<uint, QString> subInfo;
auto applets = containment->config().group("Applets");
for (const auto &applet : applets.groupList()) {
int tSubId = subContainmentId(applets.group(applet));
//! It is a subcontainment !!!
if (isValid(tSubId)) {
subInfo[tSubId] = applet;
qDebug() << "subcontainment with id "<< tSubId << " was found in the containment... ::: " << containment->id();
}
}
if (subInfo.count() > 0) {
for(const auto subId : subInfo.keys()) {
Plasma::Containment *subcontainment{nullptr};
for (const auto containment : layout->corona()->containments()) {
if (containment->id() == subId) {
subcontainment = containment;
break;
}
}
if (subcontainment) {
KConfigGroup copied_sub = KConfigGroup(&copied_conts, QString::number(subcontainment->id()));
subcontainment->config().copyTo(&copied_sub);
}
}
}
//! end of subcontainments specific code
//! update ids to unique ones
QString temp2File = newUniqueIdsLayoutFromFile(layout, temp1File);
//! Finally import the configuration
QList<Plasma::Containment *> importedDocks = importLayoutFile(layout, temp2File);
Plasma::Containment *newContainment{nullptr};
if (importedDocks.size() == 1) {
newContainment = importedDocks[0];
}
if (!newContainment || !newContainment->kPackage().isValid()) {
qWarning() << "the requested containment plugin can not be located or loaded";
return ViewDelayedCreationData();
}
auto config = newContainment->config();
//in multi-screen environment the copied dock is moved to alternative screens first
const auto screens = qGuiApp->screens();
auto dock = layout->viewForContainment(containment);
bool setOnExplicitScreen = false;
int dockScrId = IDNULL;
int copyScrId = IDNULL;
if (dock) {
dockScrId = dock->positioner()->currentScreenId();
qDebug() << "COPY DOCK SCREEN ::: " << dockScrId;
if (isValid(dockScrId) && screens.count() > 1) {
for (const auto scr : screens) {
copyScrId = layout->corona()->screenPool()->id(scr->name());
//the screen must exist and not be the same with the original dock
if (isValid(copyScrId) && copyScrId != dockScrId) {
QList<Plasma::Types::Location> fEdges = layout->freeEdges(copyScrId);
if (fEdges.contains((Plasma::Types::Location)containment->location())) {
///set this containment to an explicit screen
config.writeEntry("onPrimary", false);
config.writeEntry("lastScreen", copyScrId);
newContainment->setLocation(containment->location());
qDebug() << "COPY DOCK SCREEN NEW SCREEN ::: " << copyScrId;
setOnExplicitScreen = true;
break;
}
}
}
}
}
if (!setOnExplicitScreen) {
QList<Plasma::Types::Location> edges = layout->freeEdges(newContainment->screen());
if (edges.count() > 0) {
newContainment->setLocation(edges.at(0));
} else {
newContainment->setLocation(Plasma::Types::BottomEdge);
}
config.writeEntry("onPrimary", true);
config.writeEntry("lastScreen", dockScrId);
}
newContainment->config().sync();
ViewDelayedCreationData result;
if (setOnExplicitScreen && isValid(copyScrId)) {
qDebug() << "Copy Dock in explicit screen ::: " << copyScrId;
result.containment = newContainment;
result.forceOnPrimary = false;
result.explicitScreen = copyScrId;
result.reactToScreenChange = true;
} else {
qDebug() << "Copy Dock in current screen...";
result.containment = newContainment;
result.forceOnPrimary = false;
result.explicitScreen = dockScrId;
result.reactToScreenChange = false;
}
return result;
}
bool Storage::isBroken(const Layout::GenericLayout *layout, QStringList &errors) const
{
if (layout->file().isEmpty() || !QFile(layout->file()).exists()) {
return false;
}
QStringList ids;
QStringList conts;
QStringList applets;
KSharedConfigPtr lFile = KSharedConfig::openConfig(layout->file());
if (!layout->corona()) {
KConfigGroup containmentsEntries = KConfigGroup(lFile, "Containments");
ids << containmentsEntries.groupList();
conts << ids;
for (const auto &cId : containmentsEntries.groupList()) {
auto appletsEntries = containmentsEntries.group(cId).group("Applets");
QStringList validAppletIds;
bool updated{false};
for (const auto &appletId : appletsEntries.groupList()) {
KConfigGroup appletGroup = appletsEntries.group(appletId);
if (Layouts::Storage::appletGroupIsValid(appletGroup)) {
validAppletIds << appletId;
} else {
updated = true;
//! heal layout file by removing applet config records that are not used any more
qDebug() << "Layout: " << layout->name() << " removing deprecated applet : " << appletId;
appletsEntries.deleteGroup(appletId);
}
}
if (updated) {
appletsEntries.sync();
}
ids << validAppletIds;
applets << validAppletIds;
}
} else {
for (const auto containment : *layout->containments()) {
ids << QString::number(containment->id());
conts << QString::number(containment->id());
for (const auto applet : containment->applets()) {
ids << QString::number(applet->id());
applets << QString::number(applet->id());
}
}
}
#if QT_VERSION < QT_VERSION_CHECK(5, 14, 0)
QSet<QString> idsSet = QSet<QString>::fromList(ids);
#else
QSet<QString> idsSet(ids.begin(), ids.end());
#endif
/* a different way to count duplicates
QMap<QString, int> countOfStrings;
for (int i = 0; i < ids.count(); i++) {
countOfStrings[ids[i]]++;
}*/
if (idsSet.count() != ids.count()) {
qDebug() << " ---- ERROR - BROKEN LAYOUT :: " << layout->name() << " ----";
if (!layout->corona()) {
qDebug() << " --- storaged file : " << layout->file();
} else {
if (layout->corona()->layoutsManager()->memoryUsage() == MemoryUsage::MultipleLayouts) {
qDebug() << " --- in multiple layouts hidden file : " << Layouts::Importer::layoutUserFilePath(Layout::MULTIPLELAYOUTSHIDDENNAME);
} else {
qDebug() << " --- in active layout file : " << layout->file();
}
}
qDebug() << "Containments :: " << conts;
qDebug() << "Applets :: " << applets;
for (const QString &c : conts) {
if (applets.contains(c)) {
QString errorStr = i18n("Same applet and containment id found ::: ") + c;
qDebug() << "Error: " << errorStr;
errors << errorStr;
}
}
for (int i = 0; i < ids.count(); ++i) {
for (int j = i + 1; j < ids.count(); ++j) {
if (ids[i] == ids[j]) {
QString errorStr = i18n("Different applets with same id ::: ") + ids[i];
qDebug() << "Error: " << errorStr;
errors << errorStr;
}
}
}
qDebug() << " -- - -- - -- - -- - - -- - - - - -- - - - - ";
if (!layout->corona()) {
KConfigGroup containmentsEntries = KConfigGroup(lFile, "Containments");
for (const auto &cId : containmentsEntries.groupList()) {
auto appletsEntries = containmentsEntries.group(cId).group("Applets");
qDebug() << " CONTAINMENT : " << cId << " APPLETS : " << appletsEntries.groupList();
}
} else {
for (const auto containment : *layout->containments()) {
QStringList appletsIds;
for (const auto applet : containment->applets()) {
appletsIds << QString::number(applet->id());
}
qDebug() << " CONTAINMENT : " << containment->id() << " APPLETS : " << appletsIds.join(",");
}
}
return true;
}
return false;
}
//! AppletsData Information
Data::Applet Storage::metadata(const QString &pluginId)
{
Data::Applet data;
data.id = pluginId;
KPackage::Package pkg = KPackage::PackageLoader::self()->loadPackage(QStringLiteral("Plasma/Applet"));
pkg.setDefaultPackageRoot(QStringLiteral("plasma/plasmoids"));
pkg.setPath(pluginId);
if (pkg.isValid()) {
data.name = pkg.metadata().name();
data.description = pkg.metadata().description();
data.icon = pkg.metadata().iconName();
}
return data;
}
Data::AppletsTable Storage::plugins(const Layout::GenericLayout *layout, const int containmentid)
{
Data::AppletsTable knownapplets;
Data::AppletsTable unknownapplets;
if (!layout) {
return knownapplets;
}
//! empty means all containments are valid
QList<int> validcontainmentids;
if (isValid(containmentid)) {
validcontainmentids << containmentid;
//! searching for specific containment and subcontainments and ignore all other containments
for(auto containment : *layout->containments()) {
if (containment->id() != containmentid) {
//! ignore irrelevant containments
continue;
}
for (auto applet : containment->applets()) {
if (isSubContainment(layout, applet)) {
validcontainmentids << subContainmentId(applet->config());
}
}
}
}
//! cycle through valid contaiments in order to retrieve their metadata
for(auto containment : *layout->containments()) {
if (validcontainmentids.count()>0 && !validcontainmentids.contains(containment->id())) {
//! searching only for valid containments
continue;
}
for (auto applet : containment->applets()) {
if (!isSubContainment(layout, applet)) {
QString pluginId = applet->pluginMetaData().pluginId();
if (!knownapplets.containsId(pluginId) && !unknownapplets.containsId(pluginId)) {
Data::Applet appletdata = metadata(pluginId);
if (appletdata.isValid()) {
knownapplets.insertBasedOnName(appletdata);
} else {
unknownapplets.insertBasedOnId(appletdata);
}
}
}
}
}
knownapplets << unknownapplets;
return knownapplets;
}
Data::AppletsTable Storage::plugins(const QString &layoutfile, const int containmentid)
{
Data::AppletsTable knownapplets;
Data::AppletsTable unknownapplets;
if (layoutfile.isEmpty()) {
return knownapplets;
}
KSharedConfigPtr lFile = KSharedConfig::openConfig(layoutfile);
KConfigGroup containmentGroups = KConfigGroup(lFile, "Containments");
//! empty means all containments are valid
QList<int> validcontainmentids;
if (isValid(containmentid)) {
validcontainmentids << containmentid;
//! searching for specific containment and subcontainments and ignore all other containments
for (const auto &cId : containmentGroups.groupList()) {
if (cId.toInt() != containmentid) {
//! ignore irrelevant containments
continue;
}
auto appletGroups = containmentGroups.group(cId).group("Applets");
for (const auto &appletId : appletGroups.groupList()) {
KConfigGroup appletCfg = appletGroups.group(appletId);
if (isSubContainment(appletCfg)) {
validcontainmentids << subContainmentId(appletCfg);
}
}
}
}
//! cycle through valid contaiments in order to retrieve their metadata
for (const auto &cId : containmentGroups.groupList()) {
if (validcontainmentids.count()>0 && !validcontainmentids.contains(cId.toInt())) {
//! searching only for valid containments
continue;
}
auto appletGroups = containmentGroups.group(cId).group("Applets");
for (const auto &appletId : appletGroups.groupList()) {
KConfigGroup appletCfg = appletGroups.group(appletId);
if (!isSubContainment(appletCfg)) {
QString pluginId = appletCfg.readEntry("plugin", "");
if (!knownapplets.containsId(pluginId) && !unknownapplets.containsId(pluginId)) {
Data::Applet appletdata = metadata(pluginId);
if (appletdata.isValid()) {
knownapplets.insertBasedOnName(appletdata);
} else {
unknownapplets.insertBasedOnId(appletdata);
}
}
}
}
}
knownapplets << unknownapplets;
return knownapplets;
}
//! Data For Reports
void Storage::subContainmentsInfo(const QString &file, QHash<int, QList<int>> &subContainments, QList<int> &assignedSubContainments, QList<int> &orphanSubContainments)
{
subContainments.clear();
assignedSubContainments.clear();
orphanSubContainments.clear();
KSharedConfigPtr lFile = KSharedConfig::openConfig(file);
KConfigGroup containmentGroups = KConfigGroup(lFile, "Containments");
//! assigned subcontainments
for (const auto &cId : containmentGroups.groupList()) {
if (Layouts::Storage::self()->isLatteContainment(containmentGroups.group(cId))) {
auto applets = containmentGroups.group(cId).group("Applets");
for (const auto &applet : applets.groupList()) {
KConfigGroup appletSettings = applets.group(applet).group("Configuration");
int tSubId = appletSettings.readEntry("SystrayContainmentId", IDNULL);
if (isValid(tSubId)) {
assignedSubContainments << tSubId;
subContainments[cId.toInt()].append(tSubId);
}
}
}
}
//! orphan subcontainments
for (const auto &cId : containmentGroups.groupList()) {
if (!Layouts::Storage::self()->isLatteContainment(containmentGroups.group(cId)) && !assignedSubContainments.contains(cId.toInt())) {
orphanSubContainments << cId.toInt();
}
}
}
QList<Layout::ViewData> Storage::viewsData(const QString &file, const QHash<int, QList<int>> &subContainments)
{
QList<Layout::ViewData> viewsData;
KSharedConfigPtr lFile = KSharedConfig::openConfig(file);
KConfigGroup containmentGroups = KConfigGroup(lFile, "Containments");
for (const auto &cId : containmentGroups.groupList()) {
if (Layouts::Storage::self()->isLatteContainment(containmentGroups.group(cId))) {
Layout::ViewData vData;
int id = cId.toInt();
//! id
vData.id = id;
//! active
vData.active = false;
//! onPrimary
vData.onPrimary = containmentGroups.group(cId).readEntry("onPrimary", true);
//! Screen
vData.screenId = containmentGroups.group(cId).readEntry("lastScreen", IDNULL);
//! location
vData.location = containmentGroups.group(cId).readEntry("location", (int)Plasma::Types::BottomEdge);
//! subcontainments
vData.subContainments = subContainments[id];
viewsData << vData;
}
}
return viewsData;
}
QList<int> Storage::viewsScreens(const QString &file)
{
QList<int> screens;
KSharedConfigPtr lFile = KSharedConfig::openConfig(file);
KConfigGroup containmentGroups = KConfigGroup(lFile, "Containments");
for (const auto &cId : containmentGroups.groupList()) {
if (Layouts::Storage::self()->isLatteContainment(containmentGroups.group(cId))) {
int screenId = containmentGroups.group(cId).readEntry("lastScreen", IDNULL);
if (isValid(screenId) && !screens.contains(screenId)) {
screens << screenId;
}
}
}
return screens;
}
}
}