co-maps/libs/kml/color_parser.cpp
2025-11-22 13:58:55 +01:00

214 lines
6.7 KiB
C++

#include "color_parser.hpp"
#include "coding/hex.hpp"
#include "types.hpp"
#include "base/string_utils.hpp"
namespace kml
{
struct RGBColor
{
uint8_t r, g, b;
};
std::optional<uint32_t> ParseHexColor(std::string_view c)
{
if (c.empty())
return {};
if (c.front() == '#')
c.remove_prefix(1);
if (c.size() != 6 && c.size() != 8)
return {};
auto const colorBytes = FromHex(c);
switch (colorBytes.size())
{
case 3: return ToRGBA(colorBytes[0], colorBytes[1], colorBytes[2]);
case 4: return ToRGBA(colorBytes[1], colorBytes[2], colorBytes[3], colorBytes[0]);
default: return {};
}
}
std::tuple<int, int, int> ExtractRGB(uint32_t rgbaColor)
{
return {(rgbaColor >> 24) & 0xFF, (rgbaColor >> 16) & 0xFF, (rgbaColor >> 8) & 0xFF};
}
static int ColorDistance(uint32_t rgbaColor1, uint32_t rgbaColor2)
{
auto const [r1, g1, b1] = ExtractRGB(rgbaColor1);
auto const [r2, g2, b2] = ExtractRGB(rgbaColor2);
return (r1 - r2) * (r1 - r2) + (g1 - g2) * (g1 - g2) + (b1 - b2) * (b1 - b2);
}
struct RGBAToGarmin
{
uint32_t rgba;
std::string_view color;
};
auto constexpr kRGBAToGarmin = std::to_array<RGBAToGarmin>({{0x000000ff, "Black"},
{0x8b0000ff, "DarkRed"},
{0x006400ff, "DarkGreen"},
{0xb5b820ff, "DarkYellow"},
{0x00008bff, "DarkBlue"},
{0x8b008bff, "DarkMagenta"},
{0x008b8bff, "DarkCyan"},
{0xccccccff, "LightGray"},
{0x444444ff, "DarkGray"},
{0xff0000ff, "Red"},
{0x00ff00ff, "Green"},
{0xffff00ff, "Yellow"},
{0x0000ffff, "Blue"},
{0xff00ffff, "Magenta"},
{0x00ffffff, "Cyan"},
{0xffffffff, "White"}});
std::string_view MapGarminColor(uint32_t rgba)
{
std::string_view closestColor = kRGBAToGarmin[0].color;
auto minDistance = std::numeric_limits<int>::max();
for (auto const & [rgbaGarmin, color] : kRGBAToGarmin)
{
auto const distance = ColorDistance(rgba, rgbaGarmin);
if (distance == 0)
return color; // Exact match.
if (distance < minDistance)
{
minDistance = distance;
closestColor = color;
}
}
return closestColor;
}
struct RGBAToPredefined
{
uint32_t rgba;
PredefinedColor predefinedColor;
};
static std::array<RGBAToPredefined, kOrderedPredefinedColors.size()> buildRGBAToPredefined()
{
auto res = std::array<RGBAToPredefined, kOrderedPredefinedColors.size()>();
for (size_t i = 0; i < kOrderedPredefinedColors.size(); ++i)
res[i] = {ColorFromPredefinedColor(kOrderedPredefinedColors[i]).GetRGBA(), kOrderedPredefinedColors[i]};
return res;
}
auto const kRGBAToPredefined = buildRGBAToPredefined();
PredefinedColor MapPredefinedColor(uint32_t rgba)
{
auto closestColor = kRGBAToPredefined[0].predefinedColor;
auto minDistance = std::numeric_limits<int>::max();
for (auto const & [rgbaGarmin, color] : kRGBAToPredefined)
{
auto const distance = ColorDistance(rgba, rgbaGarmin);
if (distance == 0)
return color; // Exact match.
if (distance < minDistance)
{
minDistance = distance;
closestColor = color;
}
}
return closestColor;
}
// Garmin extensions spec: https://www8.garmin.com/xmlschemas/GpxExtensionsv3.xsd
// Color mapping: https://help.locusmap.eu/topic/extend-garmin-gpx-compatibilty
std::optional<uint32_t> ParseGarminColor(std::string_view c)
{
/// @todo Unify with RGBColor instead of string.
static std::pair<std::string_view, std::string_view> arrColors[] = {
{"Black", "000000"}, {"DarkRed", "8b0000"}, {"DarkGreen", "006400"}, {"DarkYellow", "b5b820"},
{"DarkBlue", "00008b"}, {"DarkMagenta", "8b008b"}, {"DarkCyan", "008b8b"}, {"LightGray", "cccccc"},
{"DarkGray", "444444"}, {"Red", "ff0000"}, {"Green", "00ff00"}, {"Yellow", "ffff00"},
{"Blue", "0000ff"}, {"Magenta", "ff00ff"}, {"Cyan", "00ffff"}, {"White", "ffffff"}};
for (auto const & e : arrColors)
if (c == e.first)
return ParseHexColor(e.second);
return {};
}
std::optional<uint32_t> ParseOSMColor(std::string_view c)
{
static std::pair<std::string_view, RGBColor> arrColors[] = {
{"black", {0, 0, 0}},
{"white", {255, 255, 255}},
{"red", {255, 0, 0}},
{"green", {0, 128, 0}},
{"blue", {0, 0, 255}},
{"yellow", {255, 255, 0}},
{"orange", {255, 165, 0}},
{"gray", {128, 128, 128}},
{"grey", {128, 128, 128}}, // British spelling
{"brown", {165, 42, 42}},
{"pink", {255, 192, 203}},
{"purple", {128, 0, 128}},
{"cyan", {0, 255, 255}},
{"magenta", {255, 0, 255}},
{"maroon", {128, 0, 0}},
{"olive", {128, 128, 0}},
{"teal", {0, 128, 128}},
{"navy", {0, 0, 128}},
{"silver", {192, 192, 192}},
{"lime", {0, 255, 0}},
{"aqua", {0, 255, 255}}, // cyan
{"fuchsia", {255, 0, 255}}, // magenta
// From top taginfo for "colour" and CSS standart values.
{"darkgreen", {0, 100, 0}},
{"beige", {245, 245, 220}},
{"dimgray", {105, 105, 105}},
{"lightgrey", {211, 211, 211}}, // British spelling
{"lightgray", {211, 211, 211}},
{"tan", {210, 180, 140}},
{"gold", {255, 215, 0}},
{"red;white", {255, 127, 127}},
{"red and white", {255, 127, 127}},
{"red-white", {255, 127, 127}},
};
if (!c.empty())
{
if (c[0] == '#')
{
using strings::to_uint;
if (c.size() == 7) // #rrggbb
{
uint8_t r, g, b;
if (to_uint(c.substr(1, 2), r, 16) && to_uint(c.substr(3, 2), g, 16) && to_uint(c.substr(5, 2), b, 16))
return ToRGBA(r, g, b);
}
else if (c.size() == 4) // #rgb shorthand
{
uint8_t r, g, b;
if (to_uint(c.substr(1, 1), r, 16) && to_uint(c.substr(2, 1), g, 16) && to_uint(c.substr(3, 1), b, 16))
return ToRGBA(uint8_t(r * 17), uint8_t(g * 17), uint8_t(b * 17));
}
}
else
{
for (auto const & e : arrColors)
if (c == e.first)
return ToRGBA(e.second.r, e.second.g, e.second.b);
}
}
return {};
}
} // namespace kml