/*
* Copyright (C) 2020-2022 Roy Qu (royqh1979@gmail.com)
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
#include "colorscheme.h"
#include
#include
#include
#include
#include
#include "utils.h"
#include "settings.h"
#include "qsynedit/constants.h"
ColorManager * pColorManager;
ColorScheme::ColorScheme()
{
}
PColorScheme ColorScheme::fromJson(const QJsonObject &json)
{
PColorScheme scheme = std::make_shared();
scheme->mItems.clear();
for (QString key:json.keys()) {
if (json[key].isObject()) {
scheme->mItems[key]=ColorSchemeItem::fromJson(json[key].toObject());
}
}
//backward compatibility (for config files version < 2.5)
if (!scheme->mItems.contains(SYNS_AttrReserveWord_Type)
&& scheme->mItems.contains(SYNS_AttrIdentifier)) {
scheme->mItems.insert(SYNS_AttrReserveWord_Type,
ColorSchemeItem::fromJson(json[SYNS_AttrReservedWord].toObject()));
}
return scheme;
}
void ColorScheme::toJson(QJsonObject &json)
{
for (QString key:mItems.keys()) {
PColorSchemeItem item = mItems[key];
if (item) {
QJsonObject itemObject;
item->toJson(itemObject);
json[key] = itemObject;
}
}
}
PColorScheme ColorScheme::load(const QString &filename)
{
QFile file(filename);
if (!file.open(QFile::ReadOnly)) {
qDebug()<();
mItems[name]=item;
}
QMap ColorScheme::items()
{
return mItems;
}
void ColorScheme::save(const QString &filename)
{
QFile file(filename);
QFileInfo info(filename);
info.dir().mkpath(info.dir().absolutePath());
if (!file.open(QFile::WriteOnly)) {
throw FileError(QObject::tr("Can't open file '%1' for write").arg(file.fileName()));
}
QJsonObject json;
toJson(json);
QJsonDocument doc(json);
QByteArray content = doc.toJson();
file.write(content);
}
bool ColorScheme::bundled() const
{
return mBundled;
}
void ColorScheme::setBundled(bool bundled)
{
mBundled = bundled;
}
bool ColorScheme::customed() const
{
return mCustomed;
}
void ColorScheme::setCustomed(bool customed)
{
mCustomed = customed;
}
QString ColorScheme::preferThemeType() const
{
return mPreferThemeType;
}
void ColorScheme::setPreferThemeType(const QString &preferThemeType)
{
mPreferThemeType = preferThemeType;
}
ColorSchemeItem::ColorSchemeItem():
mForeground(),
mBackground(),
mBold(false),
mItalic(false),
mUnderlined(false),
mStrikeout(false)
{
}
QColor ColorSchemeItem::foreground() const
{
return mForeground;
}
void ColorSchemeItem::setForeground(const QColor &foreground)
{
mForeground = foreground;
}
QColor ColorSchemeItem::background() const
{
return mBackground;
}
void ColorSchemeItem::setBackground(const QColor &background)
{
mBackground = background;
}
bool ColorSchemeItem::bold() const
{
return mBold;
}
void ColorSchemeItem::setBold(bool bold)
{
mBold = bold;
}
bool ColorSchemeItem::italic() const
{
return mItalic;
}
void ColorSchemeItem::setItalic(bool italic)
{
mItalic = italic;
}
bool ColorSchemeItem::underlined() const
{
return mUnderlined;
}
void ColorSchemeItem::setUnderlined(bool underlined)
{
mUnderlined = underlined;
}
bool ColorSchemeItem::strikeout() const
{
return mStrikeout;
}
void ColorSchemeItem::setStrikeout(bool strikeout)
{
mStrikeout = strikeout;
}
PColorSchemeItem ColorSchemeItem::fromJson(const QJsonObject &json)
{
PColorSchemeItem item = std::make_shared();
if (json.contains("foreground") && json["foreground"].isString()) {
item->setForeground(json["foreground"].toString());
} else {
item->setForeground(QColor());
}
if (json.contains("background") && json["background"].isString()) {
item->setBackground(json["background"].toString());
} else {
item->setBackground(QColor());
}
if (json.contains("bold") && json["bold"].isBool()) {
item->setBold(json["bold"].toBool());
} else {
item->setBold(false);
}
if (json.contains("italic") && json["italic"].isBool()) {
item->setItalic(json["italic"].toBool());
} else {
item->setItalic(false);
}
if (json.contains("underlined") && json["underlined"].isBool()) {
item->setUnderlined(json["underlined"].toBool());
} else {
item->setUnderlined(false);
}
if (json.contains("strikeout") && json["strikeout"].isBool()) {
item->setStrikeout(json["strikeout"].toBool());
} else {
item->setStrikeout(false);
}
return item;
}
void ColorSchemeItem::toJson(QJsonObject &json)
{
if (mForeground.isValid()) {
json["foreground"] = mForeground.name(QColor::HexArgb);
} else if (json.contains("foreground")){
json.remove("foreground");
}
if (mBackground.isValid()) {
json["background"] = mBackground.name(QColor::HexArgb);
} else if (json.contains("background")){
json.remove("background");
}
json["bold"] = mBold;
json["italic"] = mItalic;
json["underlined"] = mUnderlined;
json["strikeout"] = mStrikeout;
}
ColorManager::ColorManager()
{
mDefaultSchemeItemDefine = std::make_shared();
initItemDefines();
init();
}
void ColorManager::init()
{
reload();
}
void ColorManager::reload()
{
mSchemes.clear();
//bundled schemes ( the lowest priority)
loadSchemesInDir(pSettings->dirs().data(Settings::Dirs::DataType::ColorScheme),true,false);
//config schemes ( higher priority)
loadSchemesInDir(pSettings->dirs().config(Settings::Dirs::DataType::ColorScheme),false,false);
//customed schemes ( highest priority)
loadSchemesInDir(pSettings->dirs().config(Settings::Dirs::DataType::ColorScheme),false,true);
}
QStringList ColorManager::getSchemes(const QString &themeType)
{
if (themeType.isEmpty()) {
return mSchemes.keys();
}
QStringList lst;
for (QString name:mSchemes.keys()) {
PColorScheme scheme = mSchemes[name];
if (scheme && scheme->preferThemeType() == themeType) {
lst.append(name);
}
}
return lst;
}
QStringList ColorManager::getDefines()
{
return mSchemeItemDefines.keys();
}
bool ColorManager::exists(const QString name)
{
return mSchemes.contains(name);
}
QString ColorManager::copy(const QString &sourceName)
{
if (!mSchemes.contains(sourceName))
return QString();
PColorScheme sourceScheme = mSchemes[sourceName];
QString newName = sourceName+" Copy";
if (mSchemes.contains(newName))
return QString();
// save source with the new name
QString newFilepath = generateFullPathname(newName,false,false);
sourceScheme->save(newFilepath);
// then load it to the copied
PColorScheme newScheme = ColorScheme::load(newFilepath);
newScheme->setBundled(false);
newScheme->setCustomed(false);
mSchemes[newName]=newScheme;
return newName;
}
bool ColorManager::restoreToDefault(const QString &name)
{
PColorScheme scheme = get(name);
if (!scheme)
return false;
if (!scheme->customed())
return false;
QString fullPath = generateFullPathname(name,scheme->bundled(),false);
PColorScheme oldScheme = ColorScheme::load(fullPath);
if (!oldScheme)
throw FileError(QObject::tr("Can't Find the color scheme file %1!").arg(fullPath));
fullPath = generateFullPathname(name,scheme->bundled(),true);
QFile file(fullPath);
if (file.exists() && !file.remove())
throw FileError(QObject::tr("Can't remove the color scheme file %1!").arg(fullPath));
oldScheme->setBundled(scheme->bundled());
oldScheme->setCustomed(false);
mSchemes[name]=oldScheme;
return true;
}
bool ColorManager::remove(const QString &name)
{
PColorScheme scheme = get(name);
if (!scheme)
return false;
if (scheme->bundled())
return false;
if (scheme->customed()) {
QString fullPath = generateFullPathname(name,false,true);
QFile file(fullPath);
if (!file.remove())
throw FileError(QObject::tr("Can't remove the color scheme file %1!").arg(fullPath));
}
QString fullPath = generateFullPathname(name,false,false);
QFile file(fullPath);
if (!file.remove())
throw FileError(QObject::tr("Can't remove the color scheme file %1!").arg(fullPath));
mSchemes.remove(name);
return true;
}
QString ColorManager::generateFilename(const QString &name, bool isCustomed)
{
QString newName = name;
newName.replace(' ','_');
if (isCustomed)
newName += EXT_PREFIX_CUSTOM;
return newName += EXT_COLOR_SCHEME;
}
void ColorManager::loadSchemesInDir(const QString &dirName, bool isBundled, bool isCustomed)
{
QDir dir(dirName);
dir.setFilter(QDir::Files);
QFileInfoList list = dir.entryInfoList();
QString suffix;
QString customSuffix = EXT_PREFIX_CUSTOM;
customSuffix += EXT_COLOR_SCHEME;
if (isCustomed) {
suffix = customSuffix;
} else {
suffix = EXT_COLOR_SCHEME;
}
for (int i=0;isetBundled(isBundled);
scheme->setCustomed(false);
} else {
scheme->setBundled(false);
if (mSchemes.contains(name)) {
PColorScheme oldScheme = mSchemes[name];
if (oldScheme) {
scheme->setBundled(oldScheme->bundled());
}
mSchemes.remove(name);
}
scheme->setCustomed(true);
}
mSchemes[name]=scheme;
}
}
}
void ColorManager::initItemDefines()
{
//Token highlight colors
// addDefine(SYNS_AttrAssembler,
// QObject::tr("Assembler"),
// QObject::tr("Syntax"),
// true,true,true);
addDefine(SYNS_AttrCharacter,
QObject::tr("Character"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrComment,
QObject::tr("Comment"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrClass,
QObject::tr("Class"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrFloat,
QObject::tr("Float"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrFunction,
QObject::tr("Function"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrGlobalVariable,
QObject::tr("Gloabal Variable"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrHexadecimal,
QObject::tr("Hexadecimal Integer"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrIdentifier,
QObject::tr("Identifier"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrIllegalChar,
QObject::tr("Illegal Char"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrLocalVariable,
QObject::tr("Local Variable"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrNumber,
QObject::tr("Integer"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrOctal,
QObject::tr("Octal Integer"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrPreprocessor,
QObject::tr("Preprocessor"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrReservedWord,
QObject::tr("Reserve Word"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrReserveWord_Type,
QObject::tr("Reserve Word for Types"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrSpace,
QObject::tr("Space"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrString,
QObject::tr("String"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrStringEscapeSequences,
QObject::tr("Escape Sequences"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrSymbol,
QObject::tr("Symbol"),
QObject::tr("Syntax"),
true,true,true);
addDefine(SYNS_AttrVariable,
QObject::tr("Variable"),
QObject::tr("Syntax"),
true,true,true);
//Brace/Bracket/Parenthesis Level 1 2 3 4
addDefine(COLOR_SCHEME_BRACE_1,
QObject::tr("Brace/Bracket/Parenthesis Level 1"),
QObject::tr("Syntax"),
true,false,false);
addDefine(COLOR_SCHEME_BRACE_2,
QObject::tr("Brace/Bracket/Parenthesis Level 2"),
QObject::tr("Syntax"),
true,false,false);
addDefine(COLOR_SCHEME_BRACE_3,
QObject::tr("Brace/Bracket/Parenthesis Level 3"),
QObject::tr("Syntax"),
true,false,false);
addDefine(COLOR_SCHEME_BRACE_4,
QObject::tr("Brace/Bracket/Parenthesis Level 4"),
QObject::tr("Syntax"),
true,false,false);
//Gutter colors
addDefine(COLOR_SCHEME_GUTTER,
QObject::tr("Gutter"),
QObject::tr("Editor"),
true,true,true);
addDefine(COLOR_SCHEME_GUTTER_ACTIVE_LINE,
QObject::tr("Gutter Active Line"),
QObject::tr("Editor"),
true,false,false);
//Active Line
addDefine(COLOR_SCHEME_ACTIVE_LINE,
QObject::tr("Active Line"),
QObject::tr("Editor"),
false,true,false);
//Breakpoint Line
addDefine(COLOR_SCHEME_BREAKPOINT,
QObject::tr("Breakpoint"),
QObject::tr("Editor"),
true,true,false);
//Current Debug Line
addDefine(COLOR_SCHEME_ACTIVE_BREAKPOINT,
QObject::tr("Active Breakpoint"),
QObject::tr("Editor"),
true,true,false);
//Fold line
addDefine(COLOR_SCHEME_FOLD_LINE,
QObject::tr("Fold Line"),
QObject::tr("Editor"),
true,false,false);
addDefine(COLOR_SCHEME_SELECTION,
QObject::tr("Selection"),
QObject::tr("Editor"),
true,true,false);
addDefine(COLOR_SCHEME_TEXT,
QObject::tr("Editor Text"),
QObject::tr("Editor"),
true,true,false);
addDefine(COLOR_SCHEME_CURRENT_HIGHLIGHTED_WORD,
QObject::tr("Current Highlighted Word"),
QObject::tr("Editor"),
true,true,false);
//Syntax Error
addDefine(COLOR_SCHEME_ERROR,
QObject::tr("Error"),
QObject::tr("Syntax Check"),
true,false,false);
addDefine(COLOR_SCHEME_WARNING,
QObject::tr("Warning"),
QObject::tr("Syntax Check"),
true,false,false);
}
bool ColorManager::rename(const QString &oldName, const QString &newName)
{
if (mSchemes.contains(newName))
return false;
if (!isValidName(newName))
return false;
PColorScheme scheme = get(oldName);
if (!scheme)
return false;
if (scheme->bundled())
return false;
if (scheme->customed()) {
QString oldfullPath = generateFullPathname(oldName,false,true);
QString fullpath = generateFullPathname(newName,false,true);
QFile oldFile(oldfullPath);
if (oldFile.exists() && !oldFile.rename(fullpath))
throw FileError(QObject::tr("Rename file '%1' to '%2' failed!").arg(oldfullPath).arg(fullpath));
}
QString oldfullPath = generateFullPathname(oldName,false,false);
QString fullpath = generateFullPathname(newName,false,false);
QFile oldFile(oldfullPath);
if (oldFile.exists() && !oldFile.rename(fullpath))
throw FileError(QObject::tr("Rename file '%1' to '%2' failed!").arg(oldfullPath).arg(fullpath));
mSchemes.remove(oldName);
mSchemes[newName] = scheme;
return true;
}
bool ColorManager::add(const QString &name, PColorScheme scheme)
{
if (mSchemes.contains(name))
throw FileError(QObject::tr("Scheme '%1' already exists!").arg(name));
scheme->setBundled(false);
scheme->setCustomed(false);
mSchemes[name] = scheme;
saveScheme(name);
return true;
}
PColorScheme ColorManager::get(const QString &name)
{
if (mSchemes.contains(name))
return mSchemes[name];
return PColorScheme();
}
PColorSchemeItem ColorManager::getItem(const QString &schemeName, const QString &itemName)
{
PColorScheme scheme = get(schemeName);
if (!scheme)
return PColorSchemeItem();
if (!scheme->items().contains(itemName))
return PColorSchemeItem();
return scheme->items()[itemName];
}
bool ColorManager::isValidName(const QString &name)
{
for (QChar ch:name) {
if (!((ch == ' ') || (ch>='a' && ch<='z') || (ch>='A' && ch <= 'Z')
|| (ch>='0' && ch<='9') || (ch == '-') ))
return false;
}
return true;
}
void ColorManager::addDefine(const QString &name, const QString &displayName, const QString &group, bool hasForeground, bool hasBackground, bool hasFontStyle)
{
PColorSchemeItemDefine define = std::make_shared();
define->setDisplayName(displayName);
define->setGroup(group);
define->setHasForeground(hasForeground);
define->setHasBackground(hasBackground);
define->setHasFontStyle(hasFontStyle);
mSchemeItemDefines[name]=define;
}
bool ColorManager::removeDefine(const QString &name)
{
return mSchemeItemDefines.remove(name)==1;
}
PColorSchemeItemDefine ColorManager::getDefine(const QString &name)
{
if (mSchemeItemDefines.contains(name))
return mSchemeItemDefines[name];
return PColorSchemeItemDefine();
}
bool ColorManager::saveScheme(const QString &name)
{
PColorScheme scheme = get(name);
if (!scheme)
return false;
QString newFilepath = generateFullPathname(name,scheme->bundled(),scheme->customed());
scheme->save(newFilepath);
return true;
}
void ColorManager::updateStatementColors(std::shared_ptr > > statementColors, const QString &schemeName)
{
PColorSchemeItem item;
item = getItem(schemeName, SYNS_AttrFunction);
if (item) {
statementColors->insert(StatementKind::Function,item);
statementColors->insert(StatementKind::Constructor,item);
statementColors->insert(StatementKind::Destructor,item);
}
item = getItem(schemeName, SYNS_AttrClass);
if (item) {
statementColors->insert(StatementKind::Class,item);
statementColors->insert(StatementKind::Typedef,item);
statementColors->insert(StatementKind::Alias,item);
}
item = getItem(schemeName, SYNS_AttrIdentifier);
if (item) {
statementColors->insert(StatementKind::EnumType,item);
statementColors->insert(StatementKind::EnumClassType,item);
}
item = getItem(schemeName, SYNS_AttrVariable);
if (item) {
statementColors->insert(StatementKind::Variable,item);
}
item = getItem(schemeName, SYNS_AttrLocalVariable);
if (item) {
statementColors->insert(StatementKind::LocalVariable,item);
statementColors->insert(StatementKind::Parameter,item);
}
item = getItem(schemeName, SYNS_AttrGlobalVariable);
if (item) {
statementColors->insert(StatementKind::GlobalVariable,item);
}
item = getItem(schemeName, SYNS_AttrPreprocessor);
if (item) {
statementColors->insert(StatementKind::Preprocessor,item);
statementColors->insert(StatementKind::Enum,item);
}
item = getItem(schemeName, SYNS_AttrReservedWord);
if (item) {
statementColors->insert(StatementKind::Keyword,item);
statementColors->insert(StatementKind::UserCodeSnippet,item);
statementColors->insert(StatementKind::KeywordType,item);
}
item = getItem(schemeName, SYNS_AttrReserveWord_Type);
if (item) {
statementColors->insert(StatementKind::KeywordType,item);
}
item = getItem(schemeName, SYNS_AttrString);
if (item) {
statementColors->insert(StatementKind::Namespace,item);
statementColors->insert(StatementKind::NamespaceAlias,item);
}
}
QString ColorManager::generateFullPathname(const QString &name, bool isBundled, bool isCustomed)
{
QString filename = generateFilename(name,isCustomed);
if (isBundled && !isCustomed) {
return includeTrailingPathDelimiter(pSettings->dirs().data(Settings::Dirs::DataType::ColorScheme))+filename;
} else {
return includeTrailingPathDelimiter(pSettings->dirs().config(Settings::Dirs::DataType::ColorScheme))+filename;
}
}
ColorSchemeItemDefine::ColorSchemeItemDefine()
{
mHasBackground = true;
mHasForeground = true;
mHasFontStyle = true;
mGroup = QObject::tr("default");
}
bool ColorSchemeItemDefine::hasBackground() const
{
return mHasBackground;
}
void ColorSchemeItemDefine::setHasBackground(bool hasBackground)
{
mHasBackground = hasBackground;
}
bool ColorSchemeItemDefine::hasForeground() const
{
return mHasForeground;
}
void ColorSchemeItemDefine::setHasForeground(bool hasForeground)
{
mHasForeground = hasForeground;
}
bool ColorSchemeItemDefine::hasFontStyle() const
{
return mHasFontStyle;
}
void ColorSchemeItemDefine::setHasFontStyle(bool value)
{
mHasFontStyle = value;
}
QString ColorSchemeItemDefine::group() const
{
return mGroup;
}
void ColorSchemeItemDefine::setGroup(const QString &group)
{
mGroup = group;
}
QString ColorSchemeItemDefine::displayName() const
{
return mDisplayName;
}
void ColorSchemeItemDefine::setDisplayName(const QString &displayName)
{
mDisplayName = displayName;
}