Browse Source

Support folder based UI Themes

Support folder based Themes in UIThemeManager.
Add option to select config.json as them file.

PR #15888.
adaptive-webui-19844
Prince Gupta 3 years ago committed by GitHub
parent
commit
7e8a176751
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
  1. 2
      src/gui/optionsdialog.cpp
  2. 135
      src/gui/uithememanager.cpp
  3. 13
      src/gui/uithememanager.h

2
src/gui/optionsdialog.cpp

@ -256,7 +256,7 @@ OptionsDialog::OptionsDialog(QWidget *parent)
m_ui->customThemeFilePath->setSelectedPath(Preferences::instance()->customUIThemePath()); m_ui->customThemeFilePath->setSelectedPath(Preferences::instance()->customUIThemePath());
m_ui->customThemeFilePath->setMode(FileSystemPathEdit::Mode::FileOpen); m_ui->customThemeFilePath->setMode(FileSystemPathEdit::Mode::FileOpen);
m_ui->customThemeFilePath->setDialogCaption(tr("Select qBittorrent UI Theme file")); m_ui->customThemeFilePath->setDialogCaption(tr("Select qBittorrent UI Theme file"));
m_ui->customThemeFilePath->setFileNameFilter(tr("qBittorrent UI Theme file (*.qbtheme)")); m_ui->customThemeFilePath->setFileNameFilter(tr("qBittorrent UI Theme file (*.qbtheme config.json)"));
#if (defined(Q_OS_UNIX) && !defined(Q_OS_MACOS)) #if (defined(Q_OS_UNIX) && !defined(Q_OS_MACOS))
m_ui->checkUseSystemIcon->setChecked(Preferences::instance()->useSystemIconTheme()); m_ui->checkUseSystemIcon->setChecked(Preferences::instance()->useSystemIconTheme());

135
src/gui/uithememanager.cpp

