mirror of
https://github.com/swift-project/pilotclient.git
synced 2026-03-22 23:05:36 +08:00
255 lines
6.8 KiB
C++
255 lines
6.8 KiB
C++
/* Copyright (C) 2015
|
|
* swift project Community / Contributors
|
|
*
|
|
* This file is part of swift project. It is subject to the license terms in the LICENSE file found in the top-level
|
|
* directory of this distribution. No part of swift project, including this file, may be copied, modified, propagated,
|
|
* or distributed except according to the terms contained in the LICENSE file.
|
|
*/
|
|
|
|
#include "blackmisc/rgbcolor.h"
|
|
#include "blackmisc/compare.h"
|
|
#include "blackmisc/comparefunctions.h"
|
|
#include "blackmisc/icons.h"
|
|
#include "blackmisc/stringutils.h"
|
|
|
|
#include <QBrush>
|
|
#include <QPainter>
|
|
#include <QPixmap>
|
|
#include <QSize>
|
|
#include <QStringBuilder>
|
|
#include <Qt>
|
|
#include <QtGlobal>
|
|
|
|
using namespace BlackMisc;
|
|
|
|
namespace BlackMisc
|
|
{
|
|
CRgbColor::CRgbColor(const QString &color, bool isName)
|
|
{
|
|
this->setByString(color, isName);
|
|
}
|
|
|
|
CRgbColor::CRgbColor(int r, int g, int b) : m_r(r), m_g(g), m_b(b)
|
|
{ }
|
|
|
|
CRgbColor::CRgbColor(const QColor &color)
|
|
{
|
|
this->setQColor(color);
|
|
}
|
|
|
|
CIcon CRgbColor::toIcon() const
|
|
{
|
|
if (this->isValid())
|
|
{
|
|
QPixmap pixmap(QSize(16, 16));
|
|
QPainter p(&pixmap);
|
|
QBrush brush(this->toQColor());
|
|
p.setBackground(brush);
|
|
p.setBrush(this->toQColor());
|
|
p.drawRect(0, 0, 16, 16);
|
|
CIcon icon(pixmap, hex());
|
|
return icon;
|
|
}
|
|
else
|
|
{
|
|
return CIcon::iconByIndex(CIcons::StandardIconError16);
|
|
}
|
|
}
|
|
|
|
QColor CRgbColor::toQColor() const
|
|
{
|
|
return QColor(red(), green(), blue());
|
|
}
|
|
|
|
bool CRgbColor::setQColor(const QColor &color)
|
|
{
|
|
if (color.isValid())
|
|
{
|
|
m_r = color.red();
|
|
m_g = color.green();
|
|
m_b = color.blue();
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
this->setInvalid();
|
|
return false;
|
|
}
|
|
}
|
|
|
|
int CRgbColor::red() const
|
|
{
|
|
return m_r;
|
|
}
|
|
|
|
double CRgbColor::normalizedRed() const
|
|
{
|
|
double c = red() * 1.0;
|
|
return c / colorRange();
|
|
}
|
|
|
|
QString CRgbColor::redHex(int digits) const
|
|
{
|
|
return intToHex(m_r, digits);
|
|
}
|
|
|
|
int CRgbColor::green() const
|
|
{
|
|
return m_g;
|
|
}
|
|
|
|
double CRgbColor::normalizedGreen() const
|
|
{
|
|
double c = green() * 1.0;
|
|
return c / colorRange();
|
|
}
|
|
|
|
QString CRgbColor::greenHex(int digits) const
|
|
{
|
|
return intToHex(m_g, digits);
|
|
}
|
|
|
|
int CRgbColor::blue() const
|
|
{
|
|
return m_b;
|
|
}
|
|
|
|
double CRgbColor::normalizedBlue() const
|
|
{
|
|
double c = blue() * 1.0;
|
|
return c / colorRange();
|
|
}
|
|
|
|
QString CRgbColor::blueHex(int digits) const
|
|
{
|
|
return intToHex(m_b, digits);
|
|
}
|
|
|
|
QString CRgbColor::hex(bool withHash) const
|
|
{
|
|
if (!isValid()) { return {}; }
|
|
const QString h(redHex() + greenHex() + blueHex());
|
|
return withHash ? u'#' % h : h;
|
|
}
|
|
|
|
void CRgbColor::setByString(const QString &color, bool isName)
|
|
{
|
|
if (color.isEmpty()) { return; }
|
|
else if (isName)
|
|
{
|
|
const QColor q(color);
|
|
m_r = q.red();
|
|
m_g = q.green();
|
|
m_b = q.blue();
|
|
}
|
|
else
|
|
{
|
|
const QString c(color.trimmed());
|
|
QColor q(c);
|
|
if (this->setQColor(q)) { return; }
|
|
if (c.startsWith("#")) { this->setInvalid(); return; }
|
|
q.setNamedColor("#" + c);
|
|
this->setQColor(q);
|
|
}
|
|
}
|
|
|
|
bool CRgbColor::isValid() const
|
|
{
|
|
return m_r >= 0 && m_g >= 0 && m_b >= 0;
|
|
}
|
|
|
|
double CRgbColor::colorDistance(const CRgbColor &color) const
|
|
{
|
|
if (!this->isValid() && !color.isValid()) { return 0; }
|
|
if (!this->isValid() || !color.isValid()) { return 1; }
|
|
if (*this == color) { return 0.0; } // avoid rounding
|
|
|
|
// all values 0-1
|
|
const double rd = (normalizedRed() - color.normalizedRed());
|
|
const double bd = (normalizedBlue() - color.normalizedBlue());
|
|
const double gd = (normalizedGreen() - color.normalizedGreen());
|
|
return (rd * rd + bd * bd + gd * gd) / 3.0;
|
|
}
|
|
|
|
void CRgbColor::setInvalid()
|
|
{
|
|
m_r = -1;
|
|
m_g = -1;
|
|
m_b = -1;
|
|
}
|
|
|
|
QString CRgbColor::convertToQString(bool i18n) const
|
|
{
|
|
Q_UNUSED(i18n);
|
|
return this->hex();
|
|
}
|
|
|
|
CVariant CRgbColor::propertyByIndex(const BlackMisc::CPropertyIndex &index) const
|
|
{
|
|
if (index.isMyself()) { return CVariant::from(*this); }
|
|
const ColumnIndex i = index.frontCasted<ColumnIndex>();
|
|
switch (i)
|
|
{
|
|
case IndexBlue: return CVariant::fromValue(blue());
|
|
case IndexRed: return CVariant::fromValue(red());
|
|
case IndexGreen: return CVariant::fromValue(green());
|
|
case IndexWebHex: return CVariant::fromValue(hex());
|
|
default: return CValueObject::propertyByIndex(index);
|
|
}
|
|
}
|
|
|
|
void CRgbColor::setPropertyByIndex(const CPropertyIndex &index, const CVariant &variant)
|
|
{
|
|
if (index.isMyself()) { (*this) = variant.to<CRgbColor>(); return; }
|
|
const ColumnIndex i = index.frontCasted<ColumnIndex>();
|
|
switch (i)
|
|
{
|
|
case IndexBlue: m_b = variant.toInt(); break;
|
|
case IndexRed: m_r = variant.toInt(); break;
|
|
case IndexGreen: m_g = variant.toInt(); break;
|
|
case IndexWebHex: this->setByString(variant.toQString()); break;
|
|
default: CValueObject::setPropertyByIndex(index, variant); break;
|
|
}
|
|
}
|
|
|
|
int CRgbColor::comparePropertyByIndex(const CPropertyIndex &index, const CRgbColor &compareValue) const
|
|
{
|
|
if (index.isMyself()) { return this->compare(compareValue); }
|
|
const ColumnIndex i = index.frontCasted<ColumnIndex>();
|
|
switch (i)
|
|
{
|
|
case IndexBlue: return Compare::compare(m_b, compareValue.m_b);
|
|
case IndexRed: return Compare::compare(m_r, compareValue.m_r);
|
|
case IndexGreen: return Compare::compare(m_g, compareValue.m_g);
|
|
case IndexWebHex: return this->compare(compareValue);
|
|
default:
|
|
Q_ASSERT_X(false, Q_FUNC_INFO, "Missing compare");
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int CRgbColor::compare(const CRgbColor &color) const
|
|
{
|
|
int c = Compare::compare(m_r, color.m_r);
|
|
if (c != 0) { return c; }
|
|
c = Compare::compare(m_g, color.m_g);
|
|
if (c != 0) { return c; }
|
|
return Compare::compare(m_b, color.m_b);
|
|
}
|
|
|
|
double CRgbColor::colorRange() const
|
|
{
|
|
if (!this->isValid()) { return 255; }
|
|
if (m_b < 256 && m_g < 256 && m_r < 256) { return 255; }
|
|
if (m_b < 4096 && m_g < 4096 && m_r < 4096) { return 4095; }
|
|
return 65535;
|
|
}
|
|
|
|
QString CRgbColor::intToHex(int h, int digits)
|
|
{
|
|
return BlackMisc::intToHex(h, digits);
|
|
}
|
|
|
|
} // namespace
|