2014-11-13 19:30:51 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2003 Waldo Bastian <bastian@kde.org>
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "menuinfo.h"
|
2023-08-29 22:57:16 +03:00
|
|
|
#include "menufile.h"
|
2014-11-13 19:30:51 +02:00
|
|
|
|
|
|
|
#include <QRegExp>
|
|
|
|
#include <KDesktopFile>
|
|
|
|
#include <KStandardDirs>
|
|
|
|
#include <KConfigGroup>
|
|
|
|
|
|
|
|
//
|
|
|
|
// MenuFolderInfo
|
|
|
|
//
|
|
|
|
|
|
|
|
static QStringList *s_deletedApps = 0;
|
|
|
|
|
|
|
|
// Add separator
|
|
|
|
void MenuFolderInfo::add(MenuSeparatorInfo *info, bool initial)
|
|
|
|
{
|
|
|
|
if (initial)
|
|
|
|
initialLayout.append(info);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add sub menu
|
|
|
|
void MenuFolderInfo::add(MenuFolderInfo *info, bool initial)
|
|
|
|
{
|
|
|
|
subFolders.append(info);
|
|
|
|
if (initial)
|
|
|
|
initialLayout.append(info);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove sub menu (without deleting it)
|
|
|
|
void MenuFolderInfo::take(MenuFolderInfo *info)
|
|
|
|
{
|
|
|
|
subFolders.removeAll(info);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove sub menu (without deleting it)
|
|
|
|
bool MenuFolderInfo::takeRecursive(MenuFolderInfo *info)
|
|
|
|
{
|
|
|
|
if (subFolders.removeAll(info) > 0)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (MenuFolderInfo *subFolderInfo, subFolders)
|
|
|
|
{
|
|
|
|
if (subFolderInfo->takeRecursive(info))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Recursively update all fullIds
|
|
|
|
void MenuFolderInfo::updateFullId(const QString &parentId)
|
|
|
|
{
|
|
|
|
fullId = parentId + id;
|
|
|
|
|
|
|
|
foreach (MenuFolderInfo *subFolderInfo, subFolders)
|
|
|
|
{
|
|
|
|
subFolderInfo->updateFullId(fullId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add entry
|
|
|
|
void MenuFolderInfo::add(MenuEntryInfo *entry, bool initial)
|
|
|
|
{
|
|
|
|
entries.append(entry);
|
|
|
|
if (initial)
|
|
|
|
initialLayout.append(entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove entry
|
|
|
|
void MenuFolderInfo::take(MenuEntryInfo *entry)
|
|
|
|
{
|
|
|
|
entries.removeAll(entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Return a unique sub-menu caption inspired by @p caption
|
|
|
|
QString MenuFolderInfo::uniqueMenuCaption(const QString &caption)
|
|
|
|
{
|
|
|
|
QRegExp r("(.*)(?=-\\d+)");
|
|
|
|
QString cap = (r.indexIn(caption) > -1) ? r.cap(1) : caption;
|
|
|
|
|
|
|
|
QString result = caption;
|
|
|
|
|
|
|
|
for(int n = 1; ++n; )
|
|
|
|
{
|
|
|
|
bool ok = true;
|
|
|
|
foreach (MenuFolderInfo *subFolderInfo, subFolders)
|
|
|
|
{
|
|
|
|
if (subFolderInfo->caption == result)
|
|
|
|
{
|
|
|
|
ok = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (ok)
|
|
|
|
return result;
|
|
|
|
|
|
|
|
result = cap + QString("-%1").arg(n);
|
|
|
|
}
|
|
|
|
return QString(); // Never reached
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return a unique item caption inspired by @p caption
|
|
|
|
QString MenuFolderInfo::uniqueItemCaption(const QString &caption, const QString &exclude)
|
|
|
|
{
|
|
|
|
QRegExp r("(.*)(?=-\\d+)");
|
|
|
|
QString cap = (r.indexIn(caption) > -1) ? r.cap(1) : caption;
|
|
|
|
|
|
|
|
QString result = caption;
|
|
|
|
|
|
|
|
for(int n = 1; ++n; )
|
|
|
|
{
|
|
|
|
bool ok = true;
|
|
|
|
if (result == exclude)
|
|
|
|
ok = false;
|
|
|
|
foreach (MenuEntryInfo *entryInfo, entries)
|
|
|
|
{
|
|
|
|
if (entryInfo->caption == result)
|
|
|
|
{
|
|
|
|
ok = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (ok)
|
|
|
|
return result;
|
|
|
|
|
|
|
|
result = cap + QString("-%1").arg(n);
|
|
|
|
}
|
|
|
|
return QString(); // Never reached
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return a list of existing submenu ids
|
|
|
|
QStringList MenuFolderInfo::existingMenuIds()
|
|
|
|
{
|
|
|
|
QStringList result;
|
|
|
|
foreach (MenuFolderInfo *subFolderInfo, subFolders)
|
|
|
|
{
|
|
|
|
result.append(subFolderInfo->id);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MenuFolderInfo::setDirty()
|
|
|
|
{
|
|
|
|
dirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MenuFolderInfo::save(MenuFile *menuFile)
|
|
|
|
{
|
|
|
|
if (s_deletedApps)
|
|
|
|
{
|
|
|
|
delete s_deletedApps;
|
|
|
|
s_deletedApps = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dirty)
|
|
|
|
{
|
|
|
|
QString local = KDesktopFile::locateLocal(directoryFile);
|
|
|
|
|
|
|
|
KDesktopFile *df = 0;
|
|
|
|
if (directoryFile != local)
|
|
|
|
{
|
|
|
|
KDesktopFile orig("apps", directoryFile);
|
|
|
|
df = orig.copyTo(local);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
df = new KDesktopFile("apps", directoryFile);
|
|
|
|
}
|
|
|
|
|
|
|
|
KConfigGroup dg( df->desktopGroup() );
|
|
|
|
dg.writeEntry("Name", caption);
|
|
|
|
dg.writeEntry("GenericName", genericname);
|
|
|
|
dg.writeEntry("Comment", comment);
|
|
|
|
dg.writeEntry("Icon", icon);
|
|
|
|
dg.sync();
|
|
|
|
delete df;
|
|
|
|
dirty = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save sub-menus
|
|
|
|
foreach (MenuFolderInfo *subFolderInfo, subFolders)
|
|
|
|
{
|
|
|
|
subFolderInfo->save(menuFile);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save entries
|
|
|
|
foreach (MenuEntryInfo *entryInfo, entries)
|
|
|
|
{
|
|
|
|
if (entryInfo->needInsertion())
|
|
|
|
menuFile->addEntry(fullId, entryInfo->menuId());
|
|
|
|
entryInfo->save();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MenuFolderInfo::hasDirt()
|
|
|
|
{
|
|
|
|
if (dirty) return true;
|
|
|
|
|
|
|
|
// Check sub-menus
|
|
|
|
foreach (MenuFolderInfo *subFolderInfo, subFolders)
|
|
|
|
{
|
|
|
|
if (subFolderInfo->hasDirt()) return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check entries
|
|
|
|
foreach (MenuEntryInfo *entryInfo, entries)
|
|
|
|
{
|
2023-08-29 22:57:16 +03:00
|
|
|
if (entryInfo->dirty) return true;
|
2014-11-13 19:30:51 +02:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MenuFolderInfo::setInUse(bool inUse)
|
|
|
|
{
|
|
|
|
// Propagate to sub-menus
|
|
|
|
foreach (MenuFolderInfo *subFolderInfo, subFolders)
|
|
|
|
{
|
|
|
|
subFolderInfo->setInUse(inUse);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Propagate to entries
|
|
|
|
foreach (MenuEntryInfo *entryInfo, entries)
|
|
|
|
{
|
|
|
|
entryInfo->setInUse(inUse);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// MenuEntryInfo
|
|
|
|
//
|
|
|
|
|
|
|
|
MenuEntryInfo::~MenuEntryInfo()
|
|
|
|
{
|
|
|
|
m_desktopFile->markAsClean();
|
|
|
|
delete m_desktopFile;
|
|
|
|
}
|
|
|
|
|
|
|
|
KDesktopFile *MenuEntryInfo::desktopFile()
|
|
|
|
{
|
|
|
|
if (!m_desktopFile)
|
|
|
|
{
|
|
|
|
m_desktopFile = new KDesktopFile(service->entryPath());
|
|
|
|
}
|
|
|
|
return m_desktopFile;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MenuEntryInfo::setDirty()
|
|
|
|
{
|
|
|
|
if (dirty) return;
|
|
|
|
|
|
|
|
dirty = true;
|
|
|
|
|
|
|
|
QString local = KStandardDirs::locateLocal("xdgdata-apps", service->menuId());
|
|
|
|
if (local != service->entryPath())
|
|
|
|
{
|
|
|
|
KDesktopFile *oldDf = desktopFile();
|
|
|
|
m_desktopFile = oldDf->copyTo(local);
|
|
|
|
delete oldDf;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MenuEntryInfo::needInsertion()
|
|
|
|
{
|
|
|
|
// If entry is dirty and previously stored under applnk, then we need to be added explicitly
|
|
|
|
return dirty && !service->entryPath().startsWith('/');
|
|
|
|
}
|
|
|
|
|
|
|
|
void MenuEntryInfo::save()
|
|
|
|
{
|
|
|
|
if (dirty)
|
|
|
|
{
|
|
|
|
m_desktopFile->sync();
|
|
|
|
dirty = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MenuEntryInfo::setCaption(const QString &_caption)
|
|
|
|
{
|
|
|
|
if (caption == _caption)
|
|
|
|
return;
|
|
|
|
caption = _caption;
|
|
|
|
setDirty();
|
|
|
|
desktopFile()->desktopGroup().writeEntry("Name", caption);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MenuEntryInfo::setDescription(const QString &_description)
|
|
|
|
{
|
|
|
|
if (description == _description)
|
|
|
|
return;
|
|
|
|
description = _description;
|
|
|
|
setDirty();
|
|
|
|
desktopFile()->desktopGroup().writeEntry("GenericName", description);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MenuEntryInfo::setIcon(const QString &_icon)
|
|
|
|
{
|
|
|
|
if (icon == _icon)
|
|
|
|
return;
|
|
|
|
|
|
|
|
icon = _icon;
|
|
|
|
setDirty();
|
|
|
|
desktopFile()->desktopGroup().writeEntry("Icon", icon);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MenuEntryInfo::setInUse(bool inUse)
|
|
|
|
{
|
|
|
|
if (inUse)
|
|
|
|
{
|
|
|
|
if (s_deletedApps)
|
|
|
|
s_deletedApps->removeAll(service->storageId());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Add to list of deleted apps
|
|
|
|
if (!s_deletedApps)
|
|
|
|
s_deletedApps = new QStringList;
|
|
|
|
|
|
|
|
s_deletedApps->append(service->storageId());
|
|
|
|
}
|
|
|
|
}
|