mapnik/include/mapnik/svg/svg_converter.hpp

384 lines
9 KiB
C++

/*****************************************************************************
*
* This file is part of Mapnik (c++ mapping toolkit)
*
* Copyright (C) 2017 Artem Pavlenko
*
* 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
*
*****************************************************************************/
#ifndef MAPNIK_SVG_CONVERTER_HPP
#define MAPNIK_SVG_CONVERTER_HPP
// mapnik
#include <mapnik/svg/svg_path_attributes.hpp>
#include <mapnik/svg/svg_path_adapter.hpp>
#include <mapnik/util/noncopyable.hpp>
#include <mapnik/safe_cast.hpp>
#pragma GCC diagnostic push
#include <mapnik/warning_ignore_agg.hpp>
#include "agg_path_storage.h"
#include "agg_conv_transform.h"
#include "agg_conv_stroke.h"
#include "agg_conv_contour.h"
#include "agg_conv_curve.h"
#include "agg_color_rgba.h"
#include "agg_bounding_rect.h"
#pragma GCC diagnostic pop
// stl
#include <deque>
#include <stdexcept>
namespace mapnik {
namespace svg {
template <typename VertexSource, typename AttributeSource>
class svg_converter : util::noncopyable
{
public:
svg_converter(VertexSource & source, AttributeSource & attributes)
: source_(source),
attributes_(attributes),
attr_stack_(),
svg_width_(0.0),
svg_height_(0.0) {}
void begin_path()
{
std::size_t idx = source_.start_new_path();
attributes_.emplace_back(cur_attr(), safe_cast<unsigned>(idx));
}
void end_path()
{
if (attributes_.empty())
{
throw std::runtime_error("end_path : The path was not begun");
}
path_attributes& attr = attributes_.back();
unsigned idx = attr.index;
attr = cur_attr();
attr.index = idx;
}
void move_to(double x, double y, bool rel=false) // M, m
{
if(rel) source_.rel_to_abs(&x, &y);
source_.move_to(x, y);
}
void line_to(double x, double y, bool rel=false) // L, l
{
if(rel) source_.rel_to_abs(&x, &y);
source_.line_to(x, y);
}
void hline_to(double x, bool rel=false) // H, h
{
double x2 = 0.0;
double y2 = 0.0;
if(source_.total_vertices())
{
source_.vertex(safe_cast<unsigned>(source_.total_vertices() - 1), &x2, &y2);
if(rel) x += x2;
source_.line_to(x, y2);
}
}
void vline_to(double y, bool rel=false) // V, v
{
double x2 = 0.0;
double y2 = 0.0;
if(source_.total_vertices())
{
source_.vertex(safe_cast<unsigned>(source_.total_vertices() - 1), &x2, &y2);
if(rel) y += y2;
source_.line_to(x2, y);
}
}
void curve3(double x1, double y1, // Q, q
double x, double y, bool rel=false)
{
if(rel)
{
source_.rel_to_abs(&x1, &y1);
source_.rel_to_abs(&x, &y);
}
source_.curve3(x1, y1, x, y);
}
void curve3(double x, double y, bool rel=false) // T, t
{
if(rel)
{
source_.curve3_rel(x, y);
} else
{
source_.curve3(x, y);
}
}
void curve4(double x1, double y1, // C, c
double x2, double y2,
double x, double y, bool rel=false)
{
if(rel)
{
source_.rel_to_abs(&x1, &y1);
source_.rel_to_abs(&x2, &y2);
source_.rel_to_abs(&x, &y);
}
source_.curve4(x1, y1, x2, y2, x, y);
}
void curve4(double x2, double y2, // S, s
double x, double y, bool rel=false)
{
if(rel)
{
source_.curve4_rel(x2, y2, x, y);
} else
{
source_.curve4(x2, y2, x, y);
}
}
void arc_to(double rx, double ry, // A, a
double angle,
bool large_arc_flag,
bool sweep_flag,
double x, double y,bool rel=false)
{
if(rel)
{
source_.arc_rel(rx, ry, angle, large_arc_flag, sweep_flag, x, y);
}
else
{
source_.arc_to(rx, ry, angle, large_arc_flag, sweep_flag, x, y);
}
}
void close_subpath() // Z, z
{
source_.end_poly(agg::path_flags_close);
}
void push_attr()
{
if (attr_stack_.empty())
attr_stack_.push_back(path_attributes());
else
attr_stack_.push_back(attr_stack_.back());
}
void pop_attr()
{
if (attr_stack_.empty())
{
throw std::runtime_error("pop_attr : Attribute stack is empty");
}
attr_stack_.pop_back();
}
// Attribute setting functions.
void fill(agg::rgba8 const& f)
{
path_attributes& attr = cur_attr();
double a = attr.fill_color.opacity();
attr.fill_color = f;
attr.fill_color.opacity(a * f.opacity());
attr.fill_flag = true;
}
void add_fill_gradient(mapnik::gradient const& grad)
{
path_attributes& attr = cur_attr();
attr.fill_gradient = grad;
}
void add_stroke_gradient(mapnik::gradient const& grad)
{
path_attributes& attr = cur_attr();
attr.stroke_gradient = grad;
}
void stroke(agg::rgba8 const& s)
{
path_attributes& attr = cur_attr();
double a = attr.stroke_color.opacity();
attr.stroke_color = s;
attr.stroke_color.opacity(a * s.opacity());
attr.stroke_flag = true;
}
void dash_array(dash_array && dash)
{
path_attributes& attr = cur_attr();
attr.dash = std::move(dash);
}
void dash_offset(double offset)
{
cur_attr().dash_offset = offset;
}
void even_odd(bool flag)
{
cur_attr().even_odd_flag = flag;
}
void visibility(bool flag)
{
cur_attr().visibility_flag = flag;
}
bool visibility()
{
return cur_attr().visibility_flag;
}
void display(bool flag)
{
cur_attr().display_flag = flag;
}
bool display()
{
return cur_attr().display_flag;
}
void stroke_width(double w)
{
cur_attr().stroke_width = w;
}
void fill_none()
{
cur_attr().fill_none = true;
cur_attr().fill_flag = false;
}
void stroke_none()
{
cur_attr().stroke_none = true;
cur_attr().stroke_flag = false;
}
void fill_opacity(double op)
{
cur_attr().fill_opacity = op;
}
void stroke_opacity(double op)
{
cur_attr().stroke_opacity = op;
}
void opacity(double op)
{
cur_attr().opacity = op;
}
void line_join(agg::line_join_e join)
{
cur_attr().line_join = join;
}
void line_cap(agg::line_cap_e cap)
{
cur_attr().line_cap = cap;
}
void miter_limit(double ml)
{
cur_attr().miter_limit = ml;
}
// Make all polygons CCW-oriented
void arrange_orientations()
{
source_.arrange_orientations_all_paths(agg::path_flags_ccw);
}
// FIXME!!!!
unsigned operator [](unsigned idx)
{
transform_ = attributes_[idx].transform;
return attributes_[idx].index;
}
void bounding_rect(double* x1, double* y1, double* x2, double* y2)
{
agg::conv_transform<mapnik::svg::svg_path_adapter> trans(source_, transform_);
agg::bounding_rect(trans, *this, 0, attributes_.size(), x1, y1, x2, y2);
}
void set_dimensions(double w, double h)
{
svg_width_ = w;
svg_height_ = h;
}
double width() const
{
return svg_width_;
}
double height() const
{
return svg_height_;
}
VertexSource & storage()
{
return source_;
}
agg::trans_affine& transform()
{
return cur_attr().transform;
}
path_attributes& cur_attr()
{
if (attr_stack_.empty())
{
throw std::runtime_error("cur_attr : Attribute stack is empty");
}
return attr_stack_.back();
}
private:
VertexSource & source_;
AttributeSource & attributes_;
AttributeSource attr_stack_;
agg::trans_affine transform_;
double svg_width_;
double svg_height_;
};
using svg_converter_type = svg_converter<svg_path_adapter, std::deque<path_attributes> >;
}}
#endif // MAPNIK_SVG_CONVERTER_HPP