@ -1,6 +1,6 @@
/* /*
* Bittorrent Client using Qt and libtorrent. * Bittorrent Client using Qt and libtorrent.
* Copyright (C) 2019 Prince Gupta <jagannatharjun11@gmail.com> * Copyright (C) 2019, 2021 Prince Gupta <jagannatharjun11@gmail.com>
* *
* This program is free software; you can redistribute it and/or * This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License * modify it under the terms of the GNU General Public License
@ -30,6 +30,7 @@
#include "uithememanager.h" #include "uithememanager.h"
#include <QApplication> #include <QApplication>
#include <QDir>
#include <QFile> #include <QFile>
#include <QJsonDocument> #include <QJsonDocument>
#include <QJsonObject> #include <QJsonObject>
@ -42,9 +43,16 @@
namespace namespace
{ {
const QString ICONS_DIR = QStringLiteral(":icons/"); const QString CONFIG_FILE_NAME = QStringLiteral("config.json");
const QString THEME_ICONS_DIR = QStringLiteral(":uitheme/icons/"); const QString DEFAULT_ICONS_DIR = QStringLiteral(":icons/");
const QString CONFIG_FILE_NAME = QStringLiteral(":uitheme/config.json"); const QString STYLESHEET_FILE_NAME = QStringLiteral("stylesheet.qss");
// Directory used by stylesheet to reference internal resources
// for example `icon: url(:/uitheme/file.svg)` will be expected to
// point to a file `file.svg` in root directory of CONFIG_FILE_NAME
const QString STYLESHEET_RESOURCES_DIR = QStringLiteral(":/uitheme/");
const QString THEME_ICONS_DIR = QStringLiteral("icons/");
QString findIcon(const QString &iconId, const QString &dir) QString findIcon(const QString &iconId, const QString &dir)
{ {
@ -58,6 +66,90 @@ namespace
return {}; return {};
} }
QByteArray readFile(const QString &fileName)
{
QFile file {fileName};
if (!file.open(QIODevice::ReadOnly | QIODevice::Text))
{
LogMsg(UIThemeManager::tr("UITheme - Failed to open \"%1\". Reason: %2")
.arg(QFileInfo(fileName).fileName(), file.errorString())
, Log::WARNING);
return {};
}
return file.readAll();
}
class QRCThemeSource final : public UIThemeSource
{
public:
QByteArray readStyleSheet() override
{
return readFile(m_qrcThemeDir + STYLESHEET_FILE_NAME);
}
QByteArray readConfig() override
{
return readFile(m_qrcThemeDir + CONFIG_FILE_NAME);
}
QString iconPath(const QString &iconId) const override
{
return findIcon(iconId, m_qrcIconsDir);
}
private:
const QString m_qrcThemeDir {":/uitheme/"};
const QString m_qrcIconsDir = m_qrcThemeDir + THEME_ICONS_DIR;
};
class FolderThemeSource final : public UIThemeSource
{
public:
explicit FolderThemeSource(const QDir &dir)
: m_folder {dir}
, m_iconsDir {m_folder.absolutePath() + '/' + THEME_ICONS_DIR}
{
}
QByteArray readStyleSheet() override
{
QByteArray styleSheetData = readFile(m_folder.absoluteFilePath(STYLESHEET_FILE_NAME));
return styleSheetData.replace(STYLESHEET_RESOURCES_DIR.toUtf8(), (m_folder.absolutePath() + '/').toUtf8());
}
QByteArray readConfig() override
{
return readFile(m_folder.absoluteFilePath(CONFIG_FILE_NAME));
}
QString iconPath(const QString &iconId) const override
{
return findIcon(iconId, m_iconsDir);
}
private:
const QDir m_folder;
const QString m_iconsDir;
};
std::unique_ptr<UIThemeSource> createUIThemeSource(const QString &themePath)
{
const QFileInfo themeInfo {themePath};
if (themeInfo.fileName() == CONFIG_FILE_NAME)
return std::make_unique<FolderThemeSource>(themeInfo.dir());
if ((themeInfo.suffix() == QLatin1String {"qbtheme"})
&& QResource::registerResource(themePath, QLatin1String {"/uitheme"}))
{
return std::make_unique<QRCThemeSource>();
}
return nullptr;
}
} }
UIThemeManager *UIThemeManager::m_instance = nullptr; UIThemeManager *UIThemeManager::m_instance = nullptr;
@ -80,12 +172,13 @@ UIThemeManager::UIThemeManager()
, m_useSystemTheme(Preferences::instance()->useSystemIconTheme()) , m_useSystemTheme(Preferences::instance()->useSystemIconTheme())
#endif #endif
{ {
const Preferences *const pref = Preferences::instance();
if (m_useCustomTheme) if (m_useCustomTheme)
{ {
if (!QResource::registerResource(pref->customUIThemePath(), "/uitheme")) const QString themePath = Preferences::instance()->customUIThemePath();
m_themeSource = createUIThemeSource(themePath);
if (!m_themeSource)
{ {
LogMsg(tr("Failed to load UI theme from file: \"%1\"").arg(pref->customUIThemePath()), Log::WARNING); LogMsg(tr("Failed to load UI theme from file: \"%1\"").arg(themePath), Log::WARNING);
} }
else else
{ {
@ -103,16 +196,7 @@ UIThemeManager *UIThemeManager::instance()
void UIThemeManager::applyStyleSheet() const void UIThemeManager::applyStyleSheet() const
{ {
QFile qssFile(":uitheme/stylesheet.qss"); qApp->setStyleSheet(m_themeSource->readStyleSheet());
if (!qssFile.open(QIODevice::ReadOnly | QIODevice::Text))
{
qApp->setStyleSheet({});
LogMsg(tr("Couldn't apply theme stylesheet. stylesheet.qss couldn't be opened. Reason: %1").arg(qssFile.errorString())
, Log::WARNING);
return;
}
qApp->setStyleSheet(qssFile.readAll());
} }
QIcon UIThemeManager::getIcon(const QString &iconId, const QString &fallback) const QIcon UIThemeManager::getIcon(const QString &iconId, const QString &fallback) const
@ -210,34 +294,31 @@ QString UIThemeManager::getIconPath(const QString &iconId) const
QString UIThemeManager::getIconPathFromResources(const QString &iconId, const QString &fallback) const QString UIThemeManager::getIconPathFromResources(const QString &iconId, const QString &fallback) const
{ {
if (m_useCustomTheme) if (m_useCustomTheme && m_themeSource)
{ {
const QString customIcon = findIcon(iconId, THEME_ICONS_DIR); const QString customIcon = m_themeSource->iconPath(iconId);
if (!customIcon.isEmpty()) if (!customIcon.isEmpty())
return customIcon; return customIcon;
if (!fallback.isEmpty()) if (!fallback.isEmpty())
{ {
const QString fallbackIcon = findIcon(fallback, THEME_ICONS_DIR); const QString fallbackIcon = m_themeSource->iconPath(fallback);
if (!fallbackIcon.isEmpty()) if (!fallbackIcon.isEmpty())
return fallbackIcon; return fallbackIcon;
} }
} }
return findIcon(iconId, ICONS_DIR); return findIcon(iconId, DEFAULT_ICONS_DIR);
} }
void UIThemeManager::loadColorsFromJSONConfig() void UIThemeManager::loadColorsFromJSONConfig()
{ {
QFile configFile(CONFIG_FILE_NAME); const QByteArray config = m_themeSource->readConfig();
if (!configFile.open(QIODevice::ReadOnly)) if (config.isEmpty())
{
LogMsg(tr("Failed to open \"%1\". Reason: %2").arg(CONFIG_FILE_NAME, configFile.errorString()), Log::WARNING);
return; return;
}
QJsonParseError jsonError; QJsonParseError jsonError;
const QJsonDocument configJsonDoc = QJsonDocument::fromJson(configFile.readAll(), &jsonError); const QJsonDocument configJsonDoc = QJsonDocument::fromJson(config, &jsonError);
if (jsonError.error != QJsonParseError::NoError) if (jsonError.error != QJsonParseError::NoError)
{ {
LogMsg(tr("\"%1\" has invalid format. Reason: %2").arg(CONFIG_FILE_NAME, jsonError.errorString()), Log::WARNING); LogMsg(tr("\"%1\" has invalid format. Reason: %2").arg(CONFIG_FILE_NAME, jsonError.errorString()), Log::WARNING);

13
src/gui/uithememanager.h

@ -36,6 +36,16 @@
#include <QObject> #include <QObject>
#include <QString> #include <QString>
class UIThemeSource
{
public:
virtual ~UIThemeSource() = default;
virtual QByteArray readStyleSheet() = 0;
virtual QByteArray readConfig() = 0;
virtual QString iconPath(const QString &iconId) const = 0;
};
class UIThemeManager : public QObject class UIThemeManager : public QObject
{ {
Q_OBJECT Q_OBJECT
@ -64,10 +74,11 @@ private:
void applyStyleSheet() const; void applyStyleSheet() const;
static UIThemeManager *m_instance; static UIThemeManager *m_instance;
const bool m_useCustomTheme;
std::unique_ptr<UIThemeSource> m_themeSource;
QHash<QString, QColor> m_colors; QHash<QString, QColor> m_colors;
mutable QHash<QString, QIcon> m_iconCache; mutable QHash<QString, QIcon> m_iconCache;
mutable QHash<QString, QIcon> m_flagCache; mutable QHash<QString, QIcon> m_flagCache;
const bool m_useCustomTheme;
#if (defined(Q_OS_UNIX) && !defined(Q_OS_MACOS)) #if (defined(Q_OS_UNIX) && !defined(Q_OS_MACOS))
const bool m_useSystemTheme; const bool m_useSystemTheme;
#endif #endif

Loading…
Cancel
Save