mirror of
https://github.com/Chatterino/chatterino2.git
synced 2024-11-13 19:49:51 +01:00
Add command to automatically reload your theme (#4718)
This commit is contained in:
parent
fca57696bb
commit
9f8a1d8823
|
@ -13,6 +13,7 @@
|
||||||
- Minor: Add accelerators to the right click menu for messages (#4705)
|
- Minor: Add accelerators to the right click menu for messages (#4705)
|
||||||
- Minor: Add pin action to usercards and reply threads. (#4692)
|
- Minor: Add pin action to usercards and reply threads. (#4692)
|
||||||
- Minor: Stream status requests are now batched. (#4713)
|
- Minor: Stream status requests are now batched. (#4713)
|
||||||
|
- Minor: Added `/c2-theme-autoreload` command to automatically reload a custom theme. This is useful for when you're developing your own theme. (#4718)
|
||||||
- Bugfix: Increased amount of blocked users loaded from 100 to 1,000. (#4721)
|
- Bugfix: Increased amount of blocked users loaded from 100 to 1,000. (#4721)
|
||||||
- Bugfix: Fixed generation of crashdumps by the browser-extension process when the browser was closed. (#4667)
|
- Bugfix: Fixed generation of crashdumps by the browser-extension process when the browser was closed. (#4667)
|
||||||
- Bugfix: Fix spacing issue with mentions inside RTL text. (#4677)
|
- Bugfix: Fix spacing issue with mentions inside RTL text. (#4677)
|
||||||
|
|
|
@ -30,7 +30,7 @@
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"title": "SVG Color",
|
"title": "SVG Color",
|
||||||
"description": "This is stricter than Qt. You could theoretically put tabs an spaces between characters in a named color and capitalize the color.",
|
"description": "This enum is stricter than Qt. You could theoretically put tabs and spaces between characters in a named color and capitalize the color.",
|
||||||
"$comment": "https://www.w3.org/TR/SVG11/types.html#ColorKeywords",
|
"$comment": "https://www.w3.org/TR/SVG11/types.html#ColorKeywords",
|
||||||
"enum": [
|
"enum": [
|
||||||
"aliceblue",
|
"aliceblue",
|
||||||
|
|
|
@ -3241,6 +3241,7 @@ void CommandController::initialize(Settings &, Paths &paths)
|
||||||
this->registerCommand("/shoutout", &commands::sendShoutout);
|
this->registerCommand("/shoutout", &commands::sendShoutout);
|
||||||
|
|
||||||
this->registerCommand("/c2-set-logging-rules", &commands::setLoggingRules);
|
this->registerCommand("/c2-set-logging-rules", &commands::setLoggingRules);
|
||||||
|
this->registerCommand("/c2-theme-autoreload", &commands::toggleThemeReload);
|
||||||
}
|
}
|
||||||
|
|
||||||
void CommandController::save()
|
void CommandController::save()
|
||||||
|
|
|
@ -1,14 +1,18 @@
|
||||||
#include "controllers/commands/builtin/chatterino/Debugging.hpp"
|
#include "controllers/commands/builtin/chatterino/Debugging.hpp"
|
||||||
|
|
||||||
#include "common/Channel.hpp"
|
#include "common/Channel.hpp"
|
||||||
|
#include "common/Literals.hpp"
|
||||||
#include "controllers/commands/CommandContext.hpp"
|
#include "controllers/commands/CommandContext.hpp"
|
||||||
#include "messages/MessageBuilder.hpp"
|
#include "messages/MessageBuilder.hpp"
|
||||||
|
#include "singletons/Theme.hpp"
|
||||||
|
|
||||||
#include <QLoggingCategory>
|
#include <QLoggingCategory>
|
||||||
#include <QString>
|
#include <QString>
|
||||||
|
|
||||||
namespace chatterino::commands {
|
namespace chatterino::commands {
|
||||||
|
|
||||||
|
using namespace literals;
|
||||||
|
|
||||||
QString setLoggingRules(const CommandContext &ctx)
|
QString setLoggingRules(const CommandContext &ctx)
|
||||||
{
|
{
|
||||||
if (ctx.words.size() < 2)
|
if (ctx.words.size() < 2)
|
||||||
|
@ -42,4 +46,21 @@ QString setLoggingRules(const CommandContext &ctx)
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QString toggleThemeReload(const CommandContext &ctx)
|
||||||
|
{
|
||||||
|
if (getTheme()->isAutoReloading())
|
||||||
|
{
|
||||||
|
getTheme()->setAutoReload(false);
|
||||||
|
ctx.channel->addMessage(
|
||||||
|
makeSystemMessage(u"Disabled theme auto reloading."_s));
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
getTheme()->setAutoReload(true);
|
||||||
|
ctx.channel->addMessage(
|
||||||
|
makeSystemMessage(u"Auto reloading theme every %1 ms."_s.arg(
|
||||||
|
Theme::AUTO_RELOAD_INTERVAL_MS)));
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace chatterino::commands
|
} // namespace chatterino::commands
|
||||||
|
|
|
@ -12,4 +12,6 @@ namespace chatterino::commands {
|
||||||
|
|
||||||
QString setLoggingRules(const CommandContext &ctx);
|
QString setLoggingRules(const CommandContext &ctx);
|
||||||
|
|
||||||
|
QString toggleThemeReload(const CommandContext &ctx);
|
||||||
|
|
||||||
} // namespace chatterino::commands
|
} // namespace chatterino::commands
|
||||||
|
|
|
@ -2,12 +2,14 @@
|
||||||
#include "singletons/Theme.hpp"
|
#include "singletons/Theme.hpp"
|
||||||
|
|
||||||
#include "Application.hpp"
|
#include "Application.hpp"
|
||||||
|
#include "common/Literals.hpp"
|
||||||
#include "common/QLogging.hpp"
|
#include "common/QLogging.hpp"
|
||||||
#include "singletons/Paths.hpp"
|
#include "singletons/Paths.hpp"
|
||||||
#include "singletons/Resources.hpp"
|
#include "singletons/Resources.hpp"
|
||||||
|
|
||||||
#include <QColor>
|
#include <QColor>
|
||||||
#include <QDir>
|
#include <QDir>
|
||||||
|
#include <QElapsedTimer>
|
||||||
#include <QFile>
|
#include <QFile>
|
||||||
#include <QJsonDocument>
|
#include <QJsonDocument>
|
||||||
#include <QSet>
|
#include <QSet>
|
||||||
|
@ -17,8 +19,9 @@
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
using namespace chatterino;
|
using namespace chatterino;
|
||||||
|
using namespace literals;
|
||||||
|
|
||||||
void parseInto(const QJsonObject &obj, const QLatin1String &key, QColor &color)
|
void parseInto(const QJsonObject &obj, QLatin1String key, QColor &color)
|
||||||
{
|
{
|
||||||
const auto &jsonValue = obj[key];
|
const auto &jsonValue = obj[key];
|
||||||
if (!jsonValue.isString()) [[unlikely]]
|
if (!jsonValue.isString()) [[unlikely]]
|
||||||
|
@ -40,8 +43,9 @@ void parseInto(const QJsonObject &obj, const QLatin1String &key, QColor &color)
|
||||||
}
|
}
|
||||||
|
|
||||||
// NOLINTBEGIN(cppcoreguidelines-macro-usage)
|
// NOLINTBEGIN(cppcoreguidelines-macro-usage)
|
||||||
|
#define _c2StringLit(s, ty) s##ty
|
||||||
#define parseColor(to, from, key) \
|
#define parseColor(to, from, key) \
|
||||||
parseInto(from, QLatin1String(#key), (to).from.key)
|
parseInto(from, _c2StringLit(#key, _L1), (to).from.key)
|
||||||
// NOLINTEND(cppcoreguidelines-macro-usage)
|
// NOLINTEND(cppcoreguidelines-macro-usage)
|
||||||
|
|
||||||
void parseWindow(const QJsonObject &window, chatterino::Theme &theme)
|
void parseWindow(const QJsonObject &window, chatterino::Theme &theme)
|
||||||
|
@ -52,32 +56,32 @@ void parseWindow(const QJsonObject &window, chatterino::Theme &theme)
|
||||||
|
|
||||||
void parseTabs(const QJsonObject &tabs, chatterino::Theme &theme)
|
void parseTabs(const QJsonObject &tabs, chatterino::Theme &theme)
|
||||||
{
|
{
|
||||||
const auto parseTabColors = [](auto json, auto &tab) {
|
const auto parseTabColors = [](const auto &json, auto &tab) {
|
||||||
parseInto(json, QLatin1String("text"), tab.text);
|
parseInto(json, "text"_L1, tab.text);
|
||||||
{
|
{
|
||||||
const auto backgrounds = json["backgrounds"].toObject();
|
const auto backgrounds = json["backgrounds"_L1].toObject();
|
||||||
parseColor(tab, backgrounds, regular);
|
parseColor(tab, backgrounds, regular);
|
||||||
parseColor(tab, backgrounds, hover);
|
parseColor(tab, backgrounds, hover);
|
||||||
parseColor(tab, backgrounds, unfocused);
|
parseColor(tab, backgrounds, unfocused);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
const auto line = json["line"].toObject();
|
const auto line = json["line"_L1].toObject();
|
||||||
parseColor(tab, line, regular);
|
parseColor(tab, line, regular);
|
||||||
parseColor(tab, line, hover);
|
parseColor(tab, line, hover);
|
||||||
parseColor(tab, line, unfocused);
|
parseColor(tab, line, unfocused);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
parseColor(theme, tabs, dividerLine);
|
parseColor(theme, tabs, dividerLine);
|
||||||
parseTabColors(tabs["regular"].toObject(), theme.tabs.regular);
|
parseTabColors(tabs["regular"_L1].toObject(), theme.tabs.regular);
|
||||||
parseTabColors(tabs["newMessage"].toObject(), theme.tabs.newMessage);
|
parseTabColors(tabs["newMessage"_L1].toObject(), theme.tabs.newMessage);
|
||||||
parseTabColors(tabs["highlighted"].toObject(), theme.tabs.highlighted);
|
parseTabColors(tabs["highlighted"_L1].toObject(), theme.tabs.highlighted);
|
||||||
parseTabColors(tabs["selected"].toObject(), theme.tabs.selected);
|
parseTabColors(tabs["selected"_L1].toObject(), theme.tabs.selected);
|
||||||
}
|
}
|
||||||
|
|
||||||
void parseMessages(const QJsonObject &messages, chatterino::Theme &theme)
|
void parseMessages(const QJsonObject &messages, chatterino::Theme &theme)
|
||||||
{
|
{
|
||||||
{
|
{
|
||||||
const auto textColors = messages["textColors"].toObject();
|
const auto textColors = messages["textColors"_L1].toObject();
|
||||||
parseColor(theme.messages, textColors, regular);
|
parseColor(theme.messages, textColors, regular);
|
||||||
parseColor(theme.messages, textColors, caret);
|
parseColor(theme.messages, textColors, caret);
|
||||||
parseColor(theme.messages, textColors, link);
|
parseColor(theme.messages, textColors, link);
|
||||||
|
@ -85,7 +89,7 @@ void parseMessages(const QJsonObject &messages, chatterino::Theme &theme)
|
||||||
parseColor(theme.messages, textColors, chatPlaceholder);
|
parseColor(theme.messages, textColors, chatPlaceholder);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
const auto backgrounds = messages["backgrounds"].toObject();
|
const auto backgrounds = messages["backgrounds"_L1].toObject();
|
||||||
parseColor(theme.messages, backgrounds, regular);
|
parseColor(theme.messages, backgrounds, regular);
|
||||||
parseColor(theme.messages, backgrounds, alternate);
|
parseColor(theme.messages, backgrounds, alternate);
|
||||||
}
|
}
|
||||||
|
@ -114,7 +118,7 @@ void parseSplits(const QJsonObject &splits, chatterino::Theme &theme)
|
||||||
parseColor(theme, splits, resizeHandleBackground);
|
parseColor(theme, splits, resizeHandleBackground);
|
||||||
|
|
||||||
{
|
{
|
||||||
const auto header = splits["header"].toObject();
|
const auto header = splits["header"_L1].toObject();
|
||||||
parseColor(theme.splits, header, border);
|
parseColor(theme.splits, header, border);
|
||||||
parseColor(theme.splits, header, focusedBorder);
|
parseColor(theme.splits, header, focusedBorder);
|
||||||
parseColor(theme.splits, header, background);
|
parseColor(theme.splits, header, background);
|
||||||
|
@ -123,7 +127,7 @@ void parseSplits(const QJsonObject &splits, chatterino::Theme &theme)
|
||||||
parseColor(theme.splits, header, focusedText);
|
parseColor(theme.splits, header, focusedText);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
const auto input = splits["input"].toObject();
|
const auto input = splits["input"_L1].toObject();
|
||||||
parseColor(theme.splits, input, background);
|
parseColor(theme.splits, input, background);
|
||||||
parseColor(theme.splits, input, text);
|
parseColor(theme.splits, input, text);
|
||||||
}
|
}
|
||||||
|
@ -131,32 +135,26 @@ void parseSplits(const QJsonObject &splits, chatterino::Theme &theme)
|
||||||
|
|
||||||
void parseColors(const QJsonObject &root, chatterino::Theme &theme)
|
void parseColors(const QJsonObject &root, chatterino::Theme &theme)
|
||||||
{
|
{
|
||||||
const auto colors = root["colors"].toObject();
|
const auto colors = root["colors"_L1].toObject();
|
||||||
|
|
||||||
parseInto(colors, QLatin1String("accent"), theme.accent);
|
parseInto(colors, "accent"_L1, theme.accent);
|
||||||
|
|
||||||
parseWindow(colors["window"].toObject(), theme);
|
parseWindow(colors["window"_L1].toObject(), theme);
|
||||||
parseTabs(colors["tabs"].toObject(), theme);
|
parseTabs(colors["tabs"_L1].toObject(), theme);
|
||||||
parseMessages(colors["messages"].toObject(), theme);
|
parseMessages(colors["messages"_L1].toObject(), theme);
|
||||||
parseScrollbars(colors["scrollbars"].toObject(), theme);
|
parseScrollbars(colors["scrollbars"_L1].toObject(), theme);
|
||||||
parseSplits(colors["splits"].toObject(), theme);
|
parseSplits(colors["splits"_L1].toObject(), theme);
|
||||||
}
|
}
|
||||||
#undef parseColor
|
#undef parseColor
|
||||||
|
#undef _c2StringLit
|
||||||
|
|
||||||
/**
|
std::optional<QJsonObject> loadThemeFromPath(const QString &path)
|
||||||
* Load the given theme descriptor from its path
|
|
||||||
*
|
|
||||||
* Returns a JSON object containing theme data if the theme is valid, otherwise nullopt
|
|
||||||
*
|
|
||||||
* NOTE: No theme validation is done by this function
|
|
||||||
**/
|
|
||||||
std::optional<QJsonObject> loadTheme(const ThemeDescriptor &theme)
|
|
||||||
{
|
{
|
||||||
QFile file(theme.path);
|
QFile file(path);
|
||||||
if (!file.open(QFile::ReadOnly))
|
if (!file.open(QFile::ReadOnly))
|
||||||
{
|
{
|
||||||
qCWarning(chatterinoTheme)
|
qCWarning(chatterinoTheme)
|
||||||
<< "Failed to open" << file.fileName() << "at" << theme.path;
|
<< "Failed to open" << file.fileName() << "at" << path;
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -174,6 +172,18 @@ std::optional<QJsonObject> loadTheme(const ThemeDescriptor &theme)
|
||||||
return json.object();
|
return json.object();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Load the given theme descriptor from its path
|
||||||
|
*
|
||||||
|
* Returns a JSON object containing theme data if the theme is valid, otherwise nullopt
|
||||||
|
*
|
||||||
|
* NOTE: No theme validation is done by this function
|
||||||
|
**/
|
||||||
|
std::optional<QJsonObject> loadTheme(const ThemeDescriptor &theme)
|
||||||
|
{
|
||||||
|
return loadThemeFromPath(theme.path);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace chatterino {
|
namespace chatterino {
|
||||||
|
@ -227,7 +237,12 @@ void Theme::update()
|
||||||
{
|
{
|
||||||
auto oTheme = this->findThemeByKey(this->themeName);
|
auto oTheme = this->findThemeByKey(this->themeName);
|
||||||
|
|
||||||
|
constexpr const double nsToMs = 1.0 / 1000000.0;
|
||||||
|
QElapsedTimer timer;
|
||||||
|
timer.start();
|
||||||
|
|
||||||
std::optional<QJsonObject> themeJSON;
|
std::optional<QJsonObject> themeJSON;
|
||||||
|
QString themePath;
|
||||||
if (!oTheme)
|
if (!oTheme)
|
||||||
{
|
{
|
||||||
qCWarning(chatterinoTheme)
|
qCWarning(chatterinoTheme)
|
||||||
|
@ -235,12 +250,14 @@ void Theme::update()
|
||||||
<< "not found, falling back to the fallback theme";
|
<< "not found, falling back to the fallback theme";
|
||||||
|
|
||||||
themeJSON = loadTheme(fallbackTheme);
|
themeJSON = loadTheme(fallbackTheme);
|
||||||
|
themePath = fallbackTheme.path;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
const auto &theme = *oTheme;
|
const auto &theme = *oTheme;
|
||||||
|
|
||||||
themeJSON = loadTheme(theme);
|
themeJSON = loadTheme(theme);
|
||||||
|
themePath = theme.path;
|
||||||
|
|
||||||
if (!themeJSON)
|
if (!themeJSON)
|
||||||
{
|
{
|
||||||
|
@ -250,8 +267,10 @@ void Theme::update()
|
||||||
|
|
||||||
// Parsing the theme failed, fall back
|
// Parsing the theme failed, fall back
|
||||||
themeJSON = loadTheme(fallbackTheme);
|
themeJSON = loadTheme(fallbackTheme);
|
||||||
|
themePath = fallbackTheme.path;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
auto loadTs = double(timer.nsecsElapsed()) * nsToMs;
|
||||||
|
|
||||||
if (!themeJSON)
|
if (!themeJSON)
|
||||||
{
|
{
|
||||||
|
@ -260,9 +279,29 @@ void Theme::update()
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this->isAutoReloading() && this->currentThemeJson_ == *themeJSON)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
this->parseFrom(*themeJSON);
|
this->parseFrom(*themeJSON);
|
||||||
|
this->currentThemePath_ = themePath;
|
||||||
|
|
||||||
|
auto parseTs = double(timer.nsecsElapsed()) * nsToMs;
|
||||||
|
|
||||||
this->updated.invoke();
|
this->updated.invoke();
|
||||||
|
auto updateTs = double(timer.nsecsElapsed()) * nsToMs;
|
||||||
|
qCDebug(chatterinoTheme).nospace().noquote()
|
||||||
|
<< "Updated theme in " << QString::number(updateTs, 'f', 2)
|
||||||
|
<< "ms (load: " << QString::number(loadTs, 'f', 2)
|
||||||
|
<< "ms, parse: " << QString::number(parseTs - loadTs, 'f', 2)
|
||||||
|
<< "ms, update: " << QString::number(updateTs - parseTs, 'f', 2)
|
||||||
|
<< "ms)";
|
||||||
|
|
||||||
|
if (this->isAutoReloading())
|
||||||
|
{
|
||||||
|
this->currentThemeJson_ = *themeJSON;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<std::pair<QString, QVariant>> Theme::availableThemes() const
|
std::vector<std::pair<QString, QVariant>> Theme::availableThemes() const
|
||||||
|
@ -341,15 +380,20 @@ void Theme::parseFrom(const QJsonObject &root)
|
||||||
parseColors(root, *this);
|
parseColors(root, *this);
|
||||||
|
|
||||||
this->isLight_ =
|
this->isLight_ =
|
||||||
root["metadata"]["iconTheme"].toString() == QStringLiteral("dark");
|
root["metadata"_L1]["iconTheme"_L1].toString() == u"dark"_s;
|
||||||
|
|
||||||
this->splits.input.styleSheet =
|
this->splits.input.styleSheet = uR"(
|
||||||
"background:" + this->splits.input.background.name() + ";" +
|
background: %1;
|
||||||
"border:" + this->tabs.selected.backgrounds.regular.name() + ";" +
|
border: %2;
|
||||||
"color:" + this->messages.textColors.regular.name() + ";" +
|
color: %3;
|
||||||
"selection-background-color:" +
|
selection-background-color: %4;
|
||||||
(this->isLightTheme() ? "#68B1FF"
|
)"_s.arg(
|
||||||
: this->tabs.selected.backgrounds.regular.name());
|
this->splits.input.background.name(QColor::HexArgb),
|
||||||
|
this->tabs.selected.backgrounds.regular.name(QColor::HexArgb),
|
||||||
|
this->messages.textColors.regular.name(QColor::HexArgb),
|
||||||
|
this->isLightTheme()
|
||||||
|
? u"#68B1FF"_s
|
||||||
|
: this->tabs.selected.backgrounds.regular.name(QColor::HexArgb));
|
||||||
|
|
||||||
// Usercard buttons
|
// Usercard buttons
|
||||||
if (this->isLightTheme())
|
if (this->isLightTheme())
|
||||||
|
@ -364,6 +408,35 @@ void Theme::parseFrom(const QJsonObject &root)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool Theme::isAutoReloading() const
|
||||||
|
{
|
||||||
|
return this->themeReloadTimer_ != nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Theme::setAutoReload(bool autoReload)
|
||||||
|
{
|
||||||
|
if (autoReload == this->isAutoReloading())
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!autoReload)
|
||||||
|
{
|
||||||
|
this->themeReloadTimer_.reset();
|
||||||
|
this->currentThemeJson_ = {};
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this->themeReloadTimer_ = std::make_unique<QTimer>();
|
||||||
|
QObject::connect(this->themeReloadTimer_.get(), &QTimer::timeout, [this]() {
|
||||||
|
this->update();
|
||||||
|
});
|
||||||
|
this->themeReloadTimer_->setInterval(Theme::AUTO_RELOAD_INTERVAL_MS);
|
||||||
|
this->themeReloadTimer_->start();
|
||||||
|
|
||||||
|
qCDebug(chatterinoTheme) << "Enabled theme watcher";
|
||||||
|
}
|
||||||
|
|
||||||
void Theme::normalizeColor(QColor &color) const
|
void Theme::normalizeColor(QColor &color) const
|
||||||
{
|
{
|
||||||
if (this->isLightTheme())
|
if (this->isLightTheme())
|
||||||
|
|
|
@ -9,8 +9,10 @@
|
||||||
#include <QJsonObject>
|
#include <QJsonObject>
|
||||||
#include <QPixmap>
|
#include <QPixmap>
|
||||||
#include <QString>
|
#include <QString>
|
||||||
|
#include <QTimer>
|
||||||
#include <QVariant>
|
#include <QVariant>
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
#include <optional>
|
#include <optional>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
|
@ -39,6 +41,8 @@ public:
|
||||||
// The built in theme that will be used if some theme parsing fails
|
// The built in theme that will be used if some theme parsing fails
|
||||||
static const ThemeDescriptor fallbackTheme;
|
static const ThemeDescriptor fallbackTheme;
|
||||||
|
|
||||||
|
static const int AUTO_RELOAD_INTERVAL_MS = 500;
|
||||||
|
|
||||||
void initialize(Settings &settings, Paths &paths) final;
|
void initialize(Settings &settings, Paths &paths) final;
|
||||||
|
|
||||||
bool isLightTheme() const;
|
bool isLightTheme() const;
|
||||||
|
@ -138,6 +142,9 @@ public:
|
||||||
void normalizeColor(QColor &color) const;
|
void normalizeColor(QColor &color) const;
|
||||||
void update();
|
void update();
|
||||||
|
|
||||||
|
bool isAutoReloading() const;
|
||||||
|
void setAutoReload(bool autoReload);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return a list of available themes
|
* Return a list of available themes
|
||||||
**/
|
**/
|
||||||
|
@ -152,6 +159,11 @@ private:
|
||||||
|
|
||||||
std::vector<ThemeDescriptor> availableThemes_;
|
std::vector<ThemeDescriptor> availableThemes_;
|
||||||
|
|
||||||
|
QString currentThemePath_;
|
||||||
|
std::unique_ptr<QTimer> themeReloadTimer_;
|
||||||
|
// This will only be populated when auto-reloading themes
|
||||||
|
QJsonObject currentThemeJson_;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Figure out which themes are available in the Themes directory
|
* Figure out which themes are available in the Themes directory
|
||||||
*
|
*
|
||||||
|
|
Loading…
Reference in a new issue