mirror-chatterino2/src/singletons/Theme.cpp

271 lines
8 KiB
C++
Raw Normal View History

2017-01-18 01:04:54 +01:00
2018-07-15 14:03:41 +02:00
#include "singletons/Theme.hpp"
2018-11-25 15:02:48 +01:00
#include "Application.hpp"
#include "common/QLogging.hpp"
#include "singletons/Resources.hpp"
2017-01-18 04:52:47 +01:00
#include <QColor>
#include <QFile>
#include <QJsonDocument>
#include <QJsonObject>
#include <QSet>
2016-12-30 19:16:48 +01:00
#include <cmath>
namespace {
void parseInto(const QJsonObject &obj, const QLatin1String &key, QColor &color)
{
const auto &jsonValue = obj[key];
if (!jsonValue.isString()) [[unlikely]]
{
qCWarning(chatterinoTheme) << key
<< "was expected but not found in the "
"current theme - using previous value.";
return;
}
QColor parsed = {jsonValue.toString()};
if (!parsed.isValid()) [[unlikely]]
{
qCWarning(chatterinoTheme).nospace()
<< "While parsing " << key << ": '" << jsonValue.toString()
<< "' isn't a valid color.";
return;
}
color = parsed;
}
// NOLINTBEGIN(cppcoreguidelines-macro-usage)
#define parseColor(to, from, key) \
parseInto(from, QLatin1String(#key), (to).from.key)
// NOLINTEND(cppcoreguidelines-macro-usage)
void parseWindow(const QJsonObject &window, chatterino::Theme &theme)
{
parseColor(theme, window, background);
parseColor(theme, window, text);
}
void parseTabs(const QJsonObject &tabs, chatterino::Theme &theme)
{
const auto parseTabColors = [](auto json, auto &tab) {
parseInto(json, QLatin1String("text"), tab.text);
{
const auto backgrounds = json["backgrounds"].toObject();
parseColor(tab, backgrounds, regular);
parseColor(tab, backgrounds, hover);
parseColor(tab, backgrounds, unfocused);
}
{
const auto line = json["line"].toObject();
parseColor(tab, line, regular);
parseColor(tab, line, hover);
parseColor(tab, line, unfocused);
}
};
parseColor(theme, tabs, dividerLine);
parseTabColors(tabs["regular"].toObject(), theme.tabs.regular);
parseTabColors(tabs["newMessage"].toObject(), theme.tabs.newMessage);
parseTabColors(tabs["highlighted"].toObject(), theme.tabs.highlighted);
parseTabColors(tabs["selected"].toObject(), theme.tabs.selected);
}
void parseMessages(const QJsonObject &messages, chatterino::Theme &theme)
{
{
const auto textColors = messages["textColors"].toObject();
parseColor(theme.messages, textColors, regular);
parseColor(theme.messages, textColors, caret);
parseColor(theme.messages, textColors, link);
parseColor(theme.messages, textColors, system);
parseColor(theme.messages, textColors, chatPlaceholder);
}
{
const auto backgrounds = messages["backgrounds"].toObject();
parseColor(theme.messages, backgrounds, regular);
parseColor(theme.messages, backgrounds, alternate);
}
parseColor(theme, messages, disabled);
parseColor(theme, messages, selection);
parseColor(theme, messages, highlightAnimationStart);
parseColor(theme, messages, highlightAnimationEnd);
}
void parseScrollbars(const QJsonObject &scrollbars, chatterino::Theme &theme)
{
parseColor(theme, scrollbars, background);
parseColor(theme, scrollbars, thumb);
parseColor(theme, scrollbars, thumbSelected);
}
void parseSplits(const QJsonObject &splits, chatterino::Theme &theme)
{
parseColor(theme, splits, messageSeperator);
parseColor(theme, splits, background);
parseColor(theme, splits, dropPreview);
parseColor(theme, splits, dropPreviewBorder);
parseColor(theme, splits, dropTargetRect);
parseColor(theme, splits, dropTargetRectBorder);
parseColor(theme, splits, resizeHandle);
parseColor(theme, splits, resizeHandleBackground);
{
const auto header = splits["header"].toObject();
parseColor(theme.splits, header, border);
parseColor(theme.splits, header, focusedBorder);
parseColor(theme.splits, header, background);
parseColor(theme.splits, header, focusedBackground);
parseColor(theme.splits, header, text);
parseColor(theme.splits, header, focusedText);
}
{
const auto input = splits["input"].toObject();
parseColor(theme.splits, input, background);
parseColor(theme.splits, input, text);
}
}
void parseColors(const QJsonObject &root, chatterino::Theme &theme)
{
const auto colors = root["colors"].toObject();
parseInto(colors, QLatin1String("accent"), theme.accent);
parseWindow(colors["window"].toObject(), theme);
parseTabs(colors["tabs"].toObject(), theme);
parseMessages(colors["messages"].toObject(), theme);
parseScrollbars(colors["scrollbars"].toObject(), theme);
parseSplits(colors["splits"].toObject(), theme);
}
#undef parseColor
QString getThemePath(const QString &name)
{
static QSet<QString> knownThemes = {"White", "Light", "Dark", "Black"};
if (knownThemes.contains(name))
{
return QStringLiteral(":/themes/%1.json").arg(name);
}
return name;
}
} // namespace
2017-04-14 17:52:22 +02:00
namespace chatterino {
2017-01-18 21:30:23 +01:00
bool Theme::isLightTheme() const
{
return this->isLight_;
}
2018-06-28 20:03:04 +02:00
Theme::Theme()
{
this->update();
this->themeName.connectSimple(
[this](auto) {
this->update();
},
false);
2017-02-02 01:23:26 +01:00
}
void Theme::update()
{
this->parse();
this->updated.invoke();
}
void Theme::parse()
2016-12-30 19:16:48 +01:00
{
QFile file(getThemePath(this->themeName));
if (!file.open(QFile::ReadOnly))
2023-03-17 20:53:03 +01:00
{
qCWarning(chatterinoTheme) << "Failed to open" << file.fileName();
return;
}
QJsonParseError error{};
auto json = QJsonDocument::fromJson(file.readAll(), &error);
if (json.isNull())
2023-03-17 20:53:03 +01:00
{
qCWarning(chatterinoTheme) << "Failed to parse" << file.fileName()
<< "error:" << error.errorString();
return;
2023-03-17 20:53:03 +01:00
}
this->parseFrom(json.object());
}
void Theme::parseFrom(const QJsonObject &root)
{
parseColors(root, *this);
this->isLight_ =
root["metadata"]["iconTheme"].toString() == QStringLiteral("dark");
2018-01-02 02:15:11 +01:00
this->splits.input.styleSheet =
"background:" + this->splits.input.background.name() + ";" +
2023-03-17 20:53:03 +01:00
"border:" + this->tabs.selected.backgrounds.regular.name() + ";" +
"color:" + this->messages.textColors.regular.name() + ";" +
2018-06-07 17:43:21 +02:00
"selection-background-color:" +
(this->isLightTheme() ? "#68B1FF"
: this->tabs.selected.backgrounds.regular.name());
// Usercard buttons
if (this->isLightTheme())
{
this->buttons.copy = getResources().buttons.copyDark;
this->buttons.pin = getResources().buttons.pinDisabledDark;
}
else
{
this->buttons.copy = getResources().buttons.copyLight;
this->buttons.pin = getResources().buttons.pinDisabledLight;
}
2017-09-12 19:06:16 +02:00
}
2023-03-17 20:53:03 +01:00
void Theme::normalizeColor(QColor &color) const
2017-01-18 01:04:54 +01:00
{
2018-11-25 15:02:48 +01:00
if (this->isLightTheme())
2018-10-21 13:43:02 +02:00
{
if (color.lightnessF() > 0.5)
{
2018-06-01 14:20:46 +02:00
color.setHslF(color.hueF(), color.saturationF(), 0.5);
}
2018-08-06 21:17:03 +02:00
if (color.lightnessF() > 0.4 && color.hueF() > 0.1 &&
2018-10-21 13:43:02 +02:00
color.hueF() < 0.33333)
{
2018-08-06 21:17:03 +02:00
color.setHslF(color.hueF(), color.saturationF(),
color.lightnessF() - sin((color.hueF() - 0.1) /
(0.3333 - 0.1) * 3.14159) *
color.saturationF() * 0.4);
}
2018-10-21 13:43:02 +02:00
}
else
{
if (color.lightnessF() < 0.5)
{
2018-06-01 14:20:46 +02:00
color.setHslF(color.hueF(), color.saturationF(), 0.5);
2017-08-12 12:09:26 +02:00
}
2018-08-06 21:17:03 +02:00
if (color.lightnessF() < 0.6 && color.hueF() > 0.54444 &&
2018-10-21 13:43:02 +02:00
color.hueF() < 0.83333)
{
2018-01-12 23:09:05 +01:00
color.setHslF(
color.hueF(), color.saturationF(),
2018-08-06 21:17:03 +02:00
color.lightnessF() + sin((color.hueF() - 0.54444) /
(0.8333 - 0.54444) * 3.14159) *
2018-01-12 23:09:05 +01:00
color.saturationF() * 0.4);
2017-08-12 12:09:26 +02:00
}
}
2016-12-30 19:16:48 +01:00
}
2018-11-25 15:02:48 +01:00
Theme *getTheme()
{
return getApp()->themes;
}
2017-04-14 17:52:22 +02:00
} // namespace chatterino