src/colors.cpp

Wed, 25 May 2022 20:36:34 +0300

author
Teemu Piippo <teemu@hecknology.net>
date
Wed, 25 May 2022 20:36:34 +0300
changeset 199
6988973515d2
parent 178
a23024fc98e0
child 205
1a4342d80de7
permissions
-rw-r--r--

Fix pick() picking from weird places on the screen with high DPI scaling

glReadPixels reads data from the frame buffer, which contains data after
high DPI scaling, so any reads to that need to take this scaling into account

/*
 *  LDForge: LDraw parts authoring CAD
 *  Copyright (C) 2013 - 2020 Teemu Piippo
 *
 *  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 <http://www.gnu.org/licenses/>.
 */

#include "colors.h"

const ldraw::ColorDefinition ldraw::ColorTable::unknownColor{{}, {}, "Unknown", "???"};

/**
 * @brief Clears the color table
 */
void ldraw::ColorTable::clear()
{
	this->definitions = {};
}

/**
 * @brief Loads colors from LDConfig.ldr
 * @param device Opened LDConfig.ldr I/O device
 * @param errors Where to write any errors into
 * @returns whether or not it succeeded.
 */
Result ldraw::ColorTable::load(QIODevice& device, QTextStream& errors)
{
	this->clear();
	if (device.isReadable())
	{
		QTextStream stream{&device};
		QString line;
		while (stream.readLineInto(&line))
		{
			this->loadColorFromString(line);
		}
		return Success;
	}
	else
	{
		errors << "could not read colors";
		return Failure;
	}
}

/**
 * @brief Gets color information by color index.
 * @param color
 * @returns color table information
 */
const ldraw::ColorDefinition& ldraw::ColorTable::operator[](Color color) const
{
	auto it = this->definitions.find(color.index);
	if (it != this->definitions.end())
	{
		return it->second;
	}
	else
	{
		return unknownColor;
	}
}

/**
 * @brief Gets the amount of elements in the color table
 * @returns int
 */
int ldraw::ColorTable::size() const
{
	return this->definitions.size();
}

/**
 * @brief Parses an LDConfig.ldr line from a string
 * @param string LDConfig.ldr line to parse
 */
void ldraw::ColorTable::loadColorFromString(const QString& string)
{
	const QRegExp pattern{
		R"(^\s*0 \!COLOUR\s+([^\s]+)\s+)"_q +
		R"(CODE\s+(\d+)\s+)"_q +
		R"(VALUE\s+(\#[0-9a-fA-F]{3,6})\s+)"_q +
		R"(EDGE\s+(\#[0-9a-fA-F]{3,6}))"_q +
		R"((?:\s+ALPHA\s+(\d+))?)"_q
	};
	if (pattern.indexIn(string) != -1)
	{
		const int code = pattern.cap(2).toInt();
		ColorDefinition& definition = this->definitions[code];
		definition = {}; // in case there's an existing definition
		definition.name = pattern.cap(1);
		definition.displayName = definition.name;
		definition.displayName.replace("_", " ");
		definition.faceColor = pattern.cap(3);
		definition.edgeColor = pattern.cap(4);
		if (not pattern.cap(5).isEmpty())
		{
			const int alpha = pattern.cap(5).toInt();
			definition.faceColor.setAlpha(alpha);
		}
	}
}

/**
 * @brief Calculates the luma-value for the given color.
 * @details c.f. https://en.wikipedia.org/wiki/Luma_(video)
 * @param color
 * @returns luma value [0, 1]
 */
double luma(const QColor& color)
{
	return 0.2126 * color.redF() + 0.7152 * color.greenF() + 0.0722 * color.blueF();
}

/**
 * @brief Returns a direct color index that codes the specified color value
 * @param color
 * @returns direct color index
 */
ldraw::Color ldraw::directColor(const QColor& color)
{
	return ldraw::Color{0x2000000 | (color.red() << 16) | (color.green() << 8) | color.blue()};
}

/**
 * @brief Checks whether or not the specified color index is a direct color
 * @param color Color to check
 * @returns bool
 */
bool ldraw::isDirectColor(ldraw::Color color)
{
	return color.index >= 0x2000000;
}

/**
 * @brief Extracts the color value from a direct color index
 * @param color Direct color index
 * @returns color value. Returns a default-constructed QColor in case a non-direct color is given.
 */
QColor ldraw::directColorFace(ldraw::Color color)
{
	if (isDirectColor(color))
	{
		return {(color.index >> 16) & 0xff, (color.index >> 8) & 0xff, color.index & 0xff};
	}
	else
	{
		return {};
	}
}

/**
 * @brief Gets the face color for the specified color index
 * @param color Color index to get face color for
 * @param colorTable Color table to use for lookup
 * @returns QColor
 */
QColor ldraw::colorFace(ldraw::Color color, const ldraw::ColorTable& colorTable)
{
	if (isDirectColor(color))
	{
		return directColorFace(color);
	}
	else
	{
		return colorTable[color].faceColor;
	}
}

QColor ldraw::colorEdge(ldraw::Color color, const ldraw::ColorTable& colorTable)
{
	if (isDirectColor(color))
	{
		QColor const faceColor = directColorFace(color);
		return (luma(faceColor) < 0.4) ? Qt::white : Qt::black;
	}
	else
	{
		return colorTable[color].faceColor;
	}
}

/**
 * @brief Writes a color index into a @c QDataStream
 * @param stream
 * @param color
 * @returns stream
 */
QDataStream& operator<<(QDataStream& stream, ldraw::Color color)
{
	return stream << color.index;
}

/**
 * @brief Reads a color index from a @c QDataStream
 * @param stream
 * @param color
 * @returns stream
 */
QDataStream& operator>>(QDataStream& stream, ldraw::Color& color)
{
	return stream >> color.index;
}

QString ldraw::colorDisplayName(ldraw::Color color, const ldraw::ColorTable &colorTable)
{
	if (isDirectColor(color))
	{
		return directColorFace(color).name();
	}
	else
	{
		return colorTable[color].displayName;
	}
}

mercurial