mirror of https://github.com/mosra/magnum.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
277 lines
12 KiB
277 lines
12 KiB
/* |
|
This file is part of Magnum. |
|
|
|
Copyright © 2010, 2011, 2012, 2013, 2014, 2015, 2016, 2017, 2018, 2019, |
|
2020, 2021, 2022, 2023 Vladimír Vondruš <mosra@centrum.cz> |
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a |
|
copy of this software and associated documentation files (the "Software"), |
|
to deal in the Software without restriction, including without limitation |
|
the rights to use, copy, modify, merge, publish, distribute, sublicense, |
|
and/or sell copies of the Software, and to permit persons to whom the |
|
Software is furnished to do so, subject to the following conditions: |
|
|
|
The above copyright notice and this permission notice shall be included |
|
in all copies or substantial portions of the Software. |
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL |
|
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING |
|
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER |
|
DEALINGS IN THE SOFTWARE. |
|
*/ |
|
|
|
#include "Color.h" |
|
|
|
#if defined(DOXYGEN_GENERATING_OUTPUT) || defined(CORRADE_TARGET_UNIX) || (defined(CORRADE_TARGET_WINDOWS) && !defined(CORRADE_TARGET_WINDOWS_RT)) || defined(CORRADE_TARGET_EMSCRIPTEN) |
|
#include <cstring> |
|
#include <Corrade/Containers/StringView.h> |
|
#include <Corrade/Utility/Tweakable.h> |
|
#endif |
|
|
|
namespace Magnum { namespace Math { |
|
|
|
namespace { |
|
constexpr const char Hex[]{"0123456789abcdef"}; |
|
} |
|
|
|
Debug& operator<<(Debug& debug, const Color3<UnsignedByte>& value) { |
|
/* Print an actual colored square if requested */ |
|
if(debug.immediateFlags() & Debug::Flag::Color) { |
|
/* Pick a shade based on calculated lightness */ |
|
const Float valueValue = value.value(); |
|
const char* shade; |
|
if(valueValue <= 0.2f) shade = " "; |
|
else if(valueValue <= 0.4f) shade = "░░"; |
|
else if(valueValue <= 0.6f) shade = "▒▒"; |
|
else if(valueValue <= 0.8f) shade = "▓▓"; |
|
else shade = "██"; |
|
|
|
/* If ANSI colors are disabled, use just the shade */ |
|
if(debug.immediateFlags() & Debug::Flag::DisableColors) |
|
return debug << shade; |
|
else { |
|
debug << "\033[38;2;"; |
|
|
|
/* Disable space between values for everything after the initial |
|
value */ |
|
const Debug::Flags previousFlags = debug.flags(); |
|
debug.setFlags(previousFlags|Debug::Flag::NoSpace); |
|
|
|
/* Set both background and foreground, reset back after */ |
|
debug << int(value.r()) << ";" << int(value.g()) << ";" |
|
<< int(value.b()) << "m\033[48;2;" << int(value.r()) << ";" << |
|
int(value.g()) << ";" << int(value.b()) << "m" << shade << "\033[0m"; |
|
|
|
/* Reset original flags */ |
|
debug.setFlags(previousFlags); |
|
return debug; |
|
} |
|
|
|
/* Otherwise print a CSS color */ |
|
} else { |
|
char out[] = "#______"; |
|
out[1] = Hex[(value.r() >> 4) & 0xf]; |
|
out[2] = Hex[(value.r() >> 0) & 0xf]; |
|
out[3] = Hex[(value.g() >> 4) & 0xf]; |
|
out[4] = Hex[(value.g() >> 0) & 0xf]; |
|
out[5] = Hex[(value.b() >> 4) & 0xf]; |
|
out[6] = Hex[(value.b() >> 0) & 0xf]; |
|
return debug << out; |
|
} |
|
} |
|
|
|
Debug& operator<<(Debug& debug, const Color4<UnsignedByte>& value) { |
|
/* Print an actual colored square if requested */ |
|
if(debug.immediateFlags() & Debug::Flag::Color) { |
|
/* Pick a shade based on calculated lightness */ |
|
const Float valueValue = value.value(); |
|
const Float alpha = Math::unpack<Float>(value.a()); |
|
const Float valueAlpha = valueValue*alpha; |
|
const char* shade; |
|
if(valueAlpha <= 0.2f) shade = " "; |
|
else if(valueAlpha <= 0.4f) shade = "░░"; |
|
else if(valueAlpha <= 0.6f) shade = "▒▒"; |
|
else if(valueAlpha <= 0.8f) shade = "▓▓"; |
|
else shade = "██"; |
|
|
|
/* If ANSI colors are disabled, use just the shade */ |
|
if(debug.immediateFlags() & Debug::Flag::DisableColors) |
|
return debug << shade; |
|
else { |
|
debug << "\033[38;2;"; |
|
|
|
/* Disable space between values for everything after the initial |
|
value */ |
|
const Debug::Flags previousFlags = debug.flags(); |
|
debug.setFlags(previousFlags|Debug::Flag::NoSpace); |
|
|
|
/* Print foreground color */ |
|
debug << int(value.r()) << ";" << int(value.g()) << ";" |
|
<< int(value.b()) << "m"; |
|
|
|
/* If alpha is larger than perceived value, set also background */ |
|
if(alpha > valueValue) |
|
debug << "\033[48;2;" << int(value.r()) << ";" << |
|
int(value.g()) << ";" << int(value.b()) << "m"; |
|
|
|
/* Print the shade and reset color back */ |
|
debug << shade << "\033[0m"; |
|
|
|
/* Reset original flags */ |
|
debug.setFlags(previousFlags); |
|
return debug; |
|
} |
|
|
|
/* Otherwise print a CSS color */ |
|
} else { |
|
char out[] = "#________"; |
|
out[1] = Hex[(value.r() >> 4) & 0xf]; |
|
out[2] = Hex[(value.r() >> 0) & 0xf]; |
|
out[3] = Hex[(value.g() >> 4) & 0xf]; |
|
out[4] = Hex[(value.g() >> 0) & 0xf]; |
|
out[5] = Hex[(value.b() >> 4) & 0xf]; |
|
out[6] = Hex[(value.b() >> 0) & 0xf]; |
|
out[7] = Hex[(value.a() >> 4) & 0xf]; |
|
out[8] = Hex[(value.a() >> 0) & 0xf]; |
|
return debug << out; |
|
} |
|
} |
|
|
|
}} |
|
|
|
#if defined(DOXYGEN_GENERATING_OUTPUT) || defined(CORRADE_TARGET_UNIX) || (defined(CORRADE_TARGET_WINDOWS) && !defined(CORRADE_TARGET_WINDOWS_RT)) || defined(CORRADE_TARGET_EMSCRIPTEN) |
|
namespace Corrade { namespace Utility { |
|
|
|
Containers::Pair<TweakableState, Magnum::Math::Color3<Magnum::UnsignedByte>> TweakableParser<Magnum::Math::Color3<Magnum::UnsignedByte>>::parse(const Containers::StringView value) { |
|
using namespace Containers::Literals; |
|
|
|
if(value.size() < 2 || value[0] != '0' || (value[1] != 'x' && value[1] != 'X')) { |
|
Warning{} << "Utility::TweakableParser:" << value << "is not a hexadecimal color literal"; |
|
return {TweakableState::Recompile, {}}; |
|
} |
|
|
|
const bool isSrgb = value.hasSuffix("_srgb"_s); |
|
if(!isSrgb && !value.hasSuffix("_rgb"_s)) { |
|
Warning{} << "Utility::TweakableParser:" << value << "has an unexpected suffix, expected _rgb or _srgb"; |
|
return {TweakableState::Recompile, {}}; |
|
} |
|
|
|
char* end; |
|
const Magnum::UnsignedInt result = std::strtoul(value.data(), &end, 16); |
|
|
|
if(end != value.end() - (isSrgb ? 5 : 4)) { |
|
Warning{} << "Utility::TweakableParser: unexpected characters" << value.suffix(end) << "after a color literal"; |
|
return {TweakableState::Recompile, {}}; |
|
} |
|
|
|
if(value.size() != std::size_t(isSrgb ? 13 : 12)) { |
|
Error{} << "Utility::TweakableParser:" << value << "doesn't have expected number of digits"; |
|
return {TweakableState::Error, {}}; |
|
} |
|
|
|
return {TweakableState::Success, |
|
isSrgb ? Magnum::Math::Literals::operator "" _srgb(result) : |
|
Magnum::Math::Literals::operator "" _rgb(result)}; |
|
} |
|
|
|
Containers::Pair<TweakableState, Magnum::Math::Color4<Magnum::UnsignedByte>> TweakableParser<Magnum::Math::Color4<Magnum::UnsignedByte>>::parse(const Containers::StringView value) { |
|
using namespace Containers::Literals; |
|
|
|
if(value.size() < 2 || value[0] != '0' || (value[1] != 'x' && value[1] != 'X')) { |
|
Warning{} << "Utility::TweakableParser:" << value << "is not a hexadecimal color literal"; |
|
return {TweakableState::Recompile, {}}; |
|
} |
|
|
|
const bool isSrgb = value.hasSuffix("_srgba"_s); |
|
if(!isSrgb && !value.hasSuffix("_rgba"_s)) { |
|
Warning{} << "Utility::TweakableParser:" << value << "has an unexpected suffix, expected _rgba or _srgba"; |
|
return {TweakableState::Recompile, {}}; |
|
} |
|
|
|
char* end; |
|
const Magnum::UnsignedInt result = std::strtoul(value.data(), &end, 16); |
|
|
|
if(end != value.end() - (isSrgb ? 6 : 5)) { |
|
Warning{} << "Utility::TweakableParser: unexpected characters" << value.suffix(end) << "after a color literal"; |
|
return {TweakableState::Recompile, {}}; |
|
} |
|
|
|
if(value.size() != std::size_t(isSrgb ? 16 : 15)) { |
|
Error{} << "Utility::TweakableParser:" << value << "doesn't have expected number of digits"; |
|
return {TweakableState::Error, {}}; |
|
} |
|
|
|
return {TweakableState::Success, |
|
isSrgb ? Magnum::Math::Literals::operator "" _srgba(result) : |
|
Magnum::Math::Literals::operator "" _rgba(result)}; |
|
} |
|
|
|
Containers::Pair<TweakableState, Magnum::Math::Color3<Magnum::Float>> TweakableParser<Magnum::Math::Color3<Magnum::Float>>::parse(const Containers::StringView value) { |
|
using namespace Containers::Literals; |
|
|
|
if(value.size() < 2 || value[0] != '0' || (value[1] != 'x' && value[1] != 'X')) { |
|
Warning{} << "Utility::TweakableParser:" << value << "is not a hexadecimal color literal"; |
|
return {TweakableState::Recompile, {}}; |
|
} |
|
|
|
const bool isSrgb = value.hasSuffix("_srgbf"_s); |
|
if(!isSrgb && !value.hasSuffix("_rgbf"_s)) { |
|
Warning{} << "Utility::TweakableParser:" << value << "has an unexpected suffix, expected _rgbf or _srgbf"; |
|
return {TweakableState::Recompile, {}}; |
|
} |
|
|
|
char* end; |
|
const Magnum::UnsignedInt result = std::strtoul(value.data(), &end, 16); |
|
|
|
if(end != value.end() - (isSrgb ? 6 : 5)) { |
|
Warning{} << "Utility::TweakableParser: unexpected characters" << value.suffix(end) << "after a color literal"; |
|
return {TweakableState::Recompile, {}}; |
|
} |
|
|
|
if(value.size() != std::size_t(isSrgb ? 14 : 13)) { |
|
Error{} << "Utility::TweakableParser:" << value << "doesn't have expected number of digits"; |
|
return {TweakableState::Error, {}}; |
|
} |
|
|
|
return {TweakableState::Success, |
|
isSrgb ? Magnum::Math::Literals::operator "" _srgbf(result) : |
|
Magnum::Math::Literals::operator "" _rgbf(result)}; |
|
} |
|
|
|
Containers::Pair<TweakableState, Magnum::Math::Color4<Magnum::Float>> TweakableParser<Magnum::Math::Color4<Magnum::Float>>::parse(const Containers::StringView value) { |
|
using namespace Containers::Literals; |
|
|
|
if(value.size() < 2 || value[0] != '0' || (value[1] != 'x' && value[1] != 'X')) { |
|
Warning{} << "Utility::TweakableParser:" << value << "is not a hexadecimal color literal"; |
|
return {TweakableState::Recompile, {}}; |
|
} |
|
|
|
const bool isSrgb = value.hasSuffix("_srgbaf"_s); |
|
if(!isSrgb && !value.hasSuffix("_rgbaf"_s)) { |
|
Warning{} << "Utility::TweakableParser:" << value << "has an unexpected suffix, expected _rgbaf or _srgbaf"; |
|
return {TweakableState::Recompile, {}}; |
|
} |
|
|
|
char* end; |
|
const Magnum::UnsignedInt result = std::strtoul(value.data(), &end, 16); |
|
|
|
if(end != value.end() - (isSrgb ? 7 : 6)) { |
|
Warning{} << "Utility::TweakableParser: unexpected characters" << value.suffix(end) << "after a color literal"; |
|
return {TweakableState::Recompile, {}}; |
|
} |
|
|
|
if(value.size() != (isSrgb ? 17 : 16)) { |
|
Error{} << "Utility::TweakableParser:" << value << "doesn't have expected number of digits"; |
|
return {TweakableState::Error, {}}; |
|
} |
|
|
|
return {TweakableState::Success, |
|
isSrgb ? Magnum::Math::Literals::operator "" _srgbaf(result) : |
|
Magnum::Math::Literals::operator "" _rgbaf(result)}; |
|
} |
|
|
|
}} |
|
#endif
|
|
|