2021-06-17 10:39:46 +08:00
|
|
|
#include "colorscheme.h"
|
2021-06-18 08:09:32 +08:00
|
|
|
#include <QDir>
|
2021-06-17 10:39:46 +08:00
|
|
|
#include <QFile>
|
|
|
|
#include <QJsonArray>
|
|
|
|
#include <QJsonDocument>
|
|
|
|
#include <QJsonObject>
|
|
|
|
#include "utils.h"
|
2021-06-18 08:09:32 +08:00
|
|
|
#include "settings.h"
|
|
|
|
#include "qsynedit/Constants.h"
|
2021-06-17 10:39:46 +08:00
|
|
|
|
|
|
|
ColorScheme::ColorScheme()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
QString ColorScheme::name() const
|
|
|
|
{
|
|
|
|
return mName;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ColorScheme::setName(const QString &name)
|
|
|
|
{
|
|
|
|
mName = name;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ColorScheme::read(const QJsonObject &json)
|
|
|
|
{
|
|
|
|
if (json.contains("name") && json["name"].isString()) {
|
|
|
|
setName(json["name"].toString());
|
|
|
|
} else {
|
|
|
|
setName("");
|
|
|
|
}
|
|
|
|
mItems.clear();
|
|
|
|
if (json.contains("items") && json["items"].isObject()) {
|
|
|
|
QJsonObject itemsList = json["items"].toObject();
|
|
|
|
for (QString key:itemsList.keys()) {
|
|
|
|
PColorSchemeItem item = std::make_shared<ColorSchemeItem>(key);
|
|
|
|
item->read(itemsList[key].toObject());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ColorScheme::write(QJsonObject &json)
|
|
|
|
{
|
|
|
|
json["name"] = mName;
|
|
|
|
QJsonObject itemsList;
|
|
|
|
for (PColorSchemeItem item:mItems) {
|
|
|
|
if (!item->name().isEmpty()) {
|
|
|
|
QJsonObject itemObject;
|
|
|
|
item->write(itemObject);
|
|
|
|
itemsList[item->name()] = itemObject;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
json["items"]=itemsList;
|
|
|
|
}
|
|
|
|
|
2021-06-18 08:09:32 +08:00
|
|
|
PColorScheme ColorScheme::load(const QString &filename)
|
2021-06-17 10:39:46 +08:00
|
|
|
{
|
2021-06-18 08:09:32 +08:00
|
|
|
PColorScheme scheme = std::make_shared<ColorScheme>();
|
2021-06-17 10:39:46 +08:00
|
|
|
QFile file(filename);
|
|
|
|
if (!file.open(QFile::ReadOnly)) {
|
|
|
|
throw new FileError(QObject::tr("Can't open file '%1' for read").arg(file.fileName()));
|
|
|
|
}
|
|
|
|
QByteArray content = file.readAll();
|
|
|
|
QJsonParseError error;
|
|
|
|
QJsonDocument doc = QJsonDocument::fromJson(content,&error);
|
|
|
|
if (error.error!=QJsonParseError::NoError) {
|
|
|
|
throw new FileError(QObject::tr("Can't parse json file '%1' at offset %2! Error Code: %3")
|
|
|
|
.arg(file.fileName()).arg(error.offset).arg(error.error));
|
|
|
|
}
|
|
|
|
if (!doc.isObject()) {
|
|
|
|
throw new FileError(QObject::tr("Can't parse json file '%1' is not a color schema config file!")
|
|
|
|
.arg(file.fileName()));
|
|
|
|
}
|
2021-06-18 08:09:32 +08:00
|
|
|
scheme->read(doc.object());
|
|
|
|
return scheme;
|
2021-06-17 10:39:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ColorScheme::save(const QString &filename)
|
|
|
|
{
|
|
|
|
QFile file(filename);
|
|
|
|
if (!file.open(QFile::WriteOnly)) {
|
|
|
|
throw new FileError(QObject::tr("Can't open file '%1' for write").arg(file.fileName()));
|
|
|
|
}
|
|
|
|
QJsonObject json;
|
|
|
|
write(json);
|
|
|
|
QJsonDocument doc(json);
|
|
|
|
QByteArray content = doc.toJson();
|
|
|
|
file.write(content);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ColorScheme::bundled() const
|
|
|
|
{
|
|
|
|
return mBundled;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ColorScheme::setBundled(bool bundled)
|
|
|
|
{
|
|
|
|
mBundled = bundled;
|
|
|
|
}
|
|
|
|
|
2021-06-18 08:09:32 +08:00
|
|
|
bool ColorScheme::customed() const
|
2021-06-17 10:39:46 +08:00
|
|
|
{
|
2021-06-18 08:09:32 +08:00
|
|
|
return mCustomed;
|
2021-06-17 10:39:46 +08:00
|
|
|
}
|
|
|
|
|
2021-06-18 08:09:32 +08:00
|
|
|
void ColorScheme::setCustomed(bool customed)
|
2021-06-17 10:39:46 +08:00
|
|
|
{
|
2021-06-18 08:09:32 +08:00
|
|
|
mCustomed = customed;
|
2021-06-17 10:39:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
QString ColorScheme::preferThemeType() const
|
|
|
|
{
|
|
|
|
return mPreferThemeType;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ColorScheme::setPreferThemeType(const QString &preferThemeType)
|
|
|
|
{
|
|
|
|
mPreferThemeType = preferThemeType;
|
|
|
|
}
|
|
|
|
|
|
|
|
ColorSchemeItem::ColorSchemeItem(const QString& name):
|
|
|
|
mName(name),
|
|
|
|
mForeground(),
|
|
|
|
mBackground(),
|
|
|
|
mBold(false),
|
|
|
|
mItalic(false),
|
|
|
|
mUnderlined(false),
|
|
|
|
mStrikeout(false)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
QString ColorSchemeItem::name() const
|
|
|
|
{
|
|
|
|
return mName;
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ColorSchemeItem::read(const QJsonObject &json)
|
|
|
|
{
|
|
|
|
if (json.contains("foreground") && json["foreground"].isString()) {
|
|
|
|
setForeground(json["foreground"].toString());
|
|
|
|
} else {
|
|
|
|
setForeground(QColor());
|
|
|
|
}
|
|
|
|
if (json.contains("background") && json["background"].isString()) {
|
|
|
|
setBackground(json["background"].toString());
|
|
|
|
} else {
|
|
|
|
setBackground(QColor());
|
|
|
|
}
|
|
|
|
if (json.contains("bold") && json["bold"].isBool()) {
|
|
|
|
setBold(json["bold"].toBool());
|
|
|
|
} else {
|
|
|
|
setBold(false);
|
|
|
|
}
|
|
|
|
if (json.contains("italic") && json["italic"].isBool()) {
|
|
|
|
setBold(json["italic"].toBool());
|
|
|
|
} else {
|
|
|
|
setItalic(false);
|
|
|
|
}
|
|
|
|
if (json.contains("underlined") && json["underlined"].isBool()) {
|
|
|
|
setBold(json["underlined"].toBool());
|
|
|
|
} else {
|
|
|
|
setUnderlined(false);
|
|
|
|
}
|
|
|
|
if (json.contains("strikeout") && json["strikeout"].isBool()) {
|
|
|
|
setBold(json["strikeout"].toBool());
|
|
|
|
} else {
|
|
|
|
setUnderlined(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void ColorSchemeItem::write(QJsonObject &json)
|
|
|
|
{
|
|
|
|
if (mForeground.isValid()) {
|
|
|
|
json["foreground"] = mForeground.name();
|
|
|
|
} else if (json.contains("foreground")){
|
|
|
|
json.remove("foreground");
|
|
|
|
}
|
|
|
|
if (mBackground.isValid()) {
|
|
|
|
json["background"] = mBackground.name();
|
|
|
|
} else if (json.contains("background")){
|
|
|
|
json.remove("background");
|
|
|
|
}
|
|
|
|
json["bold"] = mBold;
|
|
|
|
json["italic"] = mItalic;
|
|
|
|
json["underlined"] = mUnderlined;
|
|
|
|
json["strikeout"] = mStrikeout;
|
|
|
|
}
|
|
|
|
|
2021-06-18 08:09:32 +08:00
|
|
|
ColorManager::ColorManager()
|
|
|
|
{
|
|
|
|
mDefaultSchemeItemDefine = std::make_shared<ColorSchemeItemDefine>();
|
|
|
|
initItemDefines();
|
|
|
|
}
|
|
|
|
|
2021-06-17 10:39:46 +08:00
|
|
|
void ColorManager::init()
|
|
|
|
{
|
|
|
|
reload();
|
|
|
|
}
|
|
|
|
|
2021-06-18 08:09:32 +08:00
|
|
|
void ColorManager::reload()
|
|
|
|
{
|
|
|
|
mSchemes.clear();
|
|
|
|
//bundled schemes ( the lowest priority)
|
|
|
|
loadSchemesInDir(pSettings->dirs().data(Settings::Dirs::DataType::ColorSheme),false);
|
|
|
|
//config schemes ( higher priority)
|
|
|
|
loadSchemesInDir(pSettings->dirs().config(Settings::Dirs::DataType::ColorSheme),false);
|
|
|
|
//customed schemes ( highest priority)
|
|
|
|
loadSchemesInDir(pSettings->dirs().config(Settings::Dirs::DataType::ColorSheme),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;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ColorManager::exists(const QString name)
|
|
|
|
{
|
|
|
|
return mSchemes.contains(name);
|
|
|
|
}
|
|
|
|
|
2021-06-17 10:39:46 +08:00
|
|
|
PColorScheme ColorManager::copy(const QString &sourceName)
|
|
|
|
{
|
|
|
|
if (!mSchemes.contains(sourceName))
|
|
|
|
return PColorScheme();
|
|
|
|
PColorScheme sourceScheme = mSchemes[sourceName];
|
2021-06-18 08:09:32 +08:00
|
|
|
QString newName = sourceName+" Copy";
|
|
|
|
if (mSchemes.contains(newName))
|
|
|
|
return PColorScheme();
|
|
|
|
// save source with the new name
|
|
|
|
QString newFilepath = generateFullPathname(newName,false,false);
|
|
|
|
sourceScheme->save(newFilepath);
|
2021-06-17 10:39:46 +08:00
|
|
|
// then load it to the copied
|
2021-06-18 08:09:32 +08:00
|
|
|
PColorScheme newScheme = ColorScheme::load(newFilepath);
|
|
|
|
newScheme->setName(newName);
|
|
|
|
newScheme->setBundled(false);
|
|
|
|
newScheme->setCustomed(false);
|
|
|
|
mSchemes[newName]=newScheme;
|
|
|
|
return newScheme;
|
|
|
|
}
|
2021-06-17 10:39:46 +08:00
|
|
|
|
2021-06-18 08:09:32 +08:00
|
|
|
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 isCustomed)
|
|
|
|
{
|
|
|
|
QDir dir(dirName);
|
|
|
|
dir.setFilter(QDir::Files);
|
|
|
|
QFileInfoList list = dir.entryInfoList();
|
|
|
|
QString suffix;
|
|
|
|
if (isCustomed) {
|
|
|
|
suffix = EXT_PREFIX_CUSTOM;
|
|
|
|
suffix = suffix + EXT_COLOR_SCHEME;
|
|
|
|
} else {
|
|
|
|
suffix = EXT_COLOR_SCHEME;
|
|
|
|
}
|
|
|
|
for (int i=0;i<list.size();i++) {
|
|
|
|
QFileInfo fileInfo = list[i];
|
|
|
|
if (fileInfo.fileName().toLower().endsWith(suffix)) {
|
|
|
|
PColorScheme scheme = ColorScheme::load(fileInfo.absoluteFilePath());
|
|
|
|
mSchemes[scheme->name()]=scheme;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ColorManager::initItemDefines()
|
|
|
|
{
|
|
|
|
//Highlighter colors
|
|
|
|
addDefine(SYNS_AttrAssembler,true,true,true);
|
|
|
|
addDefine(SYNS_AttrCharacter,true,true,true);
|
|
|
|
addDefine(SYNS_AttrComment,true,true,true);
|
|
|
|
addDefine(SYNS_AttrClass,true,true,true);
|
|
|
|
addDefine(SYNS_AttrFloat,true,true,true);
|
|
|
|
addDefine(SYNS_AttrFunction,true,true,true);
|
|
|
|
addDefine(SYNS_AttrGlobalVariable,true,true,true);
|
|
|
|
addDefine(SYNS_AttrHexadecimal,true,true,true);
|
|
|
|
addDefine(SYNS_AttrIdentifier,true,true,true);
|
|
|
|
addDefine(SYNS_AttrIllegalChar,true,true,true);
|
|
|
|
addDefine(SYNS_AttrLocalVariable,true,true,true);
|
|
|
|
addDefine(SYNS_AttrNumber,true,true,true);
|
|
|
|
addDefine(SYNS_AttrOctal,true,true,true);
|
|
|
|
addDefine(SYNS_AttrPreprocessor,true,true,true);
|
|
|
|
addDefine(SYNS_AttrReservedWord,true,true,true);
|
|
|
|
addDefine(SYNS_AttrSpace,true,true,true);
|
|
|
|
addDefine(SYNS_AttrString,true,true,true);
|
|
|
|
addDefine(SYNS_AttrStringEscapeSequences,true,true,true);
|
|
|
|
addDefine(SYNS_AttrSymbol,true,true,true);
|
|
|
|
addDefine(SYNS_AttrVariable,true,true,true);
|
|
|
|
|
|
|
|
//Gutter colors
|
|
|
|
addDefine(COLOR_SCHEME_GUTTER,true,true,true);
|
|
|
|
//Active Line
|
|
|
|
addDefine(COLOR_SCHEME_ACTIVE_LINE,false,true,false);
|
|
|
|
//Breakpoint Line
|
|
|
|
addDefine(COLOR_SCHEME_BREAKPOINT,true,true,false);
|
|
|
|
//Current Debug Line
|
|
|
|
addDefine(COLOR_SCHEME_ACTIVE_BREAKPOINT,true,true,false);
|
|
|
|
//Fold line
|
|
|
|
addDefine(COLOR_SCHEME_FOLD_LINE,true,false,false);
|
|
|
|
//Brace/Bracket/Parenthesis Level 1 2 3 4
|
|
|
|
addDefine(COLOR_SCHEME_BRACE_1,true,false,false);
|
|
|
|
addDefine(COLOR_SCHEME_BRACE_2,true,false,false);
|
|
|
|
addDefine(COLOR_SCHEME_BRACE_3,true,false,false);
|
|
|
|
addDefine(COLOR_SCHEME_BRACE_4,true,false,false);
|
|
|
|
addDefine(COLOR_SCHEME_SELECTION,true,true,false);
|
|
|
|
//Syntax Error
|
|
|
|
addDefine(COLOR_SCHEME_ERROR,true,false,false);
|
|
|
|
addDefine(COLOR_SCHEME_WARNING,true,false,false);
|
|
|
|
|
|
|
|
|
|
|
|
#define COLOR_SCHEME_INDENT_GUIDE_LINE "indent guide line"
|
2021-06-17 10:39:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
mSchemes[newName] = scheme;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
PColorScheme ColorManager::remove(const QString &name)
|
|
|
|
{
|
|
|
|
PColorScheme scheme=get(name);
|
|
|
|
if (scheme)
|
|
|
|
mSchemes.remove(name);
|
|
|
|
return scheme;
|
|
|
|
}
|
|
|
|
|
|
|
|
PColorScheme ColorManager::get(const QString &name)
|
|
|
|
{
|
|
|
|
if (mSchemes.contains(name))
|
|
|
|
return mSchemes[name];
|
|
|
|
return PColorScheme();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ColorManager::isValidName(const QString &name)
|
|
|
|
{
|
|
|
|
for (QChar ch:name) {
|
|
|
|
if (!((ch == ' ') or (ch>='a' && ch<='z') or (ch>='A' && ch <= 'Z')
|
|
|
|
or (ch>='0' && ch<='9') or (ch == '-') ))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ColorManager::addDefine(const QString &name, bool hasForeground, bool hasBackground, bool hasFontStyle)
|
|
|
|
{
|
|
|
|
PColorSchemeItemDefine define = std::make_shared<ColorSchemeItemDefine>();
|
|
|
|
define->setHasForeground(hasForeground);
|
|
|
|
define->setHasBackground(hasBackground);
|
|
|
|
define->setHasFontStyle(hasFontStyle);
|
2021-06-18 08:09:32 +08:00
|
|
|
mSchemeItemDefine[name]=define;
|
2021-06-17 10:39:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ColorManager::removeDefine(const QString &name)
|
|
|
|
{
|
2021-06-18 08:09:32 +08:00
|
|
|
return mSchemeItemDefine.remove(name)==1;
|
2021-06-17 10:39:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
PColorSchemeItemDefine ColorManager::getDefine(const QString &name)
|
|
|
|
{
|
2021-06-18 08:09:32 +08:00
|
|
|
if (mSchemeItemDefine.contains(name))
|
|
|
|
return mSchemeItemDefine[name];
|
2021-06-17 10:39:46 +08:00
|
|
|
return PColorSchemeItemDefine();
|
|
|
|
}
|
|
|
|
|
2021-06-18 08:09:32 +08:00
|
|
|
QString ColorManager::generateFullPathname(const QString &name, bool isBundled, bool isCustomed)
|
|
|
|
{
|
|
|
|
QString filename = generateFilename(name,isCustomed);
|
|
|
|
if (isBundled) {
|
|
|
|
return includeTrailingPathDelimiter(pSettings->dirs().data(Settings::Dirs::DataType::ColorSheme))+filename;
|
|
|
|
} else {
|
|
|
|
return includeTrailingPathDelimiter(pSettings->dirs().config(Settings::Dirs::DataType::ColorSheme))+filename;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ColorSchemeItemDefine::ColorSchemeItemDefine()
|
|
|
|
{
|
|
|
|
mHasBackground = true;
|
|
|
|
mHasForeground = true;
|
|
|
|
mHasFontStyle = true;
|
|
|
|
}
|
|
|
|
|
2021-06-17 10:39:46 +08:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2021-06-18 08:09:32 +08:00
|
|
|
bool ColorSchemeItemDefine::hasFontStyle() const
|
2021-06-17 10:39:46 +08:00
|
|
|
{
|
2021-06-18 08:09:32 +08:00
|
|
|
return mHasFontStyle;
|
2021-06-17 10:39:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ColorSchemeItemDefine::setHasFontStyle(bool value)
|
|
|
|
{
|
2021-06-18 08:09:32 +08:00
|
|
|
mHasFontStyle = value;
|
2021-06-17 10:39:46 +08:00
|
|
|
}
|