forked from mia/Aegisub
147 lines
4.4 KiB
C++
147 lines
4.4 KiB
C++
|
// Copyright (c) 2012, Thomas Goyne <plorkyeran@aegisub.org>
|
||
|
//
|
||
|
// Permission to use, copy, modify, and distribute this software for any
|
||
|
// purpose with or without fee is hereby granted, provided that the above
|
||
|
// copyright notice and this permission notice appear in all copies.
|
||
|
//
|
||
|
// THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
||
|
// WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
||
|
// MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
||
|
// ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
||
|
// WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
||
|
// ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
||
|
// OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
||
|
|
||
|
#include "../config.h"
|
||
|
|
||
|
#include "libaegisub/color.h"
|
||
|
|
||
|
#include <boost/config/warning_disable.hpp>
|
||
|
#include <boost/lexical_cast.hpp>
|
||
|
#include <boost/format.hpp>
|
||
|
#include <boost/spirit/include/qi.hpp>
|
||
|
#include <boost/spirit/include/phoenix_core.hpp>
|
||
|
#include <boost/spirit/include/phoenix_operator.hpp>
|
||
|
#include <boost/spirit/include/phoenix_fusion.hpp>
|
||
|
#include <boost/spirit/include/phoenix_stl.hpp>
|
||
|
#include <boost/spirit/include/phoenix_object.hpp>
|
||
|
#include <boost/fusion/include/adapt_struct.hpp>
|
||
|
|
||
|
BOOST_FUSION_ADAPT_STRUCT(
|
||
|
agi::Color,
|
||
|
(unsigned char, r)
|
||
|
(unsigned char, g)
|
||
|
(unsigned char, b)
|
||
|
(unsigned char, a)
|
||
|
)
|
||
|
|
||
|
namespace {
|
||
|
using namespace boost::spirit;
|
||
|
|
||
|
struct unpack_colors : public boost::static_visitor<agi::Color> {
|
||
|
template<typename T> struct result { typedef agi::Color type; };
|
||
|
|
||
|
template<class T> agi::Color operator()(T arg) const {
|
||
|
return boost::apply_visitor(*this, arg);
|
||
|
}
|
||
|
|
||
|
agi::Color operator()(int abgr) const { return (*this)((unsigned)abgr); }
|
||
|
agi::Color operator()(unsigned int abgr) const {
|
||
|
return agi::Color(abgr & 0xFF, (abgr >> 8) & 0xFF, (abgr >> 16) & 0xFF, (abgr >> 24) & 0xFF);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
template<typename Iterator>
|
||
|
struct color_grammar : qi::grammar<Iterator, agi::Color()> {
|
||
|
qi::rule<Iterator, agi::Color()> color;
|
||
|
|
||
|
qi::rule<Iterator, agi::Color()> css_color;
|
||
|
qi::rule<Iterator, agi::Color()> ass_color;
|
||
|
|
||
|
qi::rule<Iterator, unsigned char()> rgb_component;
|
||
|
qi::rule<Iterator, unsigned char()> rgb_percent;
|
||
|
qi::rule<Iterator, unsigned char()> hex_byte;
|
||
|
qi::rule<Iterator, unsigned char()> hex_char;
|
||
|
|
||
|
qi::rule<Iterator> comma;
|
||
|
qi::rule<Iterator> blank;
|
||
|
|
||
|
#define HEX_PARSER(type, len) qi::uint_parser<unsigned type, 16, len, len>()
|
||
|
|
||
|
color_grammar() : color_grammar::base_type(color) {
|
||
|
color = css_color | ass_color;
|
||
|
|
||
|
boost::phoenix::function<unpack_colors> unpack;
|
||
|
ass_color = (
|
||
|
int_
|
||
|
| -lit('&') >> -(lit('H') | lit('h')) >> (HEX_PARSER(int, 8) | HEX_PARSER(int, 6)) >> -lit('&')
|
||
|
)[_val = unpack(_1)] >> blank >> qi::eoi;
|
||
|
|
||
|
css_color
|
||
|
= "rgb(" >> blank >> rgb_component >> comma >> rgb_component >> comma >> rgb_component >> blank >> ')'
|
||
|
| '#' >> hex_byte >> hex_byte >> hex_byte
|
||
|
| '#' >> hex_char >> hex_char >> hex_char
|
||
|
;
|
||
|
|
||
|
hex_char = HEX_PARSER(int, 1)[_val = _1 * 16 + _1];
|
||
|
hex_byte = HEX_PARSER(int, 2);
|
||
|
rgb_component = qi::uint_parser<unsigned char, 10, 1, 3>();
|
||
|
|
||
|
comma = *qi::blank >> "," >> *qi::blank;
|
||
|
blank = *qi::blank;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
}
|
||
|
|
||
|
namespace agi {
|
||
|
|
||
|
Color::Color() : r(0), g(0), b(0), a(0) { }
|
||
|
|
||
|
Color::Color(unsigned char r, unsigned char g, unsigned char b, unsigned char a)
|
||
|
: r(r), g(g), b(b), a(a)
|
||
|
{ }
|
||
|
|
||
|
Color::Color(std::string const& str)
|
||
|
: r(0), g(0), b(0), a(0)
|
||
|
{
|
||
|
const char *begin = &str[0];
|
||
|
parse(begin, &str[str.size()], color_grammar<const char *>(), *this);
|
||
|
}
|
||
|
|
||
|
Color::Color(const char *str)
|
||
|
: r(0), g(0), b(0), a(0)
|
||
|
{
|
||
|
parse(str, str + strlen(str), color_grammar<const char *>(), *this);
|
||
|
}
|
||
|
|
||
|
std::string Color::GetAssStyleFormatted() const {
|
||
|
return str(boost::format("&H%02X%02X%02X%02X") % (int)a % (int)b % (int)g % (int)r);
|
||
|
}
|
||
|
|
||
|
std::string Color::GetAssOverrideFormatted() const {
|
||
|
return str(boost::format("&H%02X%02X%02X&") % (int)b % (int)g % (int)r);
|
||
|
}
|
||
|
|
||
|
std::string Color::GetSsaFormatted() const {
|
||
|
return boost::lexical_cast<std::string>((a << 24) + (b << 16) + (g << 8) + r);
|
||
|
}
|
||
|
|
||
|
std::string Color::GetHexFormatted() const {
|
||
|
return str(boost::format("#%02X%02X%02X") % (int)r % (int)g % (int)b);
|
||
|
}
|
||
|
|
||
|
std::string Color::GetRgbFormatted() const {
|
||
|
return str(boost::format("rgb(%d, %d, %d)") % (int)r % (int)g % (int)b);
|
||
|
}
|
||
|
|
||
|
bool Color::operator==(Color const& col) const {
|
||
|
return r == col.r && g == col.g && b == col.b && a == col.a;
|
||
|
}
|
||
|
|
||
|
bool Color::operator!=(Color const& col) const {
|
||
|
return !(*this == col);
|
||
|
}
|
||
|
|
||
|
}
|