mirror of
https://github.com/d47081/qBittorrent.git
synced 2025-02-10 22:04:13 +00:00
Support folder based UI Themes
Support folder based Themes in UIThemeManager. Add option to select config.json as them file. PR #15888.
This commit is contained in:
parent
61504ae3b1
commit
7e8a176751
@ -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());
|
||||||
|
@ -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);
|
||||||
|
@ -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…
x
Reference in New Issue
Block a user