2016-11-06 04:29:12 +01:00
|
|
|
#include "CustomTheme.h"
|
|
|
|
#include <QDir>
|
|
|
|
#include <Json.h>
|
|
|
|
#include <FileSystem.h>
|
|
|
|
|
|
|
|
const char * themeFile = "theme.json";
|
|
|
|
const char * styleFile = "themeStyle.css";
|
|
|
|
|
|
|
|
static bool readThemeJson(const QString &path, QPalette &palette, double &fadeAmount, QColor &fadeColor, QString &name, QString &widgets)
|
|
|
|
{
|
|
|
|
QFileInfo pathInfo(path);
|
|
|
|
if(pathInfo.exists() && pathInfo.isFile())
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
auto doc = Json::requireDocument(path, "Theme JSON file");
|
|
|
|
const QJsonObject root = doc.object();
|
|
|
|
name = Json::requireString(root, "name", "Theme name");
|
|
|
|
widgets = Json::requireString(root, "widgets", "Qt widget theme");
|
|
|
|
auto colorsRoot = Json::requireObject(root, "colors", "colors object");
|
|
|
|
auto readColor = [&](QString colorName) -> QColor
|
|
|
|
{
|
|
|
|
auto colorValue = Json::ensureString(colorsRoot, colorName, QString());
|
|
|
|
if(!colorValue.isEmpty())
|
|
|
|
{
|
|
|
|
QColor color(colorValue);
|
|
|
|
if(!color.isValid())
|
|
|
|
{
|
|
|
|
qWarning() << "Color value" << colorValue << "for" << colorName << "was not recognized.";
|
|
|
|
return QColor();
|
|
|
|
}
|
|
|
|
return color;
|
|
|
|
}
|
|
|
|
return QColor();
|
|
|
|
};
|
|
|
|
auto readAndSetColor = [&](QPalette::ColorRole role, QString colorName)
|
|
|
|
{
|
|
|
|
auto color = readColor(colorName);
|
|
|
|
if(color.isValid())
|
|
|
|
{
|
|
|
|
palette.setColor(role, color);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
qDebug() << "Color value for" << colorName << "was not present.";
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// palette
|
|
|
|
readAndSetColor(QPalette::Window, "Window");
|
|
|
|
readAndSetColor(QPalette::WindowText, "WindowText");
|
|
|
|
readAndSetColor(QPalette::Base, "Base");
|
|
|
|
readAndSetColor(QPalette::AlternateBase, "AlternateBase");
|
|
|
|
readAndSetColor(QPalette::ToolTipBase, "ToolTipBase");
|
|
|
|
readAndSetColor(QPalette::ToolTipText, "ToolTipText");
|
|
|
|
readAndSetColor(QPalette::Text, "Text");
|
|
|
|
readAndSetColor(QPalette::Button, "Button");
|
|
|
|
readAndSetColor(QPalette::ButtonText, "ButtonText");
|
|
|
|
readAndSetColor(QPalette::BrightText, "BrightText");
|
|
|
|
readAndSetColor(QPalette::Link, "Link");
|
|
|
|
readAndSetColor(QPalette::Highlight, "Highlight");
|
|
|
|
readAndSetColor(QPalette::HighlightedText, "HighlightedText");
|
|
|
|
|
|
|
|
//fade
|
|
|
|
fadeColor = readColor("fadeColor");
|
|
|
|
fadeAmount = Json::ensureDouble(colorsRoot, "fadeAmount", 0.5, "fade amount");
|
|
|
|
|
|
|
|
}
|
2018-05-19 19:18:26 -04:00
|
|
|
catch (const Exception &e)
|
2016-11-06 04:29:12 +01:00
|
|
|
{
|
|
|
|
qWarning() << "Couldn't load theme json: " << e.cause();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
qDebug() << "No theme json present.";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool writeThemeJson(const QString &path, const QPalette &palette, double fadeAmount, QColor fadeColor, QString name, QString widgets)
|
|
|
|
{
|
|
|
|
QJsonObject rootObj;
|
|
|
|
rootObj.insert("name", name);
|
|
|
|
rootObj.insert("widgets", widgets);
|
|
|
|
|
|
|
|
QJsonObject colorsObj;
|
|
|
|
auto insertColor = [&](QPalette::ColorRole role, QString colorName)
|
|
|
|
{
|
|
|
|
colorsObj.insert(colorName, palette.color(role).name());
|
|
|
|
};
|
|
|
|
|
|
|
|
// palette
|
|
|
|
insertColor(QPalette::Window, "Window");
|
|
|
|
insertColor(QPalette::WindowText, "WindowText");
|
|
|
|
insertColor(QPalette::Base, "Base");
|
|
|
|
insertColor(QPalette::AlternateBase, "AlternateBase");
|
|
|
|
insertColor(QPalette::ToolTipBase, "ToolTipBase");
|
|
|
|
insertColor(QPalette::ToolTipText, "ToolTipText");
|
|
|
|
insertColor(QPalette::Text, "Text");
|
|
|
|
insertColor(QPalette::Button, "Button");
|
|
|
|
insertColor(QPalette::ButtonText, "ButtonText");
|
|
|
|
insertColor(QPalette::BrightText, "BrightText");
|
|
|
|
insertColor(QPalette::Link, "Link");
|
|
|
|
insertColor(QPalette::Highlight, "Highlight");
|
|
|
|
insertColor(QPalette::HighlightedText, "HighlightedText");
|
|
|
|
|
|
|
|
// fade
|
|
|
|
colorsObj.insert("fadeColor", fadeColor.name());
|
|
|
|
colorsObj.insert("fadeAmount", fadeAmount);
|
|
|
|
|
|
|
|
rootObj.insert("colors", colorsObj);
|
|
|
|
try
|
|
|
|
{
|
|
|
|
Json::write(rootObj, path);
|
|
|
|
return true;
|
|
|
|
}
|
2018-05-19 19:18:26 -04:00
|
|
|
catch (const Exception &e)
|
2016-11-06 04:29:12 +01:00
|
|
|
{
|
|
|
|
qWarning() << "Failed to write theme json to" << path;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CustomTheme::CustomTheme(ITheme* baseTheme, QString folder)
|
|
|
|
{
|
|
|
|
m_id = folder;
|
|
|
|
QString path = FS::PathCombine("themes", m_id);
|
2016-11-06 05:48:52 +01:00
|
|
|
QString pathResources = FS::PathCombine("themes", m_id, "resources");
|
2016-11-06 04:29:12 +01:00
|
|
|
|
|
|
|
qDebug() << "Loading theme" << m_id;
|
|
|
|
|
2016-11-06 05:48:52 +01:00
|
|
|
if(!FS::ensureFolderPathExists(path) || !FS::ensureFolderPathExists(pathResources))
|
2016-11-06 04:29:12 +01:00
|
|
|
{
|
|
|
|
qWarning() << "couldn't create folder for theme!";
|
|
|
|
m_palette = baseTheme->colorScheme();
|
|
|
|
m_styleSheet = baseTheme->appStyleSheet();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto themeFilePath = FS::PathCombine(path, themeFile);
|
|
|
|
|
|
|
|
m_palette = baseTheme->colorScheme();
|
|
|
|
if (!readThemeJson(themeFilePath, m_palette, m_fadeAmount, m_fadeColor, m_name, m_widgets))
|
|
|
|
{
|
|
|
|
m_name = "Custom";
|
|
|
|
m_palette = baseTheme->colorScheme();
|
|
|
|
m_fadeColor = baseTheme->fadeColor();
|
|
|
|
m_fadeAmount = baseTheme->fadeAmount();
|
|
|
|
m_widgets = baseTheme->qtTheme();
|
|
|
|
|
|
|
|
QFileInfo info(themeFilePath);
|
|
|
|
if(!info.exists())
|
|
|
|
{
|
|
|
|
writeThemeJson(themeFilePath, m_palette, m_fadeAmount, m_fadeColor, "Custom", m_widgets);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_palette = fadeInactive(m_palette, m_fadeAmount, m_fadeColor);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto cssFilePath = FS::PathCombine(path, styleFile);
|
|
|
|
QFileInfo info (cssFilePath);
|
|
|
|
if(info.isFile())
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
// TODO: validate css?
|
|
|
|
m_styleSheet = QString::fromUtf8(FS::read(cssFilePath));
|
|
|
|
}
|
2018-05-19 19:18:26 -04:00
|
|
|
catch (const Exception &e)
|
2016-11-06 04:29:12 +01:00
|
|
|
{
|
|
|
|
qWarning() << "Couldn't load css:" << e.cause() << "from" << cssFilePath;
|
|
|
|
m_styleSheet = baseTheme->appStyleSheet();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
qDebug() << "No theme css present.";
|
|
|
|
m_styleSheet = baseTheme->appStyleSheet();
|
|
|
|
try
|
|
|
|
{
|
|
|
|
FS::write(cssFilePath, m_styleSheet.toUtf8());
|
|
|
|
}
|
2018-05-19 19:18:26 -04:00
|
|
|
catch (const Exception &e)
|
2016-11-06 04:29:12 +01:00
|
|
|
{
|
|
|
|
qWarning() << "Couldn't write css:" << e.cause() << "to" << cssFilePath;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-06 05:48:52 +01:00
|
|
|
QStringList CustomTheme::searchPaths()
|
|
|
|
{
|
|
|
|
return { FS::PathCombine("themes", m_id, "resources") };
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-11-06 04:29:12 +01:00
|
|
|
QString CustomTheme::id()
|
|
|
|
{
|
|
|
|
return m_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString CustomTheme::name()
|
|
|
|
{
|
|
|
|
return m_name;
|
|
|
|
}
|
|
|
|
|
2017-01-15 22:56:03 +01:00
|
|
|
bool CustomTheme::hasColorScheme()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-11-06 04:29:12 +01:00
|
|
|
QPalette CustomTheme::colorScheme()
|
|
|
|
{
|
|
|
|
return m_palette;
|
|
|
|
}
|
|
|
|
|
2017-01-15 22:56:03 +01:00
|
|
|
bool CustomTheme::hasStyleSheet()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-11-06 04:29:12 +01:00
|
|
|
QString CustomTheme::appStyleSheet()
|
|
|
|
{
|
|
|
|
return m_styleSheet;
|
|
|
|
}
|
|
|
|
|
|
|
|
double CustomTheme::fadeAmount()
|
|
|
|
{
|
|
|
|
return m_fadeAmount;
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor CustomTheme::fadeColor()
|
|
|
|
{
|
|
|
|
return m_fadeColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString CustomTheme::qtTheme()
|
|
|
|
{
|
|
|
|
return m_widgets;
|
|
|
|
}
|