File indexing completed on 2024-09-15 03:39:50
0001 /* 0002 This file is part of the KDE project 0003 SPDX-FileCopyrightText: 1998, 1999 Torben Weis <weis@kde.org> 0004 SPDX-FileCopyrightText: 1999-2006 David Faure <faure@kde.org> 0005 SPDX-FileCopyrightText: 2022 Harald Sitter <sitter@kde.org> 0006 0007 SPDX-License-Identifier: LGPL-2.0-or-later 0008 */ 0009 0010 #ifndef KSERVICE_H 0011 #define KSERVICE_H 0012 0013 #include "kserviceaction.h" 0014 #include <QCoreApplication> 0015 #include <QStringList> 0016 #include <QVariant> 0017 #include <kserviceconversioncheck_p.h> 0018 #include <ksycocaentry.h> 0019 0020 #include <optional> 0021 0022 class QDataStream; 0023 class KDesktopFile; 0024 class QWidget; 0025 0026 class KServicePrivate; 0027 0028 /** 0029 * @class KService kservice.h <KService> 0030 * 0031 * Represents an installed application. 0032 * 0033 * To obtain a KService instance for a specific application you typically use serviceByDesktopName(), e.g.: 0034 * 0035 * @code 0036 * KService::Ptr service = KService::serviceByDesktopName("org.kde.kate"); 0037 * @endcode 0038 * 0039 * Other typical usage would be in combination with KApplicationTrader to obtain e.g. the default application for a given file type. 0040 * 0041 * @see <a href="https://specifications.freedesktop.org/desktop-entry-spec/latest/">Desktop Entry Specification</a> 0042 */ 0043 class KSERVICE_EXPORT KService : public KSycocaEntry 0044 { 0045 public: 0046 /** 0047 * A shared data pointer for KService. 0048 */ 0049 typedef QExplicitlySharedDataPointer<KService> Ptr; 0050 /** 0051 * A list of shared data pointers for KService. 0052 */ 0053 typedef QList<Ptr> List; 0054 0055 /** 0056 * Construct a temporary service with a given name, exec-line and icon. 0057 * @param name the name of the service 0058 * @param exec the executable 0059 * @param icon the name of the icon 0060 */ 0061 KService(const QString &name, const QString &exec, const QString &icon); 0062 0063 /** 0064 * Construct a service and take all information from a .desktop file. 0065 * 0066 * @param fullpath Full path to the .desktop file. 0067 */ 0068 explicit KService(const QString &fullpath); 0069 0070 /** 0071 * Construct a service and take all information from a desktop file. 0072 * @param config the desktop file to read 0073 * @param optional relative path to store for findByName 0074 */ 0075 explicit KService(const KDesktopFile *config, const QString &entryPath = QString()); 0076 0077 KService(const KService &other); 0078 0079 ~KService() override; 0080 0081 /** 0082 * Whether this service is an application 0083 * @return true if this service is an application, i.e. it has Type=Application in its 0084 * .desktop file and exec() will not be empty. 0085 */ 0086 bool isApplication() const; 0087 0088 /** 0089 * Returns the executable. 0090 * @return the command that the service executes, 0091 * or QString() if not set 0092 */ 0093 QString exec() const; 0094 0095 /** 0096 * Returns the name of the icon. 0097 * @return the icon associated with the service, 0098 * or QString() if not set 0099 */ 0100 QString icon() const; 0101 /** 0102 * Checks whether the application should be run in a terminal. 0103 * 0104 * This corresponds to `Terminal=true` in the .desktop file. 0105 * 0106 * @return @c true if the application should be run in a terminal. 0107 */ 0108 bool terminal() const; 0109 0110 /** 0111 * Returns any options associated with the terminal the application 0112 * runs in, if it requires a terminal. 0113 * 0114 * The application must be a TTY-oriented program. 0115 * @return the terminal options, 0116 * or QString() if not set 0117 */ 0118 QString terminalOptions() const; 0119 0120 /** 0121 * Returns @c true if the application indicates that it's preferred to run 0122 * on a discrete graphics card, otherwise return @c false. 0123 * 0124 * In releases older than 5.86, this method checked for the @c X-KDE-RunOnDiscreteGpu 0125 * key in the .desktop file represented by this service; starting from 5.86 this method 0126 * now also checks for @c PrefersNonDefaultGPU key (added to the Freedesktop.org desktop 0127 * entry spec in version 1.4 of the spec). 0128 * 0129 * @since 5.30 0130 */ 0131 bool runOnDiscreteGpu() const; 0132 0133 /** 0134 * @brief Checks whether the application needs to run under a different UID. 0135 * @return @c true if the application needs to run under a different UID. 0136 * @see username() 0137 */ 0138 bool substituteUid() const; 0139 /** 0140 * Returns the user name if the application runs with a 0141 * different user id. 0142 * @return the username under which the service has to be run, 0143 * or QString() if not set 0144 * @see substituteUid() 0145 */ 0146 QString username() const; 0147 0148 /** 0149 * Returns the filename of the desktop entry without any 0150 * extension, e.g. "org.kde.kate" 0151 * @return the name of the desktop entry without path or extension, 0152 * or QString() if not set 0153 */ 0154 QString desktopEntryName() const; 0155 0156 /** 0157 * Returns the menu ID of the application desktop entry. 0158 * The menu ID is used to add or remove the entry to a menu. 0159 * @return the menu ID 0160 */ 0161 QString menuId() const; 0162 0163 /** 0164 * Returns a normalized ID suitable for storing in configuration files. 0165 * It will be based on the menu-id when available and otherwise falls 0166 * back to entryPath() 0167 * @return the storage ID 0168 */ 0169 QString storageId() const; 0170 0171 /** 0172 * @return the working directory to run the program in, 0173 * or QString() if not set 0174 * @since 5.63 0175 */ 0176 QString workingDirectory() const; 0177 0178 /** 0179 * Returns the descriptive comment for the application, if there is one. 0180 * @return the descriptive comment for the application, or QString() 0181 * if not set 0182 */ 0183 QString comment() const; 0184 0185 /** 0186 * Returns the generic name for the application, if there is one 0187 * (e.g. "Mail Client"). 0188 * @return the generic name, 0189 * or QString() if not set 0190 */ 0191 QString genericName() const; 0192 0193 /** 0194 * Returns the untranslated (US English) generic name 0195 * for the application, if there is one 0196 * (e.g. "Mail Client"). 0197 * @return the generic name, 0198 * or QString() if not set 0199 */ 0200 QString untranslatedGenericName() const; 0201 0202 /** 0203 * @return untranslated name for the given service 0204 * 0205 * @since 6.0 0206 */ 0207 QString untranslatedName() const; 0208 /** 0209 * Returns a list of descriptive keywords for the application, if there are any. 0210 * @return the list of keywords 0211 */ 0212 QStringList keywords() const; 0213 0214 /** 0215 * Returns a list of VFolder categories. 0216 * @return the list of VFolder categories 0217 */ 0218 QStringList categories() const; 0219 0220 /** 0221 * Returns the list of MIME types that this application supports. 0222 * Note that this doesn't include inherited MIME types, 0223 * only the MIME types listed in the .desktop file. 0224 * @since 4.8.3 0225 */ 0226 QStringList mimeTypes() const; 0227 0228 /** 0229 * Returns the list of scheme handlers this application supports. 0230 * 0231 * For example a web browser could return {"http", "https"}. 0232 * 0233 * This is taken from the x-scheme-handler MIME types 0234 * listed in the .desktop file. 0235 * 0236 * @since 6.0 0237 * 0238 */ 0239 QStringList schemeHandlers() const; 0240 0241 /** 0242 * Returns the list of protocols this application supports. 0243 * 0244 * This is taken from the x-scheme-handler MIME types 0245 * listed in the .desktop file as well as the 'X-KDE-Protocols' 0246 * entry 0247 * 0248 * For example a web browser could return {"http", "https"}. 0249 * @since 6.0 0250 */ 0251 QStringList supportedProtocols() const; 0252 0253 /** 0254 * Checks whether the application supports this MIME type 0255 * @param mimeType The name of the MIME type you are 0256 * interested in determining whether this service supports. 0257 * @since 4.6 0258 */ 0259 bool hasMimeType(const QString &mimeType) const; 0260 0261 /** 0262 * Returns the actions defined in this desktop file 0263 */ 0264 QList<KServiceAction> actions() const; 0265 0266 /** 0267 * Checks whether this application can handle several files as 0268 * startup arguments. 0269 * @return true if multiple files may be passed to this service at 0270 * startup. False if only one file at a time may be passed. 0271 */ 0272 bool allowMultipleFiles() const; 0273 0274 /** 0275 * What preference to associate with this application initially (before 0276 * the user has had any chance to define a profile for it). 0277 * The bigger the value, the stronger the preference for the application. 0278 * @return the service preference level of the application 0279 */ 0280 int initialPreference() const; 0281 0282 /** 0283 * Whether the entry should be hidden from the menu. 0284 * @return @c true to hide this application from the menu 0285 * 0286 * Such services still appear in trader queries, i.e. in 0287 * "Open With" popup menus for instance. 0288 */ 0289 bool noDisplay() const; 0290 0291 /** 0292 * Whether the application should be shown in the current desktop 0293 * (including in context menus). 0294 * @return true if the application should be shown in the current desktop. 0295 * 0296 * KApplicationTrader honors this and removes such services 0297 * from its results. 0298 * 0299 * @since 5.0 0300 */ 0301 bool showInCurrentDesktop() const; 0302 0303 /** 0304 * Whether the application should be shown on the current 0305 * platform (e.g. on xcb or on wayland). 0306 * @return @c true if the application should be shown on the current platform. 0307 * 0308 * @since 5.0 0309 */ 0310 bool showOnCurrentPlatform() const; 0311 0312 /** 0313 * The path to the documentation for this application. 0314 * @since 4.2 0315 * @return the documentation path, or QString() if not set 0316 */ 0317 QString docPath() const; 0318 0319 /** 0320 * Returns the requested property. 0321 * 0322 * @tparam T the type of the requested property. 0323 * 0324 * @param name the name of the property. 0325 * 0326 * @since 6.0 0327 */ 0328 template<typename T> 0329 T property(const QString &name) const 0330 { 0331 KServiceConversionCheck::to_QVariant<T>(); 0332 return property(name, static_cast<QMetaType::Type>(qMetaTypeId<T>())).value<T>(); 0333 } 0334 0335 /** 0336 * Returns a path that can be used for saving changes to this 0337 * application 0338 * @return path that can be used for saving changes to this application 0339 */ 0340 QString locateLocal() const; 0341 0342 /** 0343 * @internal 0344 * Set the menu id 0345 */ 0346 void setMenuId(const QString &menuId); 0347 /** 0348 * @internal 0349 * Sets whether to use a terminal or not 0350 */ 0351 void setTerminal(bool b); 0352 /** 0353 * @internal 0354 * Sets the terminal options to use 0355 */ 0356 void setTerminalOptions(const QString &options); 0357 0358 /** 0359 * Overrides the "Exec=" line of the service. 0360 * 0361 * If @ref exec is not empty, its value will override the one 0362 * the one set when this application was created. 0363 * 0364 * Please note that @ref entryPath is also cleared so the application 0365 * will no longer be associated with a specific config file. 0366 * 0367 * @internal 0368 * @since 4.11 0369 */ 0370 void setExec(const QString &exec); 0371 0372 /** 0373 * Overrides the "Path=" line of the application. 0374 * 0375 * If @ref workingDir is not empty, its value will override 0376 * the one set when this application was created. 0377 * 0378 * Please note that @ref entryPath is also cleared so the application 0379 * will no longer be associated with a specific config file. 0380 * 0381 * @internal 0382 * @param workingDir 0383 * @since 5.79 0384 */ 0385 void setWorkingDirectory(const QString &workingDir); 0386 0387 /** 0388 * Find a application based on its path as returned by entryPath(). 0389 * It's usually better to use serviceByStorageId() instead. 0390 * 0391 * @param _path the path of the configuration file 0392 * @return a pointer to the requested application or @c nullptr if the application is 0393 * unknown. 0394 * @em Very @em important: Don't store the result in a KService* ! 0395 */ 0396 static Ptr serviceByDesktopPath(const QString &_path); 0397 0398 /** 0399 * Find an application by the name of its desktop file, not depending on 0400 * its actual location (as long as it's under the applications or application 0401 * directories). For instance "konqbrowser" or "kcookiejar". Note that 0402 * the ".desktop" extension is implicit. 0403 * 0404 * This is the recommended method (safe even if the user moves stuff) 0405 * but note that it assumes that no two entries have the same filename. 0406 * 0407 * @param _name the name of the configuration file 0408 * @return a pointer to the requested application or @c nullptr if the application is 0409 * unknown. 0410 * @em Very @em important: Don't store the result in a KService* ! 0411 */ 0412 static Ptr serviceByDesktopName(const QString &_name); 0413 0414 /** 0415 * Find a application by its menu-id 0416 * 0417 * @param _menuId the menu id of the application 0418 * @return a pointer to the requested application or @c nullptr if the application is 0419 * unknown. 0420 * @em Very @em important: Don't store the result in a KService* ! 0421 */ 0422 static Ptr serviceByMenuId(const QString &_menuId); 0423 0424 /** 0425 * Find a application by its storage-id or desktop-file path. This 0426 * function will try very hard to find a matching application. 0427 * 0428 * @param _storageId the storage id or desktop-file path of the application 0429 * @return a pointer to the requested application or @c nullptr if the application is 0430 * unknown. 0431 * @em Very @em important: Don't store the result in a KService* ! 0432 */ 0433 static Ptr serviceByStorageId(const QString &_storageId); 0434 0435 /** 0436 * Returns the whole list of applications. 0437 * 0438 * Useful for being able to 0439 * to display them in a list box, for example. 0440 * More memory consuming than the ones above, don't use unless 0441 * really necessary. 0442 * @return the list of all applications 0443 */ 0444 static List allServices(); 0445 0446 /** 0447 * Returns a path that can be used to create a new KService based 0448 * on @p suggestedName. 0449 * @param showInMenu @c true, if the application should be shown in the KDE menu 0450 * @c false, if the application should be hidden from the menu 0451 * This argument isn't used anymore, use `NoDisplay=true` to hide the application. 0452 * @param suggestedName name to base the file on, if an application with such a 0453 * name already exists, a suffix will be added to make it unique 0454 * (e.g. foo.desktop, foo-1.desktop, foo-2.desktop). 0455 * @param menuId If provided, menuId will be set to the menu id to use for 0456 * the KService 0457 * @param reservedMenuIds If provided, the path and menu id will be chosen 0458 * in such a way that the new menu id does not conflict with any 0459 * of the reservedMenuIds 0460 * @return The path to use for the new KService. 0461 */ 0462 static QString newServicePath(bool showInMenu, const QString &suggestedName, QString *menuId = nullptr, const QStringList *reservedMenuIds = nullptr); 0463 0464 /** 0465 * @brief A desktop file name that this application is an alias for. 0466 * 0467 * This is used when a `NoDisplay` application is used to enforce specific handling 0468 * for an application. In that case the `NoDisplay` application is an `AliasFor` another 0469 * application and be considered roughly equal to the `AliasFor` application (which should 0470 * not be `NoDisplay=true`) 0471 * For example Okular supplies a desktop file for each supported format (e.g. PDF), all 0472 * of which `NoDisplay` and it is merely there to selectively support specific file formats. 0473 * A UI may choose to display the aliased entry org.kde.okular instead of the NoDisplay entries. 0474 * 0475 * @since 5.96 0476 * 0477 * @return QString desktopName of the aliased application (excluding .desktop suffix) 0478 */ 0479 QString aliasFor() const; 0480 0481 /** 0482 * Returns the value of StartupNotify for this service. 0483 * 0484 * If the service doesn't define a value nullopt is returned. 0485 * 0486 * See StartupNotify in the <a href="https://specifications.freedesktop.org/desktop-entry-spec/latest/">Desktop Entry Specification</a>. 0487 * 0488 * @since 6.0 0489 */ 0490 std::optional<bool> startupNotify() const; 0491 0492 private: 0493 friend class KBuildServiceFactory; 0494 0495 QVariant property(const QString &_name, QMetaType::Type t) const; 0496 0497 /// @internal for KBuildSycoca only 0498 struct KSERVICE_NO_EXPORT ServiceTypeAndPreference { 0499 ServiceTypeAndPreference() 0500 : preference(-1) 0501 , serviceType() 0502 { 0503 } 0504 ServiceTypeAndPreference(int pref, const QString &servType) 0505 : preference(pref) 0506 , serviceType(servType) 0507 { 0508 } 0509 int preference; 0510 QString serviceType; // or MIME type 0511 }; 0512 /// @internal for KBuildSycoca only 0513 QList<ServiceTypeAndPreference> _k_accessServiceTypes(); 0514 0515 void setActions(const QList<KServiceAction> &actions); 0516 0517 friend QDataStream &operator>>(QDataStream &, ServiceTypeAndPreference &); 0518 friend QDataStream &operator<<(QDataStream &, const ServiceTypeAndPreference &); 0519 0520 Q_DECLARE_PRIVATE(KService) 0521 0522 friend class KServiceFactory; 0523 0524 /** 0525 * @internal 0526 * Construct a service from a stream. 0527 * The stream must already be positioned at the correct offset. 0528 */ 0529 KSERVICE_NO_EXPORT KService(QDataStream &str, int offset); 0530 }; 0531 0532 template<> 0533 KSERVICE_EXPORT QString KService::property<QString>(const QString &name) const; 0534 0535 #endif