1
0
mirror of https://github.com/KDE/latte-dock.git synced 2025-01-05 01:18:02 +03:00
latte-dock/app/layouts/storage.cpp

1872 lines
63 KiB
C++
Raw Normal View History

/*
2021-05-27 18:01:00 +03:00
SPDX-FileCopyrightText: 2020 Michail Vourlakos <mvourlakos@gmail.com>
SPDX-License-Identifier: GPL-2.0-or-later
*/
#include "storage.h"
// local
#include <coretypes.h>
#include "importer.h"
#include "manager.h"
#include "../lattecorona.h"
#include "../screenpool.h"
2021-04-29 17:58:45 +03:00
#include "../data/errordata.h"
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
#include "../data/viewdata.h"
#include "../layout/abstractlayout.h"
#include "../view/view.h"
// Qt
#include <QDebug>
#include <QDir>
#include <QFile>
#include <QFileInfo>
#include <QLatin1String>
// 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();
//! Known Errors / Warnings
2021-04-30 08:19:49 +03:00
s_knownErrors << Data::Generic(Data::Error::APPLETSWITHSAMEID, i18n("Different Applets With Same Id"));
s_knownErrors << Data::Generic(Data::Error::ORPHANEDPARENTAPPLETOFSUBCONTAINMENT, i18n("Orphaned Parent Applet Of Subcontainment"));
s_knownErrors<< Data::Generic(Data::Warning::APPLETANDCONTAINMENTWITHSAMEID, i18n("Different Applet And Containment With Same Id"));
s_knownErrors << Data::Generic(Data::Warning::ORPHANEDSUBCONTAINMENT, i18n("Orphaned Subcontainment"));
2021-04-29 17:58:45 +03:00
2021-04-30 08:19:49 +03:00
//! Known SubContainment Families
SubContaimentIdentityData data;
//! Systray Family
m_subIdentities << SubContaimentIdentityData{.cfgGroup="Configuration", .cfgProperty="SystrayContainmentId"};
2021-04-30 08:19:49 +03:00
//! Group applet Family
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(const Plasma::Containment *containment) const
{
if (!containment) {
return false;
}
if (containment->pluginMetaData().pluginId() == QLatin1String("org.kde.latte.containment")) {
return true;
}
return false;
}
bool Storage::isLatteContainment(const KConfigGroup &group) const
{
QString pluginId = group.readEntry("plugin", "");
return pluginId == QLatin1String("org.kde.latte.containment");
}
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
bool Storage::isSubContainment(const Plasma::Corona *corona, const Plasma::Applet *applet) const
{
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
if (!corona || !applet) {
return false;
}
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
for (const auto containment : corona->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;
}
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
Plasma::Containment *Storage::subContainmentOf(const Plasma::Corona *corona, const Plasma::Applet *applet)
{
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
if (!corona || !applet) {
return nullptr;
}
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
if (isSubContainment(corona, applet)) {
for (const auto containment : corona->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);
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
removeAllClonedViews(layout->file());
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();
}
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
if (copyFile.exists()) {
copyFile.remove();
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
}
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);
newFile->reparseConfiguration();
//! update ids to unique ones
QString temp2File = newUniqueIdsFile(temp1FilePath, layout);
//! 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) == QLatin1String("Configuration")
&& appletGroup.group("Configuration").keyList().count() == 1
&& appletGroup.group("Configuration").hasKey("PreloadWeight") );
}
QStringList Storage::containmentsIds(const QString &filepath)
{
QStringList ids;
KSharedConfigPtr filePtr = KSharedConfig::openConfig(filepath);
KConfigGroup containments = KConfigGroup(filePtr, "Containments");
for(const auto &cId : containments.groupList()) {
ids << cId;
}
return ids;
}
QStringList Storage::appletsIds(const QString &filepath)
{
QStringList ids;
KSharedConfigPtr filePtr = KSharedConfig::openConfig(filepath);
KConfigGroup containments = KConfigGroup(filePtr, "Containments");
for(const auto &cId : containments.groupList()) {
for(const auto &aId : containments.group(cId).group("Applets").groupList()) {
ids << aId;
}
}
return ids;
}
QString Storage::newUniqueIdsFile(QString originFile, const Layout::GenericLayout *destinationLayout)
{
if (!destinationLayout) {
return QString();
}
QString currentdestinationname = destinationLayout->name();
QString currentdestinationfile = "";
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
if (!destinationLayout->hasCorona()) {
currentdestinationfile = destinationLayout->file();
}
QString tempFile = m_storageTmpDir.path() + "/" + currentdestinationname + ".views.newids";
QFile copyFile(tempFile);
if (copyFile.exists()) {
copyFile.remove();
}
//! BEGIN updating the ids in the temp file
QStringList allIds;
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
if (destinationLayout->hasCorona()) {
allIds << destinationLayout->corona()->containmentsIds();
allIds << destinationLayout->corona()->appletsIds();
} else {
allIds << containmentsIds(currentdestinationfile);
allIds << appletsIds(currentdestinationfile);
}
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;
2021-04-17 12:30:23 +03:00
KSharedConfigPtr filePtr = KSharedConfig::openConfig(originFile);
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;
if (contId.toInt()>=12 && !allIds.contains(contId) && !assignedIds.contains(contId)) {
newId = contId;
} else {
newId = availableId(allIds, assignedIds, 12);
}
assignedIds << newId;
assigned[contId] = newId;
}
for (const auto &appId : toInvestigateAppletIds) {
QString newId;
if (appId.toInt()>=40 && !allIds.contains(appId) && !assignedIds.contains(appId)) {
newId = appId;
} else {
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);
}
}
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
if (destinationLayout->hasCorona() && destinationLayout->corona()->layoutsManager()->memoryUsage() == MemoryUsage::MultipleLayouts) {
//! will be added in main corona multiple layouts file
2021-04-17 12:30:23 +03:00
investigate_conts.group(cId).writeEntry("layoutId", destinationLayout->name());
} else {
//! will be added in inactive layout
investigate_conts.group(cId).writeEntry("layoutId", QString());
}
}
//! 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);
}
}
}
file2Ptr->reparseConfiguration();
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();
}
filePtr->reparseConfiguration();
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
removeAllClonedViews(layout->file());
}
void Storage::moveToLayoutFile(const QString &layoutName)
{
if (layoutName.isEmpty()) {
return;
}
QString linkedFilePath = Importer::layoutUserFilePath(Layout::MULTIPLELAYOUTSHIDDENNAME);
QString layoutFilePath = Importer::layoutUserFilePath(layoutName);
if (linkedFilePath.isEmpty() || layoutFilePath.isEmpty() || !QFileInfo(linkedFilePath).exists() || !QFileInfo(layoutFilePath).exists()) {
return;
}
KSharedConfigPtr layoutFilePtr = KSharedConfig::openConfig(layoutFilePath);
KConfigGroup singleContainments = KConfigGroup(layoutFilePtr, "Containments");
singleContainments.deleteGroup();
KSharedConfigPtr multiFilePtr = KSharedConfig::openConfig(linkedFilePath);
KConfigGroup multiContainments = KConfigGroup(multiFilePtr, "Containments");
for(const auto &cId : multiContainments.groupList()) {
QString cname = multiContainments.group(cId).readEntry("layoutId", QString());
if (!cname.isEmpty() && cname == layoutName) {
multiContainments.group(cId).writeEntry("layoutId", "");
KConfigGroup singleGroup = singleContainments.group(cId);
multiContainments.group(cId).copyTo(&singleGroup);
singleGroup.writeEntry("layoutId", "");
singleGroup.sync();
multiContainments.group(cId).deleteGroup();
}
}
layoutFilePtr->reparseConfiguration();
removeAllClonedViews(layoutFilePath);
}
QList<Plasma::Containment *> Storage::importLayoutFile(const Layout::GenericLayout *layout, QString file)
{
KSharedConfigPtr filePtr = KSharedConfig::openConfig(file);
auto newContainments = layout->corona()->importLayout(KConfigGroup(filePtr, ""));
QList<Plasma::Containment *> importedViews;
for (const auto containment : newContainments) {
if (isLatteContainment(containment)) {
importedViews << containment;
}
}
return importedViews;
}
void Storage::importContainments(const QString &originFile, const QString &destinationFile)
{
if (originFile.isEmpty() || destinationFile.isEmpty()) {
return;
}
KSharedConfigPtr originPtr = KSharedConfig::openConfig(originFile);
KSharedConfigPtr destinationPtr = KSharedConfig::openConfig(destinationFile);
KConfigGroup originContainments = KConfigGroup(originPtr, "Containments");
KConfigGroup destinationContainments = KConfigGroup(destinationPtr, "Containments");
for (const auto originContId : originContainments.groupList()) {
KConfigGroup destinationContainment(&destinationContainments, originContId);
originContainments.group(originContId).copyTo(&destinationContainment);
}
destinationContainments.sync();
}
Data::View Storage::newView(const Layout::GenericLayout *destinationLayout, const Data::View &nextViewData)
{
if (!destinationLayout || nextViewData.originFile().isEmpty()) {
return Data::View();
}
qDebug() << "new view for layout";
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
if (destinationLayout->hasCorona()) {
//! Setting mutable for create a containment
destinationLayout->corona()->setImmutability(Plasma::Types::Mutable);
}
QString templateFile = nextViewData.originFile();
//! copy view template path in temp file
QString templateTmpAbsolutePath = m_storageTmpDir.path() + "/" + QFileInfo(templateFile).fileName() + ".newids";
if (QFile(templateTmpAbsolutePath).exists()) {
QFile(templateTmpAbsolutePath).remove();
}
QFile(templateFile).copy(templateTmpAbsolutePath);
//! update ids to unique ones
QString temp2File = newUniqueIdsFile(templateTmpAbsolutePath, destinationLayout);
//! update view containment data in case next data are provided
if (nextViewData.state() != Data::View::IsInvalid) {
KSharedConfigPtr lFile = KSharedConfig::openConfig(temp2File);
KConfigGroup containments = KConfigGroup(lFile, "Containments");
for (const auto cId : containments.groupList()) {
if (Layouts::Storage::self()->isLatteContainment(containments.group(cId))) {
//! first view we will find, we update its value
updateView(containments.group(cId), nextViewData);
break;
}
}
lFile->reparseConfiguration();
}
Data::ViewsTable updatedNextViews = views(temp2File);
if (updatedNextViews.rowCount() <= 0) {
return Data::View();
}
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
if (destinationLayout->hasCorona()) {
//! import views for active layout
QList<Plasma::Containment *> importedViews = importLayoutFile(destinationLayout, temp2File);
Plasma::Containment *newContainment = (importedViews.size() == 1 ? importedViews[0] : nullptr);
if (!newContainment || !newContainment->kPackage().isValid()) {
qWarning() << "the requested containment plugin can not be located or loaded from:" << templateFile;
return Data::View();
}
} else {
//! import views for inactive layout
importContainments(temp2File, destinationLayout->file());
}
return updatedNextViews[0];
}
void Storage::clearExportedLayoutSettings(KConfigGroup &layoutSettingsGroup)
{
layoutSettingsGroup.writeEntry("preferredForShortcutsTouched", false);
layoutSettingsGroup.writeEntry("lastUsedActivity", QString());
layoutSettingsGroup.writeEntry("activities", QStringList());
layoutSettingsGroup.sync();
}
2021-02-13 02:22:44 +03:00
bool Storage::exportTemplate(const QString &originFile, const QString &destinationFile,const Data::AppletsTable &approvedApplets)
{
if (originFile.isEmpty() || !QFile(originFile).exists() || destinationFile.isEmpty()) {
return false;
}
if (QFile(destinationFile).exists()) {
QFile::remove(destinationFile);
}
QFile(originFile).copy(destinationFile);
KSharedConfigPtr destFilePtr = KSharedConfig::openConfig(destinationFile);
destFilePtr->reparseConfiguration();
2021-02-13 02:22:44 +03:00
KConfigGroup containments = KConfigGroup(destFilePtr, "Containments");
QStringList rejectedSubContainments;
//! clear applets that are not approved
2021-02-13 02:22:44 +03:00
for (const auto &cId : containments.groupList()) {
//! clear properties
containments.group(cId).writeEntry("layoutId", QString());
if (isLatteContainment(containments.group(cId))) {
containments.group(cId).writeEntry("isPreferredForShortcuts", false);
}
//! clear applets
2021-02-13 02:22:44 +03:00
auto applets = containments.group(cId).group("Applets");
for (const auto &aId: applets.groupList()) {
QString pluginId = applets.group(aId).readEntry("plugin", "");
if (!approvedApplets.containsId(pluginId)) {
if (!isSubContainment(applets.group(aId))) {
//!remove all configuration for that applet
for (const auto &configId: applets.group(aId).groupList()) {
applets.group(aId).group(configId).deleteGroup();
}
} else {
//! register which subcontaiments should return to default properties
rejectedSubContainments << QString::number(subContainmentId(applets.group(aId)));
2021-02-13 02:22:44 +03:00
}
}
}
}
//! clear rejected SubContainments
for (const auto &cId : containments.groupList()) {
if (rejectedSubContainments.contains(cId)) {
containments.group(cId).group("General").deleteGroup();
}
};
KConfigGroup layoutSettingsGrp(destFilePtr, "LayoutSettings");
clearExportedLayoutSettings(layoutSettingsGrp);
destFilePtr->reparseConfiguration();
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
removeAllClonedViews(destinationFile);
2021-02-13 02:22:44 +03:00
return true;
}
bool Storage::exportTemplate(const Layout::GenericLayout *layout, Plasma::Containment *containment, const QString &destinationFile, const Data::AppletsTable &approvedApplets)
{
if (!layout || !containment || destinationFile.isEmpty()) {
return false;
}
if (QFile(destinationFile).exists()) {
QFile::remove(destinationFile);
}
KSharedConfigPtr destFilePtr = KSharedConfig::openConfig(destinationFile);
destFilePtr->reparseConfiguration();
KConfigGroup copied_conts = KConfigGroup(destFilePtr, "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
QStringList rejectedSubContainments;
//! clear applets that are not approved
for (const auto &cId : copied_conts.groupList()) {
//! clear properties
copied_conts.group(cId).writeEntry("layoutId", QString());
if (isLatteContainment(copied_conts.group(cId))) {
copied_conts.group(cId).writeEntry("isPreferredForShortcuts", false);
}
//! clear applets
auto applets = copied_conts.group(cId).group("Applets");
for (const auto &aId: applets.groupList()) {
QString pluginId = applets.group(aId).readEntry("plugin", "");
if (!approvedApplets.containsId(pluginId)) {
if (!isSubContainment(applets.group(aId))) {
//!remove all configuration for that applet
for (const auto &configId: applets.group(aId).groupList()) {
applets.group(aId).group(configId).deleteGroup();
}
} else {
//! register which subcontaiments should return to default properties
rejectedSubContainments << QString::number(subContainmentId(applets.group(aId)));
}
}
}
}
//! clear rejected SubContainments
for (const auto &cId : copied_conts.groupList()) {
if (rejectedSubContainments.contains(cId)) {
copied_conts.group(cId).group("General").deleteGroup();
}
};
KConfigGroup layoutSettingsGrp(destFilePtr, "LayoutSettings");
clearExportedLayoutSettings(layoutSettingsGrp);
destFilePtr->reparseConfiguration();
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
removeAllClonedViews(destinationFile);
return true;
}
bool Storage::hasDifferentAppletsWithSameId(const Layout::GenericLayout *layout, Data::Error &error)
{
if (!layout || layout->file().isEmpty() || !QFile(layout->file()).exists()) {
return false;
}
2021-04-30 08:19:49 +03:00
error.id = s_knownErrors[Data::Error::APPLETSWITHSAMEID].id;
error.name = s_knownErrors[Data::Error::APPLETSWITHSAMEID].name;
if (layout->isActive()) { // active layout
QStringList registeredapplets;
QStringList conflictedapplets;
//! split ids to normal registered and conflicted
for (const auto containment : *layout->containments()) {
QString cid = QString::number(containment->id());
for (const auto applet : containment->applets()) {
QString aid = QString::number(applet->id());
if (!registeredapplets.contains(aid)) {
registeredapplets << aid;
} else if (!conflictedapplets.contains(aid)) {
conflictedapplets << aid;
}
}
}
//! create error data
for (const auto containment : *layout->containments()) {
QString cid = QString::number(containment->id());
for (const auto applet : containment->applets()) {
QString aid = QString::number(applet->id());
if (!conflictedapplets.contains(aid)) {
continue;
}
Data::ErrorInformation errorinfo;
2021-04-30 01:38:02 +03:00
errorinfo.id = QString::number(error.information.rowCount());
errorinfo.containment = metadata(containment->pluginMetaData().pluginId());
errorinfo.containment.storageId = cid;
errorinfo.applet = metadata(applet->pluginMetaData().pluginId());
errorinfo.applet.storageId = aid;
error.information << errorinfo;
}
}
} else { // inactive layout
KSharedConfigPtr lfile = KSharedConfig::openConfig(layout->file());
KConfigGroup containmentsEntries = KConfigGroup(lfile, "Containments");
QStringList registeredapplets;
QStringList conflictedapplets;
//! split ids to normal registered and conflicted
for (const auto &cid : containmentsEntries.groupList()) {
for (const auto &aid : containmentsEntries.group(cid).group("Applets").groupList()) {
if (!registeredapplets.contains(aid)) {
registeredapplets << aid;
} else if (!conflictedapplets.contains(aid)) {
conflictedapplets << aid;
}
}
}
//! create error data
for (const auto &cid : containmentsEntries.groupList()) {
for (const auto &aid : containmentsEntries.group(cid).group("Applets").groupList()) {
if (!conflictedapplets.contains(aid)) {
continue;
}
Data::ErrorInformation errorinfo;
2021-04-30 01:38:02 +03:00
errorinfo.id = QString::number(error.information.rowCount());
errorinfo.containment = metadata(containmentsEntries.group(cid).readEntry("plugin", ""));
errorinfo.containment.storageId = cid;
errorinfo.applet = metadata(containmentsEntries.group(cid).group("Applets").group(aid).readEntry("plugin", ""));
errorinfo.applet.storageId = aid;
error.information << errorinfo;
}
}
}
return !error.information.isEmpty();
}
bool Storage::hasAppletsAndContainmentsWithSameId(const Layout::GenericLayout *layout, Data::Warning &warning)
{
if (!layout || layout->file().isEmpty() || !QFile(layout->file()).exists()) {
return false;
}
warning.id = s_knownErrors[Data::Error::APPLETANDCONTAINMENTWITHSAMEID].id;
warning.name = s_knownErrors[Data::Error::APPLETANDCONTAINMENTWITHSAMEID].name;
if (layout->isActive()) { // active layout
QStringList registeredcontainments;
QStringList conflicted;
//! discover normal containment ids
for (const auto containment : *layout->containments()) {
QString cid = QString::number(containment->id());
if (registeredcontainments.contains(cid)) {
continue;
}
registeredcontainments << cid;
}
//! discover conflicted ids between containments and applets
for (const auto containment : *layout->containments()) {
QString cid = QString::number(containment->id());
for (const auto applet : containment->applets()) {
QString aid = QString::number(applet->id());
if (!registeredcontainments.contains(aid)) {
continue;
} else if (!conflicted.contains(aid)) {
conflicted << aid;
}
}
}
//! create warning data
for (const auto containment : *layout->containments()) {
QString cid = QString::number(containment->id());
if (conflicted.contains(cid)) {
Data::WarningInformation warninginfo;
2021-04-30 01:38:02 +03:00
warninginfo.id = QString::number(warning.information.rowCount());
warninginfo.containment = metadata(containment->pluginMetaData().pluginId());
warninginfo.containment.storageId = cid;
warning.information << warninginfo;
}
for (const auto applet : containment->applets()) {
QString aid = QString::number(applet->id());
if (!conflicted.contains(aid)) {
continue;
}
Data::WarningInformation warninginfo;
2021-04-30 01:38:02 +03:00
warninginfo.id = QString::number(warning.information.rowCount());
warninginfo.containment = metadata(containment->pluginMetaData().pluginId());
warninginfo.containment.storageId = cid;
warninginfo.applet = metadata(applet->pluginMetaData().pluginId());
warninginfo.applet.storageId = aid;
warning.information << warninginfo;
}
}
} else { // inactive layout
KSharedConfigPtr lfile = KSharedConfig::openConfig(layout->file());
KConfigGroup containmentsEntries = KConfigGroup(lfile, "Containments");
QStringList registeredcontainments;
QStringList conflicted;
//! discover normal containment ids
for (const auto &cid : containmentsEntries.groupList()) {
if (registeredcontainments.contains(cid)) {
continue;
}
registeredcontainments << cid;
}
//! discover conflicted ids between containments and applets
for (const auto &cid : containmentsEntries.groupList()) {
for (const auto &aid : containmentsEntries.group(cid).group("Applets").groupList()) {
if (!registeredcontainments.contains(aid)) {
continue;
} else if (!conflicted.contains(aid)) {
conflicted << aid;
}
}
}
//! create warning data
for (const auto &cid : containmentsEntries.groupList()) {
if (conflicted.contains(cid)) {
Data::WarningInformation warninginfo;
2021-04-30 01:38:02 +03:00
warninginfo.id = QString::number(warning.information.rowCount());
warninginfo.containment = metadata(containmentsEntries.group(cid).readEntry("plugin", ""));
warninginfo.containment.storageId = cid;
warning.information << warninginfo;
}
for (const auto &aid : containmentsEntries.group(cid).group("Applets").groupList()) {
if (!conflicted.contains(aid)) {
continue;
}
Data::WarningInformation warninginfo;
2021-04-30 01:38:02 +03:00
warninginfo.id = QString::number(warning.information.rowCount());
warninginfo.containment = metadata(containmentsEntries.group(cid).readEntry("plugin", ""));
warninginfo.containment.storageId = cid;
warninginfo.applet = metadata(containmentsEntries.group(cid).group("Applets").group(aid).readEntry("plugin", ""));
warninginfo.applet.storageId = aid;
warning.information << warninginfo;
}
}
}
return !warning.information.isEmpty();
}
bool Storage::hasOrphanedParentAppletOfSubContainment(const Layout::GenericLayout *layout, Data::Error &error)
{
if (!layout || layout->file().isEmpty() || !QFile(layout->file()).exists()) {
return false;
}
error.id = s_knownErrors[Data::Error::ORPHANEDPARENTAPPLETOFSUBCONTAINMENT].id;
error.name = s_knownErrors[Data::Error::ORPHANEDPARENTAPPLETOFSUBCONTAINMENT].name;
Data::ViewsTable views = Layouts::Storage::self()->views(layout);
if (layout->isActive()) { // active layout
//! create error data
for (const auto containment : *layout->containments()) {
QString cid = QString::number(containment->id());
for (const auto applet : containment->applets()) {
QString aid = QString::number(applet->id());
int subid = subContainmentId(applet->config());
if (subid == IDNULL || hasContainment(layout, subid)) {
continue;
}
Data::ErrorInformation errorinfo;
errorinfo.id = QString::number(error.information.rowCount());
errorinfo.containment = metadata(containment->pluginMetaData().pluginId());
errorinfo.containment.storageId = cid;
errorinfo.applet = metadata(applet->pluginMetaData().pluginId());
errorinfo.applet.storageId = aid;
errorinfo.applet.subcontainmentId = subid;
error.information << errorinfo;
}
}
} else {
KSharedConfigPtr lfile = KSharedConfig::openConfig(layout->file());
KConfigGroup containmentsEntries = KConfigGroup(lfile, "Containments");
//! create error data
for (const auto &cid : containmentsEntries.groupList()) {
for (const auto &aid : containmentsEntries.group(cid).group("Applets").groupList()) {
int subid = subContainmentId(containmentsEntries.group(cid).group("Applets").group(aid));
if (subid == IDNULL || hasContainment(layout, subid)) {
continue;
}
Data::ErrorInformation errorinfo;
errorinfo.id = QString::number(error.information.rowCount());
errorinfo.containment = metadata(containmentsEntries.group(cid).readEntry("plugin", ""));
errorinfo.containment.storageId = cid;
errorinfo.applet = metadata(containmentsEntries.group(cid).group("Applets").group(aid).readEntry("plugin", ""));
errorinfo.applet.storageId = aid;
errorinfo.applet.subcontainmentId = subid;
error.information << errorinfo;
}
}
}
Data::Warning warning1;
if (!error.information.isEmpty() && hasOrphanedSubContainments(layout, warning1)) {
error.information << warning1.information;
}
return !error.information.isEmpty();
}
bool Storage::hasOrphanedSubContainments(const Layout::GenericLayout *layout, Data::Warning &warning)
{
if (!layout || layout->file().isEmpty() || !QFile(layout->file()).exists()) {
return false;
}
warning.id = s_knownErrors[Data::Error::ORPHANEDSUBCONTAINMENT].id;
warning.name = s_knownErrors[Data::Error::ORPHANEDSUBCONTAINMENT].name;
Data::ViewsTable views = Layouts::Storage::self()->views(layout);
if (layout->isActive()) { // active layout
//! create warning data
for (const auto containment : *layout->containments()) {
QString cid = QString::number(containment->id());
Plasma::Applet *parentApplet = qobject_cast<Plasma::Applet *>(containment->parent());
Plasma::Containment *parentContainment = parentApplet ? qobject_cast<Plasma::Containment *>(parentApplet->parent()) : nullptr;
if (isLatteContainment(containment) || (parentApplet && parentContainment && layout->contains(parentContainment))) {
//! is latte containment or is subcontainment that belongs to latte containment
continue;
}
Data::WarningInformation warninginfo;
2021-04-30 01:38:02 +03:00
warninginfo.id = QString::number(warning.information.rowCount());
warninginfo.containment = metadata(containment->pluginMetaData().pluginId());
warninginfo.containment.storageId = cid;
warning.information << warninginfo;
}
} else { // inactive layout
KSharedConfigPtr lfile = KSharedConfig::openConfig(layout->file());
KConfigGroup containmentsEntries = KConfigGroup(lfile, "Containments");
//! create warning data
for (const auto &cid : containmentsEntries.groupList()) {
if (views.hasContainmentId(cid)) {
continue;
}
Data::WarningInformation warninginfo;
2021-04-30 01:38:02 +03:00
warninginfo.id = QString::number(warning.information.rowCount());
warninginfo.containment = metadata(containmentsEntries.group(cid).readEntry("plugin", ""));
warninginfo.containment.storageId = cid;
warning.information << warninginfo;
}
}
return !warning.information.isEmpty();
}
Data::ErrorsList Storage::errors(const Layout::GenericLayout *layout)
{
Data::ErrorsList errs;
if (!layout || layout->file().isEmpty() || !QFile(layout->file()).exists()) {
return errs;
}
Data::Error error1;
if (hasDifferentAppletsWithSameId(layout, error1)) {
errs << error1;
}
Data::Error error2;
if (hasOrphanedParentAppletOfSubContainment(layout, error2)) {
errs << error2;
}
return errs;
}
Data::WarningsList Storage::warnings(const Layout::GenericLayout *layout)
{
Data::WarningsList warns;
if (!layout || layout->file().isEmpty() || !QFile(layout->file()).exists()) {
return warns;
}
Data::Warning warning1;
if (hasAppletsAndContainmentsWithSameId(layout, warning1)) {
warns << warning1;
}
Data::Error error1;
Data::Warning warning2;
if (!hasOrphanedParentAppletOfSubContainment(layout, error1) /*this is needed because this error has higher priority*/
&& hasOrphanedSubContainments(layout, warning2)) {
warns << warning2;
}
return warns;
}
//! 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();
2021-02-13 03:20:31 +03:00
QString iconName = pkg.metadata().iconName();
if (!iconName.startsWith("/") && iconName.contains("/")) {
data.icon = QFileInfo(pkg.metadata().fileName()).absolutePath() + "/" + iconName;
} else {
data.icon = iconName;
}
}
if (data.name.isEmpty()) {
//! this is also a way to identify if a package is installed or not in current system
data.name = data.id;
}
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 (((int)containment->id()) != containmentid) {
//! ignore irrelevant containments
continue;
}
for (auto applet : containment->applets()) {
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
if (isSubContainment(layout->corona(), 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()) {
QString pluginId = applet->pluginMetaData().pluginId();
if (!knownapplets.containsId(pluginId) && !unknownapplets.containsId(pluginId)) {
Data::Applet appletdata = metadata(pluginId);
if (appletdata.isInstalled()) {
knownapplets.insertBasedOnName(appletdata);
} else if (appletdata.isValid()) {
unknownapplets.insertBasedOnName(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);
QString pluginId = appletCfg.readEntry("plugin", "");
if (!knownapplets.containsId(pluginId) && !unknownapplets.containsId(pluginId)) {
Data::Applet appletdata = metadata(pluginId);
if (appletdata.isInstalled()) {
knownapplets.insertBasedOnName(appletdata);
} else if (appletdata.isValid()) {
unknownapplets.insertBasedOnName(appletdata);
}
}
}
}
knownapplets << unknownapplets;
return knownapplets;
}
//! Views Data
void Storage::syncContainmentConfig(Plasma::Containment *containment)
{
if (!containment) {
return;
}
for(auto applet: containment->applets()) {
KConfigGroup appletGeneralConfig = applet->config().group("General");
if (appletGeneralConfig.exists()) {
appletGeneralConfig.sync();
}
applet->config().sync();
}
containment->config().sync();
}
bool Storage::containsView(const QString &filepath, const int &viewId)
{
KSharedConfigPtr lFile = KSharedConfig::openConfig(filepath);
KConfigGroup containmentGroups = KConfigGroup(lFile, "Containments");
KConfigGroup viewGroup = containmentGroups.group(QString::number(viewId));
return viewGroup.exists() && isLatteContainment(viewGroup);
}
bool Storage::hasContainment(const Layout::GenericLayout *layout, const int &id)
{
if (!layout || layout->file().isEmpty() || !QFile(layout->file()).exists()) {
return false;
}
if (layout->isActive()) { // active layout
for(const auto containment : *layout->containments()) {
if ((int)containment->id() == id) {
return true;
}
}
} else { // inactive layout
KSharedConfigPtr lfile = KSharedConfig::openConfig(layout->file());
KConfigGroup containmentsEntries = KConfigGroup(lfile, "Containments");
//! create warning data
for (const auto &cid : containmentsEntries.groupList()) {
if (cid.toInt() == id) {
return true;
}
}
}
return false;
}
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
bool Storage::isClonedView(const Plasma::Containment *containment) const
{
if (!containment) {
return false;
}
return isClonedView(containment->config());
}
bool Storage::isClonedView(const KConfigGroup &containmentGroup) const
{
if (!isLatteContainment(containmentGroup)) {
return false;
}
int isClonedFrom = containmentGroup.readEntry("isClonedFrom", Data::View::ISCLONEDNULL);
return (isClonedFrom != IDNULL);
}
void Storage::removeAllClonedViews(const QString &filepath)
{
KSharedConfigPtr lFile = KSharedConfig::openConfig(filepath);
KConfigGroup containmentGroups = KConfigGroup(lFile, "Containments");
QList<Data::View> clones;
for (const auto &contId : containmentGroups.groupList()) {
if (isClonedView(containmentGroups.group(contId))) {
clones << view(containmentGroups.group(contId));
}
}
if (clones.size() <= 0) {
return;
}
if (clones.count()>0) {
qDebug() << "org.kde.layout :: Removing clones from file: " << filepath;
}
for (const auto &clonedata : clones) {
qDebug() << "org.kde.layout :: Removing clone:" << clonedata.id << " and its subcontainments:" << clonedata.subcontainments;
removeView(filepath, clonedata);
}
}
Data::GenericTable<Data::Generic> Storage::subcontainments(const Layout::GenericLayout *layout, const Plasma::Containment *lattecontainment) const
{
Data::GenericTable<Data::Generic> subs;
if (!layout || !Layouts::Storage::self()->isLatteContainment(lattecontainment)) {
return subs;
}
for (const auto containment : (*layout->containments())) {
if (containment == lattecontainment) {
continue;
}
Plasma::Applet *parentApplet = qobject_cast<Plasma::Applet *>(containment->parent());
//! add subcontainments for that lattecontainment
if (parentApplet && parentApplet->containment() && parentApplet->containment() == lattecontainment) {
Data::Generic subdata;
subdata.id = QString::number(containment->id());
subs << subdata;
}
}
return subs;
}
Data::GenericTable<Data::Generic> Storage::subcontainments(const KConfigGroup &containmentGroup)
{
Data::GenericTable<Data::Generic> subs;
if (!Layouts::Storage::self()->isLatteContainment(containmentGroup)) {
return subs;
}
auto applets = containmentGroup.group("Applets");
for (const auto &applet : applets.groupList()) {
if (isSubContainment(applets.group(applet))) {
Data::Generic subdata;
subdata.id = QString::number(subContainmentId(applets.group(applet)));
subs << subdata;
}
}
return subs;
}
Data::View Storage::view(const Layout::GenericLayout *layout, const Plasma::Containment *lattecontainment)
{
Data::View vdata;
if (!layout || !Layouts::Storage::self()->isLatteContainment(lattecontainment)) {
return vdata;
}
vdata = view(lattecontainment->config());
vdata.screen = lattecontainment->screen();
if (!isValid(vdata.screen)) {
vdata.screen = lattecontainment->lastScreen();
}
vdata.subcontainments = subcontainments(layout, lattecontainment);
return vdata;
}
Data::View Storage::view(const KConfigGroup &containmentGroup)
{
Data::View vdata;
if (!Layouts::Storage::self()->isLatteContainment(containmentGroup)) {
return vdata;
}
vdata.id = containmentGroup.name();
2021-04-01 11:00:59 +03:00
vdata.name = containmentGroup.readEntry("name", QString());
vdata.isActive = false;
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
vdata.screensGroup = static_cast<Latte::Types::ScreensGroup>(containmentGroup.readEntry("screensGroup", (int)Latte::Types::SingleScreenGroup));
vdata.onPrimary = containmentGroup.readEntry("onPrimary", true);
vdata.screen = containmentGroup.readEntry("lastScreen", IDNULL);
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
vdata.isClonedFrom = containmentGroup.readEntry("isClonedFrom", Data::View::ISCLONEDNULL);
vdata.screenEdgeMargin = containmentGroup.group("General").readEntry("screenEdgeMargin", (int)-1);
int location = containmentGroup.readEntry("location", (int)Plasma::Types::BottomEdge);
vdata.edge = (Plasma::Types::Location)location;
vdata.maxLength = containmentGroup.group("General").readEntry("maxLength", (float)100.0);
int alignment = containmentGroup.group("General").readEntry("alignment", (int)Latte::Types::Center) ;
vdata.alignment = (Latte::Types::Alignment)alignment;
vdata.subcontainments = subcontainments(containmentGroup);
vdata.setState(Data::View::IsCreated);
return vdata;
}
void Storage::updateView(KConfigGroup viewGroup, const Data::View &viewData)
{
if (!Layouts::Storage::self()->isLatteContainment(viewGroup)) {
return;
}
viewGroup.writeEntry("name", viewData.name);
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
viewGroup.writeEntry("screensGroup", (int)viewData.screensGroup);
viewGroup.writeEntry("onPrimary", viewData.onPrimary);
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
viewGroup.writeEntry("isClonedFrom", viewData.isClonedFrom);
viewGroup.writeEntry("lastScreen", viewData.screen);
viewGroup.group("General").writeEntry("screenEdgeMargin", viewData.screenEdgeMargin);
viewGroup.writeEntry("location", (int)viewData.edge);
viewGroup.writeEntry("maxLength", viewData.maxLength);
viewGroup.group("General").writeEntry("alignment", (int)viewData.alignment);
viewGroup.sync();
}
void Storage::updateView(const Layout::GenericLayout *layout, const Data::View &viewData)
{
if (!layout) {
return;
}
auto view = layout->viewForContainment(viewData.id.toUInt());
if (view) {
qDebug() << "Storage::updateView should not be called because view is active and present...";
return;
}
if (layout->isActive()) {
//! active view but is not present in active screens;
auto containment = layout->containmentForId(viewData.id.toUInt());
if (containment) {
2021-04-13 12:16:41 +03:00
//! update containment
containment->setLocation(viewData.edge);
updateView(containment->config(), viewData);
}
} else {
//! inactive view and in layout storage
KSharedConfigPtr lFile = KSharedConfig::openConfig(layout->file());
KConfigGroup containmentGroups = KConfigGroup(lFile, "Containments");
KConfigGroup viewContainment = containmentGroups.group(viewData.id);
if (viewContainment.exists() && Layouts::Storage::self()->isLatteContainment(viewContainment)) {
updateView(viewContainment, viewData);
}
}
}
void Storage::removeView(const QString &filepath, const Data::View &viewData)
{
if (!viewData.isValid()) {
return;
}
removeContainment(filepath, viewData.id);
for (int i=0; i<viewData.subcontainments.rowCount(); ++i) {
removeContainment(filepath, viewData.subcontainments[i].id);
}
}
void Storage::removeContainment(const QString &filepath, const QString &containmentId)
{
if (containmentId.isEmpty()) {
return;
}
KSharedConfigPtr lFile = KSharedConfig::openConfig(filepath);
KConfigGroup containmentGroups = KConfigGroup(lFile, "Containments");
if (!containmentGroups.group(containmentId).exists()) {
return;
}
containmentGroups.group(containmentId).deleteGroup();
lFile->reparseConfiguration();
}
QStringList Storage::storedLayoutsInMultipleFile()
{
QStringList layouts;
QString linkedFilePath = Importer::layoutUserFilePath(Layout::MULTIPLELAYOUTSHIDDENNAME);
if (linkedFilePath.isEmpty() || !QFileInfo(linkedFilePath).exists()) {
return layouts;
}
KSharedConfigPtr filePtr = KSharedConfig::openConfig(linkedFilePath);
KConfigGroup linkedContainments = KConfigGroup(filePtr, "Containments");
for(const auto &cId : linkedContainments.groupList()) {
QString layoutName = linkedContainments.group(cId).readEntry("layoutId", QString());
if (!layoutName.isEmpty() && !layouts.contains(layoutName)) {
layouts << layoutName;
}
}
return layouts;
}
QString Storage::storedView(const Layout::GenericLayout *layout, const int &containmentId)
{
//! make sure that layout and containmentId are valid
if (!layout) {
return QString();
}
if (layout->isActive()) {
auto containment = layout->containmentForId((uint)containmentId);
if (!containment || !isLatteContainment(containment)) {
return QString();
}
} else {
if (!containsView(layout->file(), containmentId)) {
return QString();
}
}
//! at this point we are sure that both layout and containmentId are acceptable
QString nextTmpStoredViewAbsolutePath = m_storageTmpDir.path() + "/" + QFileInfo(layout->name()).fileName() + "." + QString::number(containmentId) + ".stored.tmp";
QFile tempStoredViewFile(nextTmpStoredViewAbsolutePath);
if (tempStoredViewFile.exists()) {
tempStoredViewFile.remove();
}
KSharedConfigPtr destinationPtr = KSharedConfig::openConfig(nextTmpStoredViewAbsolutePath);
KConfigGroup destinationContainments = KConfigGroup(destinationPtr, "Containments");
if (layout->isActive()) {
//! update and copy containments
auto containment = layout->containmentForId((uint)containmentId);
syncContainmentConfig(containment);
KConfigGroup destinationViewContainment(&destinationContainments, QString::number(containment->id()));
containment->config().copyTo(&destinationViewContainment);
QList<Plasma::Containment *> subconts = layout->subContainmentsOf(containment->id());
for(const auto subcont : subconts) {
syncContainmentConfig(subcont);
KConfigGroup destinationsubcontainment(&destinationContainments, QString::number(subcont->id()));
subcont->config().copyTo(&destinationsubcontainment);
}
//! update with latest view data if active view is present
auto view = layout->viewForContainment(containment);
if (view) {
Data::View currentviewdata = view->data();
updateView(destinationViewContainment, currentviewdata);
}
} else {
QString containmentid = QString::number(containmentId);
KConfigGroup destinationViewContainment(&destinationContainments, containmentid);
KSharedConfigPtr originPtr = KSharedConfig::openConfig(layout->file());
KConfigGroup originContainments = KConfigGroup(originPtr, "Containments");
originContainments.group(containmentid).copyTo(&destinationViewContainment);
Data::GenericTable<Data::Generic> subconts = subcontainments(originContainments.group(containmentid));
for(int i=0; i<subconts.rowCount(); ++i) {
QString subid = subconts[i].id;
KConfigGroup destinationsubcontainment(&destinationContainments, subid);
originContainments.group(subid).copyTo(&destinationsubcontainment);
}
}
destinationPtr->reparseConfiguration();
return nextTmpStoredViewAbsolutePath;
}
fix #96,FEATURE:AllScreens and AllSecondaryScreens --This is a HUGE FEATURE and so important for multi-screens users. It is introduced as one single commit because it reimplements plenty of infrastructure changes and it will be easier to identify newly introduced bugs. --Users can now choose for their docks and panels to belong at various screen groups. The first two screen groups introduced are AllScreens and AllSecondayScreens. In the future it might be possible to provide CustomScreensGroup that the user will be able to define specific screens in which a dock or panel should be always present. --Current solution specifies an Original dock or panel and clones/copies itself automatically to other screens. So docks and panels in other screens are just real docks and panels that reference themselves to original docks and panels. --Clones are destroyed during layout startup and are automaticaly recreated. It is suggested to export your layouts through the official Layouts Editor in order to share them because in that case clones are not included in the new generated layout file. If in any case you do not this and you share your layout with any previous versions then your clones will just appear as separate docks and panels that belong to specific screens. --Automatic syncing was introduced in order to keep up-to-date the configuration of Original docks and panels with their referenced Clones. --Automatic syncing currently works for all docks and panels settings, for all normal applets configurations and for all subcontaiments configuration such as systrays. --Automatic syncing does not work for applets inside subcontainments such as Group Plasmoid. In such case it is suggested to configure your applets inside your Group Plasmoid in the original dock or panel and afterwards to trigger a recreation for the relevant clones --Manual recreation of clones is easily possible by just choosing the dock or panel to be OnPrimary or OnSpecificScreen and rechoosing afterwards the AllScreensGroup or AllSecondaryScreensGroup
2021-12-11 18:39:44 +03:00
int Storage::expectedViewScreenId(const Latte::Corona *corona, const KConfigGroup &containmentGroup) const
{
return expectedViewScreenId(corona, self()->view(containmentGroup));
}
int Storage::expectedViewScreenId(const Layout::GenericLayout *layout, const Plasma::Containment *lattecontainment) const
{
if (!layout || !layout->corona()) {
return Latte::ScreenPool::NOSCREENID;
}
return expectedViewScreenId(layout->corona(), self()->view(layout, lattecontainment));
}
int Storage::expectedViewScreenId(const Latte::Corona *corona, const Data::View &view) const
{
if (!corona || !view.isValid()) {
return Latte::ScreenPool::NOSCREENID;
}
if (view.screensGroup == Latte::Types::SingleScreenGroup || view.isCloned()) {
return view.onPrimary ? corona->screenPool()->primaryScreenId() : view.screen;
} else if (view.screensGroup == Latte::Types::AllScreensGroup) {
return corona->screenPool()->primaryScreenId();
} else if (view.screensGroup == Latte::Types::AllSecondaryScreensGroup) {
QList<int> secondaryscreens = corona->screenPool()->secondaryScreenIds();
return secondaryscreens.contains(view.screen) || secondaryscreens.isEmpty() ? view.screen : secondaryscreens[0];
}
return Latte::ScreenPool::NOSCREENID;
}
Data::ViewsTable Storage::views(const Layout::GenericLayout *layout)
{
Data::ViewsTable vtable;
if (!layout) {
return vtable;
} else if (!layout->isActive()) {
return views(layout->file());
}
for (const auto containment : (*layout->containments())) {
if (!isLatteContainment(containment)) {
continue;
}
Latte::View *vw = layout->viewForContainment(containment);
if (vw) {
vtable << vw->data();
} else {
vtable << view(layout, containment);
}
}
return vtable;
}
Data::ViewsTable Storage::views(const QString &file)
{
Data::ViewsTable vtable;
KSharedConfigPtr lFile = KSharedConfig::openConfig(file);
KConfigGroup containmentGroups = KConfigGroup(lFile, "Containments");
for (const auto &cId : containmentGroups.groupList()) {
if (Layouts::Storage::self()->isLatteContainment(containmentGroups.group(cId))) {
vtable << view(containmentGroups.group(cId));
}
}
return vtable;
}
}
}