2009-01-14 13:10:24 +01:00
|
|
|
/*****************************************************************************
|
2012-02-02 02:53:35 +01:00
|
|
|
*
|
2009-01-14 13:10:24 +01:00
|
|
|
* This file is part of Mapnik (c++ mapping toolkit)
|
|
|
|
*
|
2017-05-05 13:02:01 +02:00
|
|
|
* Copyright (C) 2017 Artem Pavlenko
|
2009-01-14 13:10:24 +01:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library 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
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*
|
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
// mapnik
|
|
|
|
#include <mapnik/color.hpp>
|
|
|
|
#include <mapnik/color_factory.hpp>
|
2012-03-11 23:24:28 +01:00
|
|
|
#include <mapnik/config_error.hpp>
|
2012-10-04 22:53:58 +02:00
|
|
|
|
2014-10-22 01:37:27 +02:00
|
|
|
#pragma GCC diagnostic push
|
2015-11-08 02:53:09 +01:00
|
|
|
#include <mapnik/warning_ignore.hpp>
|
2013-06-21 12:32:06 +02:00
|
|
|
#include <boost/spirit/include/karma.hpp>
|
2014-10-22 01:37:27 +02:00
|
|
|
#pragma GCC diagnostic pop
|
2014-11-20 15:25:50 +01:00
|
|
|
|
2009-01-14 13:10:24 +01:00
|
|
|
namespace mapnik {
|
|
|
|
|
2015-01-25 03:48:15 +01:00
|
|
|
color::color(std::string const& str, bool premultiplied)
|
2010-06-02 13:03:30 +02:00
|
|
|
{
|
2012-08-29 19:44:04 +02:00
|
|
|
*this = parse_color(str);
|
2015-01-25 03:48:15 +01:00
|
|
|
premultiplied_ = premultiplied;
|
2010-06-02 13:03:30 +02:00
|
|
|
}
|
2009-01-14 13:10:24 +01:00
|
|
|
|
2010-06-02 13:03:30 +02:00
|
|
|
std::string color::to_string() const
|
|
|
|
{
|
2013-06-21 15:47:38 +02:00
|
|
|
namespace karma = boost::spirit::karma;
|
2014-01-26 23:00:58 +01:00
|
|
|
boost::spirit::karma::eps_type eps;
|
|
|
|
boost::spirit::karma::double_type double_;
|
2016-02-22 12:45:28 +01:00
|
|
|
boost::spirit::karma::uint_generator<uint8_t,10> color_;
|
2013-06-21 15:47:38 +02:00
|
|
|
std::string str;
|
|
|
|
std::back_insert_iterator<std::string> sink(str);
|
2016-02-22 12:45:28 +01:00
|
|
|
karma::generate(sink, eps(alpha() < 255)
|
2013-06-21 15:47:38 +02:00
|
|
|
// begin grammar
|
2016-02-22 12:45:28 +01:00
|
|
|
<< "rgba("
|
|
|
|
<< color_(red()) << ','
|
|
|
|
<< color_(green()) << ','
|
|
|
|
<< color_(blue()) << ','
|
|
|
|
<< double_(alpha()/255.0) << ')'
|
|
|
|
|
|
|
|
|
"rgb("
|
|
|
|
<< color_(red()) << ','
|
|
|
|
<< color_(green()) << ','
|
|
|
|
<< color_(blue()) << ')'
|
2013-06-21 15:47:38 +02:00
|
|
|
// end grammar
|
|
|
|
);
|
|
|
|
return str;
|
2010-06-02 13:03:30 +02:00
|
|
|
}
|
2011-06-07 18:14:02 +02:00
|
|
|
|
2010-06-02 13:03:30 +02:00
|
|
|
std::string color::to_hex_string() const
|
|
|
|
{
|
2013-06-21 12:32:06 +02:00
|
|
|
namespace karma = boost::spirit::karma;
|
2014-01-26 23:00:58 +01:00
|
|
|
boost::spirit::karma::hex_type hex;
|
|
|
|
boost::spirit::karma::eps_type eps;
|
|
|
|
boost::spirit::karma::right_align_type right_align;
|
2013-06-21 12:32:06 +02:00
|
|
|
std::string str;
|
|
|
|
std::back_insert_iterator<std::string> sink(str);
|
|
|
|
karma::generate(sink,
|
|
|
|
// begin grammar
|
|
|
|
'#'
|
2016-02-22 12:45:28 +01:00
|
|
|
<< right_align(2,'0')[hex(red())]
|
|
|
|
<< right_align(2,'0')[hex(green())]
|
|
|
|
<< right_align(2,'0')[hex(blue())]
|
|
|
|
<< eps(alpha() < 255) << right_align(2,'0')[hex(alpha())]
|
2013-06-21 12:32:06 +02:00
|
|
|
// end grammar
|
|
|
|
);
|
|
|
|
return str;
|
2010-06-02 13:03:30 +02:00
|
|
|
}
|
2011-06-07 18:14:02 +02:00
|
|
|
|
2016-02-22 14:39:31 +01:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
static std::uint8_t multiply(std::uint8_t c, std::uint8_t a)
|
|
|
|
{
|
|
|
|
std::uint32_t t = c * a + 128;
|
|
|
|
return std::uint8_t(((t >> 8) + t) >> 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-01-24 01:08:59 +01:00
|
|
|
bool color::premultiply()
|
2012-10-04 22:53:58 +02:00
|
|
|
{
|
2015-01-24 01:08:59 +01:00
|
|
|
if (premultiplied_) return false;
|
2016-02-22 14:39:31 +01:00
|
|
|
if (alpha_ != 255)
|
|
|
|
{
|
|
|
|
red_ = multiply(red_, alpha_);
|
|
|
|
green_ = multiply(green_, alpha_);
|
|
|
|
blue_ = multiply(blue_, alpha_);
|
|
|
|
}
|
2015-01-24 01:08:59 +01:00
|
|
|
premultiplied_ = true;
|
|
|
|
return true;
|
2012-10-04 22:53:58 +02:00
|
|
|
}
|
|
|
|
|
2015-01-24 01:08:59 +01:00
|
|
|
bool color::demultiply()
|
2012-10-04 22:53:58 +02:00
|
|
|
{
|
2015-01-24 01:08:59 +01:00
|
|
|
if (!premultiplied_) return false;
|
2016-02-22 14:39:31 +01:00
|
|
|
if (alpha_ < 255)
|
|
|
|
{
|
|
|
|
if (alpha_ == 0)
|
|
|
|
{
|
|
|
|
red_ = green_ = blue_ = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::uint32_t r = (std::uint32_t(red_) * 255) / alpha_;
|
|
|
|
std::uint32_t g = (std::uint32_t(green_) * 255) / alpha_;
|
|
|
|
std::uint32_t b = (std::uint32_t(blue_) * 255) / alpha_;
|
|
|
|
red_ = (r > 255) ? 255 : r;
|
|
|
|
green_ = (g > 255) ? 255 : g;
|
|
|
|
blue_ = (b > 255) ? 255 : b;
|
|
|
|
}
|
|
|
|
}
|
2015-01-24 01:08:59 +01:00
|
|
|
premultiplied_ = false;
|
|
|
|
return true;
|
2012-10-04 22:53:58 +02:00
|
|
|
}
|
2009-01-14 13:10:24 +01:00
|
|
|
}
|