File indexing completed on 2024-09-08 13:16:59

0001 /*
0002  *   SPDX-FileCopyrightText: 2003 Waldo Bastian <bastian@kde.org>
0003  *
0004  *   SPDX-License-Identifier: GPL-2.0-or-later
0005  *
0006  */
0007 
0008 #include "menuinfo.h"
0009 
0010 #include <QRegularExpressionMatch>
0011 
0012 #include <KDesktopFile>
0013 
0014 #include <KConfigGroup>
0015 #include <QStandardPaths>
0016 
0017 #include "globalaccel.h"
0018 #include "menufile.h"
0019 
0020 //
0021 // MenuFolderInfo
0022 //
0023 
0024 static QStringList *s_newShortcuts = nullptr;
0025 static QStringList *s_freeShortcuts = nullptr;
0026 static QStringList *s_deletedApps = nullptr;
0027 
0028 // Add separator
0029 MenuFolderInfo::~MenuFolderInfo()
0030 {
0031     qDeleteAll(subFolders);
0032     subFolders.clear();
0033 }
0034 
0035 void MenuFolderInfo::add(MenuSeparatorInfo *info, bool initial)
0036 {
0037     if (initial) {
0038         initialLayout.append(info);
0039     }
0040 }
0041 
0042 // Add sub menu
0043 void MenuFolderInfo::add(MenuFolderInfo *info, bool initial)
0044 {
0045     subFolders.append(info);
0046     if (initial) {
0047         initialLayout.append(info);
0048     }
0049 }
0050 
0051 // Remove sub menu (without deleting it)
0052 void MenuFolderInfo::take(MenuFolderInfo *info)
0053 {
0054     subFolders.removeAll(info);
0055 }
0056 
0057 // Remove sub menu (without deleting it)
0058 bool MenuFolderInfo::takeRecursive(MenuFolderInfo *info)
0059 {
0060     if (subFolders.removeAll(info) > 0) {
0061         return true;
0062     }
0063 
0064     foreach (MenuFolderInfo *subFolderInfo, subFolders) {
0065         if (subFolderInfo->takeRecursive(info)) {
0066             return true;
0067         }
0068     }
0069     return false;
0070 }
0071 
0072 // Recursively update all fullIds
0073 void MenuFolderInfo::updateFullId(const QString &parentId)
0074 {
0075     fullId = parentId + id;
0076 
0077     foreach (MenuFolderInfo *subFolderInfo, subFolders) {
0078         subFolderInfo->updateFullId(fullId);
0079     }
0080 }
0081 
0082 // Add entry
0083 void MenuFolderInfo::add(MenuEntryInfo *entry, bool initial)
0084 {
0085     entries.append(entry);
0086     if (initial) {
0087         initialLayout.append(entry);
0088     }
0089 }
0090 
0091 // Remove entry
0092 void MenuFolderInfo::take(MenuEntryInfo *entry)
0093 {
0094     entries.removeAll(entry);
0095 }
0096 
0097 QString uniqueCaption(const QString &caption)
0098 {
0099     static const QRegularExpression re(QStringLiteral("(.*)(?=-\\d+)"));
0100     const QRegularExpressionMatch match = re.match(caption);
0101     return match.hasMatch() ? match.captured(1) : caption;
0102 }
0103 
0104 // Return a unique sub-menu caption inspired by @p caption
0105 QString MenuFolderInfo::uniqueMenuCaption(const QString &caption)
0106 {
0107     QString cap = uniqueCaption(caption);
0108 
0109     QString result = caption;
0110 
0111     for (int n = 1; ++n;) {
0112         bool ok = true;
0113         foreach (MenuFolderInfo *subFolderInfo, subFolders) {
0114             if (subFolderInfo->caption == result) {
0115                 ok = false;
0116                 break;
0117             }
0118         }
0119         if (ok) {
0120             return result;
0121         }
0122 
0123         result = cap + QStringLiteral("-%1").arg(n);
0124     }
0125     return QString(); // Never reached
0126 }
0127 
0128 // Return a unique item caption inspired by @p caption
0129 QString MenuFolderInfo::uniqueItemCaption(const QString &caption, const QString &exclude)
0130 {
0131     QString cap = uniqueCaption(caption);
0132 
0133     QString result = caption;
0134 
0135     for (int n = 1; ++n;) {
0136         bool ok = true;
0137         if (result == exclude) {
0138             ok = false;
0139         }
0140         foreach (MenuEntryInfo *entryInfo, entries) {
0141             if (entryInfo->caption == result) {
0142                 ok = false;
0143                 break;
0144             }
0145         }
0146         if (ok) {
0147             return result;
0148         }
0149 
0150         result = cap + QStringLiteral("-%1").arg(n);
0151     }
0152     return QString(); // Never reached
0153 }
0154 
0155 // Return a list of existing submenu ids
0156 QStringList MenuFolderInfo::existingMenuIds()
0157 {
0158     QStringList result;
0159     foreach (MenuFolderInfo *subFolderInfo, subFolders) {
0160         result.append(subFolderInfo->id);
0161     }
0162     return result;
0163 }
0164 
0165 void MenuFolderInfo::setDirty()
0166 {
0167     dirty = true;
0168 }
0169 
0170 void MenuFolderInfo::save(MenuFile *menuFile)
0171 {
0172     if (s_deletedApps) {
0173         // Remove hotkeys for applications that have been deleted
0174         for (QStringList::ConstIterator it = s_deletedApps->constBegin(); it != s_deletedApps->constEnd(); ++it) {
0175             // The shorcut is deleted if we set a empty sequence
0176             GlobalAccel::changeMenuEntryShortcut(KService::serviceByStorageId(*it), QKeySequence());
0177         }
0178         delete s_deletedApps;
0179         s_deletedApps = nullptr;
0180     }
0181 
0182     if (dirty) {
0183         QString local = KDesktopFile::locateLocal(directoryFile);
0184 
0185         KDesktopFile *df = nullptr;
0186         if (directoryFile != local) {
0187             KDesktopFile orig(QStandardPaths::ApplicationsLocation, directoryFile);
0188             df = orig.copyTo(local);
0189         } else {
0190             df = new KDesktopFile(QStandardPaths::ApplicationsLocation, directoryFile);
0191         }
0192 
0193         KConfigGroup dg(df->desktopGroup());
0194         dg.writeEntry("Name", caption);
0195         dg.writeEntry("GenericName", genericname);
0196         dg.writeEntry("Comment", comment);
0197         dg.writeEntry("Icon", icon);
0198         dg.sync();
0199         delete df;
0200         dirty = false;
0201     }
0202 
0203     // Save sub-menus
0204     foreach (MenuFolderInfo *subFolderInfo, subFolders) {
0205         subFolderInfo->save(menuFile);
0206     }
0207 
0208     // Save entries
0209     foreach (MenuEntryInfo *entryInfo, entries) {
0210         if (entryInfo->needInsertion()) {
0211             menuFile->addEntry(fullId, entryInfo->menuId());
0212         }
0213         entryInfo->save();
0214     }
0215 }
0216 
0217 bool MenuFolderInfo::hasDirt()
0218 {
0219     if (dirty) {
0220         return true;
0221     }
0222 
0223     // Check sub-menus
0224     foreach (MenuFolderInfo *subFolderInfo, subFolders) {
0225         if (subFolderInfo->hasDirt()) {
0226             return true;
0227         }
0228     }
0229 
0230     // Check entries
0231     foreach (MenuEntryInfo *entryInfo, entries) {
0232         if (entryInfo->dirty || entryInfo->shortcutDirty) {
0233             return true;
0234         }
0235     }
0236     return false;
0237 }
0238 
0239 KService::Ptr MenuFolderInfo::findServiceShortcut(const QKeySequence &cut)
0240 {
0241     KService::Ptr result;
0242     // Check sub-menus
0243     foreach (MenuFolderInfo *subFolderInfo, subFolders) {
0244         result = subFolderInfo->findServiceShortcut(cut);
0245         if (result) {
0246             return result;
0247         }
0248     }
0249 
0250     // Check entries
0251     foreach (MenuEntryInfo *entryInfo, entries) {
0252         if (entryInfo->shortCut == cut) {
0253             return entryInfo->service;
0254         }
0255     }
0256     return KService::Ptr();
0257 }
0258 
0259 void MenuFolderInfo::setInUse(bool inUse)
0260 {
0261     // Propagate to sub-menus
0262     foreach (MenuFolderInfo *subFolderInfo, subFolders) {
0263         subFolderInfo->setInUse(inUse);
0264     }
0265 
0266     // Propagate to entries
0267     foreach (MenuEntryInfo *entryInfo, entries) {
0268         entryInfo->setInUse(inUse);
0269     }
0270 }
0271 
0272 //
0273 // MenuEntryInfo
0274 //
0275 
0276 MenuEntryInfo::~MenuEntryInfo()
0277 {
0278     m_desktopFile->markAsClean();
0279     delete m_desktopFile;
0280 }
0281 
0282 KDesktopFile *MenuEntryInfo::desktopFile()
0283 {
0284     if (!m_desktopFile) {
0285         m_desktopFile = new KDesktopFile(service->entryPath());
0286     }
0287     return m_desktopFile;
0288 }
0289 
0290 void MenuEntryInfo::setDirty()
0291 {
0292     if (dirty) {
0293         return;
0294     }
0295 
0296     dirty = true;
0297 
0298     QString local = QStandardPaths::writableLocation(QStandardPaths::ApplicationsLocation) + QLatin1Char('/') + service->menuId();
0299     if (local != service->entryPath()) {
0300         KDesktopFile *oldDf = desktopFile();
0301         m_desktopFile = oldDf->copyTo(local);
0302         delete oldDf;
0303     }
0304 }
0305 
0306 bool MenuEntryInfo::needInsertion()
0307 {
0308     // If entry is dirty and previously stored under applnk, then we need to be added explicitly
0309     return dirty && !service->entryPath().startsWith(QLatin1Char('/'));
0310 }
0311 
0312 void MenuEntryInfo::save()
0313 {
0314     if (dirty) {
0315         m_desktopFile->sync();
0316         dirty = false;
0317     }
0318     if (shortcutDirty) {
0319         GlobalAccel::changeMenuEntryShortcut(service, shortCut);
0320         shortcutDirty = false;
0321     }
0322 }
0323 
0324 void MenuEntryInfo::setCaption(const QString &_caption)
0325 {
0326     if (caption == _caption) {
0327         return;
0328     }
0329     caption = _caption;
0330     setDirty();
0331     desktopFile()->desktopGroup().writeEntry("Name", caption);
0332 }
0333 
0334 void MenuEntryInfo::setDescription(const QString &_description)
0335 {
0336     if (description == _description) {
0337         return;
0338     }
0339     description = _description;
0340     setDirty();
0341     desktopFile()->desktopGroup().writeEntry("GenericName", description);
0342 }
0343 
0344 void MenuEntryInfo::setIcon(const QString &_icon)
0345 {
0346     if (icon == _icon) {
0347         return;
0348     }
0349 
0350     icon = _icon;
0351     setDirty();
0352     desktopFile()->desktopGroup().writeEntry("Icon", icon);
0353 }
0354 
0355 QKeySequence MenuEntryInfo::shortcut()
0356 {
0357     if (!shortcutLoaded) {
0358         shortcutLoaded = true;
0359         shortCut = GlobalAccel::getMenuEntryShortcut(service);
0360     }
0361     return shortCut;
0362 }
0363 
0364 static void freeShortcut(const QKeySequence &shortCut)
0365 {
0366     if (!shortCut.isEmpty()) {
0367         QString shortcutKey = shortCut.toString();
0368         if (s_newShortcuts) {
0369             s_newShortcuts->removeAll(shortcutKey);
0370         }
0371 
0372         if (!s_freeShortcuts) {
0373             s_freeShortcuts = new QStringList;
0374         }
0375 
0376         s_freeShortcuts->append(shortcutKey);
0377     }
0378 }
0379 
0380 static void allocateShortcut(const QKeySequence &shortCut)
0381 {
0382     if (!shortCut.isEmpty()) {
0383         QString shortcutKey = shortCut.toString();
0384         if (s_freeShortcuts) {
0385             s_freeShortcuts->removeAll(shortcutKey);
0386         }
0387 
0388         if (!s_newShortcuts) {
0389             s_newShortcuts = new QStringList;
0390         }
0391 
0392         s_newShortcuts->append(shortcutKey);
0393     }
0394 }
0395 
0396 void MenuEntryInfo::setShortcut(const QKeySequence &_shortcut)
0397 {
0398     if (shortCut == _shortcut) {
0399         return;
0400     }
0401 
0402     freeShortcut(shortCut);
0403     allocateShortcut(_shortcut);
0404 
0405     shortCut = _shortcut;
0406     if (shortCut.isEmpty()) {
0407         shortCut = QKeySequence(); // Normalize
0408     }
0409     shortcutLoaded = true;
0410     shortcutDirty = true;
0411 }
0412 
0413 void MenuEntryInfo::setInUse(bool inUse)
0414 {
0415     if (inUse) {
0416         QKeySequence temp = shortcut();
0417         shortCut = QKeySequence();
0418         if (isShortcutAvailable(temp)) {
0419             shortCut = temp;
0420         } else {
0421             shortcutDirty = true;
0422         }
0423         allocateShortcut(shortCut);
0424 
0425         if (s_deletedApps) {
0426             s_deletedApps->removeAll(service->storageId());
0427         }
0428     } else {
0429         freeShortcut(shortcut());
0430 
0431         // Add to list of deleted apps
0432         if (!s_deletedApps) {
0433             s_deletedApps = new QStringList;
0434         }
0435 
0436         s_deletedApps->append(service->storageId());
0437     }
0438 }
0439 
0440 bool MenuEntryInfo::isShortcutAvailable(const QKeySequence &_shortcut)
0441 {
0442     // We only have to check agains not saved local shortcuts.
0443     // KKeySequenceWidget checks against all other registered shortcuts.
0444     if (shortCut == _shortcut) {
0445         return true;
0446     }
0447 
0448     QString shortcutKey = _shortcut.toString();
0449     bool available = true;
0450     if (available && s_newShortcuts) {
0451         available = !s_newShortcuts->contains(shortcutKey);
0452     }
0453     if (!available && s_freeShortcuts) {
0454         available = s_freeShortcuts->contains(shortcutKey);
0455     }
0456     return available;
0457 }