// Copyright (C) 2017 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only #include "qxdgdesktopportaltheme.h" #include "qxdgdesktopportalfiledialog_p.h" #include #include #include #include #include #include #include #include #include #include QT_BEGIN_NAMESPACE using namespace Qt::StringLiterals; class QXdgDesktopPortalThemePrivate : public QPlatformThemePrivate { public: enum XdgColorschemePref { None, PreferDark, PreferLight }; QXdgDesktopPortalThemePrivate() : QPlatformThemePrivate() { } ~QXdgDesktopPortalThemePrivate() { delete baseTheme; } /*! \internal Converts the given Freedesktop color scheme setting \a colorschemePref to a Qt::Appearance value. Specification: https://github.com/flatpak/xdg-desktop-portal/blob/d7a304a00697d7d608821253cd013f3b97ac0fb6/data/org.freedesktop.impl.portal.Settings.xml#L33-L45 Unfortunately the enum numerical values are not defined identically, so we have to convert them. The mapping is as follows: Enum Index: Freedesktop definition | Qt definition ----------------------------------- | ------------- 0: No preference | 0: Unknown 1: Prefer dark appearance | 2: Dark 2: Prefer light appearance | 1: Light */ static Qt::Appearance appearanceFromXdgPref(const XdgColorschemePref colorschemePref) { switch (colorschemePref) { case PreferDark: return Qt::Appearance::Dark; case PreferLight: return Qt::Appearance::Light; default: return Qt::Appearance::Unknown; } } QPlatformTheme *baseTheme = nullptr; uint fileChooserPortalVersion = 0; Qt::Appearance appearance = Qt::Appearance::Unknown; }; QXdgDesktopPortalTheme::QXdgDesktopPortalTheme() : d_ptr(new QXdgDesktopPortalThemePrivate) { Q_D(QXdgDesktopPortalTheme); QStringList themeNames; themeNames += QGuiApplicationPrivate::platform_integration->themeNames(); // 1) Look for a theme plugin. for (const QString &themeName : std::as_const(themeNames)) { d->baseTheme = QPlatformThemeFactory::create(themeName, nullptr); if (d->baseTheme) break; } // 2) If no theme plugin was found ask the platform integration to // create a theme if (!d->baseTheme) { for (const QString &themeName : std::as_const(themeNames)) { d->baseTheme = QGuiApplicationPrivate::platform_integration->createPlatformTheme(themeName); if (d->baseTheme) break; } // No error message; not having a theme plugin is allowed. } // 3) Fall back on the built-in "null" platform theme. if (!d->baseTheme) d->baseTheme = new QPlatformTheme; // Get information about portal version QDBusMessage message = QDBusMessage::createMethodCall("org.freedesktop.portal.Desktop"_L1, "/org/freedesktop/portal/desktop"_L1, "org.freedesktop.DBus.Properties"_L1, "Get"_L1); message << "org.freedesktop.portal.FileChooser"_L1 << "version"_L1; QDBusPendingCall pendingCall = QDBusConnection::sessionBus().asyncCall(message); QDBusPendingCallWatcher *watcher = new QDBusPendingCallWatcher(pendingCall); QObject::connect(watcher, &QDBusPendingCallWatcher::finished, [d] (QDBusPendingCallWatcher *watcher) { QDBusPendingReply reply = *watcher; if (reply.isValid()) { d->fileChooserPortalVersion = reply.value().toUInt(); } watcher->deleteLater(); }); // Get information about system theme preference message = QDBusMessage::createMethodCall("org.freedesktop.portal.Desktop"_L1, "/org/freedesktop/portal/desktop"_L1, "org.freedesktop.portal.Settings"_L1, "Read"_L1); message << "org.freedesktop.appearance"_L1 << "color-scheme"_L1; // this must not be asyncCall() because we have to set appearance now QDBusReply reply = QDBusConnection::sessionBus().call(message); if (reply.isValid()) { const QDBusVariant dbusVariant = qvariant_cast(reply.value()); const QXdgDesktopPortalThemePrivate::XdgColorschemePref xdgPref = static_cast(dbusVariant.variant().toUInt()); d->appearance = QXdgDesktopPortalThemePrivate::appearanceFromXdgPref(xdgPref); } } QPlatformMenuItem* QXdgDesktopPortalTheme::createPlatformMenuItem() const { Q_D(const QXdgDesktopPortalTheme); return d->baseTheme->createPlatformMenuItem(); } QPlatformMenu* QXdgDesktopPortalTheme::createPlatformMenu() const { Q_D(const QXdgDesktopPortalTheme); return d->baseTheme->createPlatformMenu(); } QPlatformMenuBar* QXdgDesktopPortalTheme::createPlatformMenuBar() const { Q_D(const QXdgDesktopPortalTheme); return d->baseTheme->createPlatformMenuBar(); } void QXdgDesktopPortalTheme::showPlatformMenuBar() { Q_D(const QXdgDesktopPortalTheme); return d->baseTheme->showPlatformMenuBar(); } bool QXdgDesktopPortalTheme::usePlatformNativeDialog(DialogType type) const { Q_D(const QXdgDesktopPortalTheme); if (type == FileDialog) return true; return d->baseTheme->usePlatformNativeDialog(type); } QPlatformDialogHelper* QXdgDesktopPortalTheme::createPlatformDialogHelper(DialogType type) const { Q_D(const QXdgDesktopPortalTheme); if (type == FileDialog && d->fileChooserPortalVersion) { // Older versions of FileChooser portal don't support opening directories, therefore we fallback // to native file dialog opened inside the sandbox to open a directory. if (d->baseTheme->usePlatformNativeDialog(type)) return new QXdgDesktopPortalFileDialog(static_cast(d->baseTheme->createPlatformDialogHelper(type)), d->fileChooserPortalVersion); return new QXdgDesktopPortalFileDialog; } return d->baseTheme->createPlatformDialogHelper(type); } #ifndef QT_NO_SYSTEMTRAYICON QPlatformSystemTrayIcon* QXdgDesktopPortalTheme::createPlatformSystemTrayIcon() const { Q_D(const QXdgDesktopPortalTheme); return d->baseTheme->createPlatformSystemTrayIcon(); } #endif const QPalette *QXdgDesktopPortalTheme::palette(Palette type) const { Q_D(const QXdgDesktopPortalTheme); return d->baseTheme->palette(type); } const QFont* QXdgDesktopPortalTheme::font(Font type) const { Q_D(const QXdgDesktopPortalTheme); return d->baseTheme->font(type); } QVariant QXdgDesktopPortalTheme::themeHint(ThemeHint hint) const { Q_D(const QXdgDesktopPortalTheme); return d->baseTheme->themeHint(hint); } Qt::Appearance QXdgDesktopPortalTheme::appearance() const { Q_D(const QXdgDesktopPortalTheme); return d->appearance; } QPixmap QXdgDesktopPortalTheme::standardPixmap(StandardPixmap sp, const QSizeF &size) const { Q_D(const QXdgDesktopPortalTheme); return d->baseTheme->standardPixmap(sp, size); } QIcon QXdgDesktopPortalTheme::fileIcon(const QFileInfo &fileInfo, QPlatformTheme::IconOptions iconOptions) const { Q_D(const QXdgDesktopPortalTheme); return d->baseTheme->fileIcon(fileInfo, iconOptions); } QIconEngine * QXdgDesktopPortalTheme::createIconEngine(const QString &iconName) const { Q_D(const QXdgDesktopPortalTheme); return d->baseTheme->createIconEngine(iconName); } #if QT_CONFIG(shortcut) QList QXdgDesktopPortalTheme::keyBindings(QKeySequence::StandardKey key) const { Q_D(const QXdgDesktopPortalTheme); return d->baseTheme->keyBindings(key); } #endif QString QXdgDesktopPortalTheme::standardButtonText(int button) const { Q_D(const QXdgDesktopPortalTheme); return d->baseTheme->standardButtonText(button); } QT_END_NAMESPACE