kde-workspace/kmenuedit/menuinfo.cpp

472 lines
10 KiB
C++
Raw Normal View History

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"
#include <QRegExp>
#include <KDesktopFile>
#include <KStandardDirs>
#include <KConfigGroup>
#include "menufile.h"
#include "khotkeys.h"
//
// MenuFolderInfo
//
static QStringList *s_newShortcuts = 0;
static QStringList *s_freeShortcuts = 0;
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)
{
// Remove hotkeys for applications that have been deleted
for(QStringList::ConstIterator it = s_deletedApps->constBegin();
it != s_deletedApps->constEnd(); ++it)
{
// The shorcut is deleted if we set a empty sequence
KHotKeys::changeMenuEntryShortcut(*it, "");
}
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)
{
if (entryInfo->dirty || entryInfo->shortcutDirty) return true;
}
return false;
}
KService::Ptr MenuFolderInfo::findServiceShortcut(const KShortcut&cut)
{
KService::Ptr result;
// Check sub-menus
foreach (MenuFolderInfo *subFolderInfo, subFolders)
{
result = subFolderInfo->findServiceShortcut(cut);
if (result)
return result;
}
// Check entries
foreach (MenuEntryInfo *entryInfo, entries)
{
if (entryInfo->shortCut == cut)
return entryInfo->service;
}
return KService::Ptr();
}
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;
}
if (shortcutDirty)
{
if( KHotKeys::present())
{
KHotKeys::changeMenuEntryShortcut( service->storageId(), shortCut.toString() );
}
shortcutDirty = 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);
}
KShortcut MenuEntryInfo::shortcut()
{
if (!shortcutLoaded)
{
shortcutLoaded = true;
if( KHotKeys::present())
{
shortCut = KShortcut(KHotKeys::getMenuEntryShortcut( service->storageId() ));
}
}
return shortCut;
}
static void freeShortcut(const KShortcut &shortCut)
{
if (!shortCut.isEmpty())
{
QString shortcutKey = shortCut.toString();
if (s_newShortcuts)
s_newShortcuts->removeAll(shortcutKey);
if (!s_freeShortcuts)
s_freeShortcuts = new QStringList;
s_freeShortcuts->append(shortcutKey);
}
}
static void allocateShortcut(const KShortcut &shortCut)
{
if (!shortCut.isEmpty())
{
QString shortcutKey = shortCut.toString();
if (s_freeShortcuts)
s_freeShortcuts->removeAll(shortcutKey);
if (!s_newShortcuts)
s_newShortcuts = new QStringList;
s_newShortcuts->append(shortcutKey);
}
}
void MenuEntryInfo::setShortcut(const KShortcut &_shortcut)
{
if (shortCut == _shortcut)
return;
freeShortcut(shortCut);
allocateShortcut(_shortcut);
shortCut = _shortcut;
if (shortCut.isEmpty())
shortCut = KShortcut(); // Normalize
shortcutLoaded = true;
shortcutDirty = true;
}
void MenuEntryInfo::setInUse(bool inUse)
{
if (inUse)
{
KShortcut temp = shortcut();
shortCut = KShortcut();
if (isShortcutAvailable(temp))
shortCut = temp;
else
shortcutDirty = true;
allocateShortcut(shortCut);
if (s_deletedApps)
s_deletedApps->removeAll(service->storageId());
}
else
{
freeShortcut(shortcut());
// Add to list of deleted apps
if (!s_deletedApps)
s_deletedApps = new QStringList;
s_deletedApps->append(service->storageId());
}
}
bool MenuEntryInfo::isShortcutAvailable(const KShortcut &_shortcut)
{
// We only have to check agains not saved local shortcuts.
// KKeySequenceWidget checks against all other registered shortcuts.
if (shortCut == _shortcut)
return true;
QString shortcutKey = _shortcut.toString();
bool available = true;
if (available && s_newShortcuts)
{
available = !s_newShortcuts->contains(shortcutKey);
}
if (!available && s_freeShortcuts)
{
available = s_freeShortcuts->contains(shortcutKey);
}
return available;
}