2012-03-19 23:42:44 +01:00
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* This file is part of Mapnik (c++ mapping toolkit)
|
|
|
|
*
|
2014-11-20 15:25:50 +01:00
|
|
|
* Copyright (C) 2014 Artem Pavlenko
|
2012-03-19 23:42:44 +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/image_compositing.hpp>
|
|
|
|
#include <mapnik/image_data.hpp>
|
2015-01-14 19:42:30 +01:00
|
|
|
#include <mapnik/image_data_any.hpp>
|
2012-03-19 23:42:44 +01:00
|
|
|
|
2012-03-21 16:45:23 +01:00
|
|
|
// boost
|
2014-10-22 01:37:27 +02:00
|
|
|
#pragma GCC diagnostic push
|
|
|
|
#pragma GCC diagnostic ignored "-Wunused-local-typedef"
|
|
|
|
#pragma GCC diagnostic ignored "-Wredeclared-class-member"
|
2012-03-21 16:45:23 +01:00
|
|
|
#include <boost/assign/list_of.hpp>
|
|
|
|
#include <boost/bimap.hpp>
|
2014-10-22 01:37:27 +02:00
|
|
|
#pragma GCC diagnostic pop
|
2012-03-21 16:45:23 +01:00
|
|
|
|
2012-03-19 23:42:44 +01:00
|
|
|
// agg
|
|
|
|
#include "agg_rendering_buffer.h"
|
|
|
|
#include "agg_rasterizer_scanline_aa.h"
|
|
|
|
#include "agg_scanline_u.h"
|
|
|
|
#include "agg_renderer_scanline.h"
|
|
|
|
#include "agg_pixfmt_rgba.h"
|
2014-12-02 15:56:40 +01:00
|
|
|
#include "agg_pixfmt_gray.h"
|
2013-07-24 00:45:25 +02:00
|
|
|
#include "agg_color_rgba.h"
|
2012-03-19 23:42:44 +01:00
|
|
|
|
2014-12-02 15:56:40 +01:00
|
|
|
|
2012-03-19 23:42:44 +01:00
|
|
|
namespace mapnik
|
|
|
|
{
|
|
|
|
|
2014-07-07 19:23:15 +02:00
|
|
|
using comp_op_lookup_type = boost::bimap<composite_mode_e, std::string>;
|
2012-03-21 16:45:23 +01:00
|
|
|
static const comp_op_lookup_type comp_lookup = boost::assign::list_of<comp_op_lookup_type::relation>
|
|
|
|
(clear,"clear")
|
|
|
|
(src,"src")
|
|
|
|
(dst,"dst")
|
2012-04-23 14:02:02 +02:00
|
|
|
(src_over,"src-over")
|
|
|
|
(dst_over,"dst-over")
|
|
|
|
(src_in,"src-in")
|
|
|
|
(dst_in,"dst-in")
|
|
|
|
(src_out,"src-out")
|
|
|
|
(dst_out,"dst-out")
|
|
|
|
(src_atop,"src-atop")
|
|
|
|
(dst_atop,"dst-atop")
|
2012-03-21 16:45:23 +01:00
|
|
|
(_xor,"xor")
|
|
|
|
(plus,"plus")
|
|
|
|
(minus,"minus")
|
|
|
|
(multiply,"multiply")
|
|
|
|
(screen,"screen")
|
|
|
|
(overlay,"overlay")
|
|
|
|
(darken,"darken")
|
|
|
|
(lighten,"lighten")
|
2012-04-23 14:02:02 +02:00
|
|
|
(color_dodge,"color-dodge")
|
|
|
|
(color_burn,"color-burn")
|
|
|
|
(hard_light,"hard-light")
|
|
|
|
(soft_light,"soft-light")
|
2012-03-21 16:45:23 +01:00
|
|
|
(difference,"difference")
|
|
|
|
(exclusion,"exclusion")
|
|
|
|
(contrast,"contrast")
|
|
|
|
(invert,"invert")
|
2012-05-25 11:35:43 +02:00
|
|
|
(invert_rgb,"invert-rgb")
|
|
|
|
(grain_merge,"grain-merge")
|
2012-05-29 16:09:33 +02:00
|
|
|
(grain_extract,"grain-extract")
|
2012-06-17 21:19:29 +02:00
|
|
|
(hue,"hue")
|
|
|
|
(saturation,"saturation")
|
|
|
|
(_color,"color")
|
2012-06-18 20:30:01 +02:00
|
|
|
(_value,"value")
|
2014-08-04 13:33:37 +02:00
|
|
|
(linear_dodge,"linear-dodge")
|
|
|
|
(linear_burn,"linear-burn")
|
|
|
|
(divide,"divide")
|
2012-05-25 11:35:43 +02:00
|
|
|
;
|
2012-03-21 16:45:23 +01:00
|
|
|
|
2012-04-18 16:37:14 +02:00
|
|
|
boost::optional<composite_mode_e> comp_op_from_string(std::string const& name)
|
2012-03-21 16:45:23 +01:00
|
|
|
{
|
2012-04-18 16:37:14 +02:00
|
|
|
boost::optional<composite_mode_e> mode;
|
2012-03-21 16:45:23 +01:00
|
|
|
comp_op_lookup_type::right_const_iterator right_iter = comp_lookup.right.find(name);
|
|
|
|
if (right_iter != comp_lookup.right.end())
|
|
|
|
{
|
2012-04-18 16:37:14 +02:00
|
|
|
mode.reset(right_iter->second);
|
2012-03-21 16:45:23 +01:00
|
|
|
}
|
2012-04-18 16:37:14 +02:00
|
|
|
return mode;
|
2012-03-21 16:45:23 +01:00
|
|
|
}
|
|
|
|
|
2012-07-06 02:06:41 +02:00
|
|
|
boost::optional<std::string> comp_op_to_string(composite_mode_e comp_op)
|
|
|
|
{
|
|
|
|
boost::optional<std::string> mode;
|
|
|
|
comp_op_lookup_type::left_const_iterator left_iter = comp_lookup.left.find(comp_op);
|
|
|
|
if (left_iter != comp_lookup.left.end())
|
|
|
|
{
|
|
|
|
mode.reset(left_iter->second);
|
|
|
|
}
|
|
|
|
return mode;
|
|
|
|
}
|
|
|
|
|
2012-10-01 23:13:54 +02:00
|
|
|
/*
|
|
|
|
Note: the difference between agg::pixfmt_rgba32 and agg:pixfmt_rgba32_pre is subtle.
|
|
|
|
|
|
|
|
From http://www.antigrain.com/news/release_notes/v22.agdoc.html:
|
|
|
|
|
|
|
|
Format agg::pixfmt_rgba32 is the main and the fastest pixel format and it's supposed to be used in most cases. But it always uses plain colors as input and produces pre-multiplied result on the canvas. It has even less number of calculations than agg::pixfmt_rgba32_pre. Format agg::pixfmt_rgba32_plain is slow because of division operations. APIs allowing for alpha-blending require premultiplied colors. Besides, if you display RGBA with RGB API (that is, without alpha, like WinAPI BitBlt), the colors still must be premultiplied. Note that the formulas in agg::pixfmt_rgba32 and agg::pixfmt_rgb24 are exactly the same! So, premultiplied colors are more natural and agg::pixfmt_rgba32_plain is rather useless.
|
|
|
|
|
2014-07-07 19:23:15 +02:00
|
|
|
Format agg::pixfmt_rgba32_pre is a bit slower than agg::pixfmt_rgba32 because of additional "cover" values, i.e. secondary alphas, that are to be mixed with the source premultiplied color. That spoils the beauty of the premultiplied colors idea. But the "cover" values are important because there can be other color spaces and color types that don't have any "alpha" at all, or the alpha is incompatible with integral types. So, the "cover" is a secondary, uniform alpha in range of 0…255, used specifically for anti-aliasing purposes.
|
|
|
|
One needs to consider this issue when transforming images. Actually, all RGBA images are supposed to be in the premultiplied color space and the result of filtering is also premultiplied. Since the resulting colors of the filtered images are the source for the renderers, one should use the premultiplied renderers, that is, agg::pixfmt_rgba32_pre, or the new one, agg::pixfmt_rgb24_pre. But it's important only if images are translucent, that is, have actual alpha channel.
|
2012-10-01 23:13:54 +02:00
|
|
|
|
|
|
|
For example, if you generate some pattern with AGG (premultiplied) and would like to use it for filling, you'll need to use agg::pixfmt_rgba32_pre. If you use agg::span_image_filter_rgb24_gamma_bilinear (that is, RGB for input) and draw it on the RGBA canvas, you still need to use agg::pixfmt_rgba32_pre as the destination canvas. The only thing you need is to premultiply the background color used out of bounds.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2015-01-08 12:42:38 +01:00
|
|
|
namespace detail {
|
|
|
|
|
|
|
|
// non-mutable rendering_buffer implementation
|
|
|
|
template <typename T>
|
|
|
|
struct rendering_buffer
|
|
|
|
{
|
|
|
|
using image_data_type = T;
|
|
|
|
using pixel_type = typename image_data_type::pixel_type;
|
|
|
|
using row_data = agg::const_row_info<uint8_t>;
|
|
|
|
|
|
|
|
rendering_buffer(T const& data)
|
|
|
|
: data_(data) {}
|
|
|
|
|
|
|
|
uint8_t const* buf() const { return data_.getBytes(); }
|
|
|
|
unsigned width() const { return data_.width();}
|
|
|
|
unsigned height() const { return data_.height();}
|
|
|
|
int stride() const { return data_.width() * sizeof(pixel_type);}
|
|
|
|
uint8_t const* row_ptr(int, int y, unsigned) {return row_ptr(y);}
|
|
|
|
uint8_t const* row_ptr(int y) const { return reinterpret_cast<uint8_t const*>(data_.getRow(y)); }
|
|
|
|
row_data row (int y) const { return row_data(0, data_.width() - 1, row_ptr(y)); }
|
|
|
|
image_data_type const& data_;
|
|
|
|
};
|
|
|
|
|
2015-01-14 19:42:30 +01:00
|
|
|
} // end detail ns
|
2015-01-08 12:42:38 +01:00
|
|
|
|
2014-12-12 11:39:36 +01:00
|
|
|
template <>
|
2015-01-08 12:42:38 +01:00
|
|
|
MAPNIK_DECL void composite(image_data_rgba8 & dst, image_data_rgba8 const& src, composite_mode_e mode,
|
2012-05-12 00:19:05 +02:00
|
|
|
float opacity,
|
2012-05-12 00:31:35 +02:00
|
|
|
int dx,
|
2015-01-14 19:42:30 +01:00
|
|
|
int dy)
|
2012-03-19 23:42:44 +01:00
|
|
|
{
|
2014-07-07 19:23:15 +02:00
|
|
|
using color = agg::rgba8;
|
|
|
|
using order = agg::order_rgba;
|
2015-01-08 12:42:38 +01:00
|
|
|
using const_rendering_buffer = detail::rendering_buffer<image_data_rgba8>;
|
2014-07-07 19:23:15 +02:00
|
|
|
using blender_type = agg::comp_op_adaptor_rgba_pre<color, order>;
|
|
|
|
using pixfmt_type = agg::pixfmt_custom_blend_rgba<blender_type, agg::rendering_buffer>;
|
|
|
|
using renderer_type = agg::renderer_base<pixfmt_type>;
|
2012-05-07 13:15:04 +02:00
|
|
|
|
2012-06-19 00:50:32 +02:00
|
|
|
agg::rendering_buffer dst_buffer(dst.getBytes(),dst.width(),dst.height(),dst.width() * 4);
|
2015-01-08 12:42:38 +01:00
|
|
|
const_rendering_buffer src_buffer(src);
|
2012-06-19 00:50:32 +02:00
|
|
|
pixfmt_type pixf(dst_buffer);
|
2012-03-21 16:45:23 +01:00
|
|
|
pixf.comp_op(static_cast<agg::comp_op_e>(mode));
|
2015-01-08 12:42:38 +01:00
|
|
|
agg::pixfmt_alpha_blend_rgba<agg::blender_rgba32, const_rendering_buffer, agg::pixel32_type> pixf_mask(src_buffer);
|
2015-01-17 01:26:56 +01:00
|
|
|
#ifdef MAPNIK_DEBUG
|
2015-01-15 01:29:00 +01:00
|
|
|
if (!src.get_premultiplied())
|
|
|
|
{
|
2015-01-22 00:09:53 +01:00
|
|
|
abort();
|
2015-01-15 01:29:00 +01:00
|
|
|
throw std::runtime_error("SOURCE MUST BE PREMULTIPLIED FOR COMPOSITING!");
|
|
|
|
}
|
|
|
|
if (!dst.get_premultiplied())
|
|
|
|
{
|
|
|
|
throw std::runtime_error("DESTINATION MUST BE PREMULTIPLIED FOR COMPOSITING!");
|
|
|
|
}
|
2015-01-17 01:26:56 +01:00
|
|
|
#endif
|
2012-03-19 23:42:44 +01:00
|
|
|
renderer_type ren(pixf);
|
2012-06-19 14:43:15 +02:00
|
|
|
ren.blend_from(pixf_mask,0,dx,dy,unsigned(255*opacity));
|
2012-05-11 11:55:58 +02:00
|
|
|
}
|
|
|
|
|
2014-12-12 11:39:36 +01:00
|
|
|
template <>
|
2015-01-08 12:42:38 +01:00
|
|
|
MAPNIK_DECL void composite(image_data_gray32f & dst, image_data_gray32f const& src, composite_mode_e mode,
|
2014-12-02 15:56:40 +01:00
|
|
|
float opacity,
|
|
|
|
int dx,
|
2015-01-14 19:42:30 +01:00
|
|
|
int dy)
|
2014-12-02 15:56:40 +01:00
|
|
|
{
|
2015-01-08 12:42:38 +01:00
|
|
|
using const_rendering_buffer = detail::rendering_buffer<image_data_gray32f>;
|
|
|
|
using src_pixfmt_type = agg::pixfmt_alpha_blend_gray<agg::blender_gray<agg::gray32>, const_rendering_buffer, 1, 0>;
|
|
|
|
using dst_pixfmt_type = agg::pixfmt_alpha_blend_gray<agg::blender_gray<agg::gray32>, agg::rendering_buffer, 1, 0>;
|
|
|
|
using renderer_type = agg::renderer_base<dst_pixfmt_type>;
|
2014-12-02 15:56:40 +01:00
|
|
|
|
|
|
|
agg::rendering_buffer dst_buffer(dst.getBytes(),dst.width(),dst.height(),dst.width());
|
2015-01-08 12:42:38 +01:00
|
|
|
const_rendering_buffer src_buffer(src);
|
|
|
|
dst_pixfmt_type pixf(dst_buffer);
|
|
|
|
src_pixfmt_type pixf_mask(src_buffer);
|
2014-12-02 15:56:40 +01:00
|
|
|
renderer_type ren(pixf);
|
2015-01-08 12:42:38 +01:00
|
|
|
ren.copy_from(pixf_mask,0,dx,dy);
|
2014-12-02 15:56:40 +01:00
|
|
|
}
|
|
|
|
|
2015-01-14 19:42:30 +01:00
|
|
|
namespace detail {
|
|
|
|
|
|
|
|
struct composite_visitor
|
|
|
|
{
|
|
|
|
composite_visitor(image_data_any const& src,
|
|
|
|
composite_mode_e mode,
|
|
|
|
float opacity,
|
|
|
|
int dx,
|
|
|
|
int dy)
|
|
|
|
: src_(src),
|
|
|
|
mode_(mode),
|
|
|
|
opacity_(opacity),
|
|
|
|
dx_(dx),
|
|
|
|
dy_(dy) {}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
void operator() (T & dst);
|
|
|
|
|
|
|
|
private:
|
|
|
|
image_data_any const& src_;
|
|
|
|
composite_mode_e mode_;
|
|
|
|
float opacity_;
|
|
|
|
int dx_;
|
|
|
|
int dy_;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
void composite_visitor::operator() (T & dst)
|
|
|
|
{
|
|
|
|
throw std::runtime_error("Error: Composite with " + std::string(typeid(dst).name()) + " is not supported");
|
|
|
|
}
|
|
|
|
|
|
|
|
template <>
|
|
|
|
void composite_visitor::operator()<image_data_rgba8> (image_data_rgba8 & dst)
|
|
|
|
{
|
|
|
|
composite(dst, util::get<image_data_rgba8>(src_), mode_, opacity_, dx_, dy_);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <>
|
|
|
|
void composite_visitor::operator()<image_data_gray32f> (image_data_gray32f & dst)
|
|
|
|
{
|
|
|
|
composite(dst, util::get<image_data_gray32f>(src_), mode_, opacity_, dx_, dy_);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // end ns
|
|
|
|
|
|
|
|
template <>
|
|
|
|
MAPNIK_DECL void composite(image_data_any & dst, image_data_any const& src, composite_mode_e mode,
|
|
|
|
float opacity,
|
|
|
|
int dx,
|
|
|
|
int dy)
|
|
|
|
{
|
|
|
|
util::apply_visitor(detail::composite_visitor(src, mode, opacity, dx, dy), dst);
|
|
|
|
}
|
|
|
|
|
2012-03-19 23:42:44 +01:00
|
|
|
}
|