2013-07-06 14:33:49 +00:00
|
|
|
#ifndef slic3r_ExPolygon_hpp_
|
|
|
|
#define slic3r_ExPolygon_hpp_
|
|
|
|
|
2015-12-07 23:39:54 +00:00
|
|
|
#include "libslic3r.h"
|
2013-07-14 13:53:53 +00:00
|
|
|
#include "Polygon.hpp"
|
2014-05-13 18:06:01 +00:00
|
|
|
#include "Polyline.hpp"
|
2013-07-15 18:31:43 +00:00
|
|
|
#include <vector>
|
2013-07-06 14:33:49 +00:00
|
|
|
|
2013-07-07 20:36:14 +00:00
|
|
|
namespace Slic3r {
|
|
|
|
|
2013-11-22 01:16:10 +00:00
|
|
|
class ExPolygon;
|
|
|
|
typedef std::vector<ExPolygon> ExPolygons;
|
|
|
|
|
2013-07-06 14:33:49 +00:00
|
|
|
class ExPolygon
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Polygon contour;
|
|
|
|
Polygons holes;
|
2014-01-09 16:26:39 +00:00
|
|
|
operator Points() const;
|
2013-11-21 14:12:06 +00:00
|
|
|
operator Polygons() const;
|
2016-09-30 13:23:18 +00:00
|
|
|
operator Polylines() const;
|
|
|
|
void clear() { contour.points.clear(); holes.clear(); }
|
2013-09-13 12:48:40 +00:00
|
|
|
void scale(double factor);
|
|
|
|
void translate(double x, double y);
|
2016-04-10 17:12:32 +00:00
|
|
|
void rotate(double angle);
|
2014-04-24 11:43:24 +00:00
|
|
|
void rotate(double angle, const Point ¢er);
|
2013-09-13 12:48:40 +00:00
|
|
|
double area() const;
|
2016-11-07 21:49:11 +00:00
|
|
|
bool empty() const { return contour.points.empty(); }
|
2013-09-13 12:48:40 +00:00
|
|
|
bool is_valid() const;
|
2016-09-30 13:23:18 +00:00
|
|
|
|
|
|
|
// Contains the line / polyline / polylines etc COMPLETELY.
|
2014-11-23 19:14:13 +00:00
|
|
|
bool contains(const Line &line) const;
|
|
|
|
bool contains(const Polyline &polyline) const;
|
2016-09-30 13:23:18 +00:00
|
|
|
bool contains(const Polylines &polylines) const;
|
2014-11-23 19:14:13 +00:00
|
|
|
bool contains(const Point &point) const;
|
2015-01-06 19:52:36 +00:00
|
|
|
bool contains_b(const Point &point) const;
|
|
|
|
bool has_boundary_point(const Point &point) const;
|
2016-09-30 13:23:18 +00:00
|
|
|
|
|
|
|
// Does this expolygon overlap another expolygon?
|
|
|
|
// Either the ExPolygons intersect, or one is fully inside the other,
|
|
|
|
// and it is not inside a hole of the other expolygon.
|
2016-09-30 14:11:19 +00:00
|
|
|
bool overlaps(const ExPolygon &other) const;
|
2016-09-30 13:23:18 +00:00
|
|
|
|
2015-07-06 23:17:31 +00:00
|
|
|
void simplify_p(double tolerance, Polygons* polygons) const;
|
2013-11-22 01:16:10 +00:00
|
|
|
Polygons simplify_p(double tolerance) const;
|
2013-11-21 19:25:24 +00:00
|
|
|
ExPolygons simplify(double tolerance) const;
|
2015-12-21 13:46:35 +00:00
|
|
|
void simplify(double tolerance, ExPolygons* expolygons) const;
|
2016-03-19 14:33:58 +00:00
|
|
|
void medial_axis(double max_width, double min_width, ThickPolylines* polylines) const;
|
2014-03-09 16:46:02 +00:00
|
|
|
void medial_axis(double max_width, double min_width, Polylines* polylines) const;
|
2014-04-24 11:43:24 +00:00
|
|
|
void get_trapezoids(Polygons* polygons) const;
|
|
|
|
void get_trapezoids(Polygons* polygons, double angle) const;
|
2014-04-28 18:14:20 +00:00
|
|
|
void get_trapezoids2(Polygons* polygons) const;
|
|
|
|
void get_trapezoids2(Polygons* polygons, double angle) const;
|
2014-04-24 15:06:16 +00:00
|
|
|
void triangulate(Polygons* polygons) const;
|
2014-05-01 08:37:38 +00:00
|
|
|
void triangulate_pp(Polygons* polygons) const;
|
2014-05-01 10:07:11 +00:00
|
|
|
void triangulate_p2t(Polygons* polygons) const;
|
2015-01-06 19:52:36 +00:00
|
|
|
Lines lines() const;
|
2016-05-20 04:24:05 +00:00
|
|
|
std::string dump_perl() const;
|
2013-07-06 14:33:49 +00:00
|
|
|
};
|
|
|
|
|
2016-11-07 21:49:11 +00:00
|
|
|
// Count a nuber of polygons stored inside the vector of expolygons.
|
|
|
|
// Useful for allocating space for polygons when converting expolygons to polygons.
|
|
|
|
inline size_t number_polygons(const ExPolygons &expolys)
|
|
|
|
{
|
|
|
|
size_t n_polygons = 0;
|
|
|
|
for (ExPolygons::const_iterator it = expolys.begin(); it != expolys.end(); ++ it)
|
|
|
|
n_polygons += it->holes.size() + 1;
|
|
|
|
return n_polygons;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline Lines to_lines(const ExPolygon &src)
|
|
|
|
{
|
|
|
|
size_t n_lines = src.contour.points.size();
|
|
|
|
for (size_t i = 0; i < src.holes.size(); ++ i)
|
|
|
|
n_lines += src.holes[i].points.size();
|
|
|
|
Lines lines;
|
|
|
|
lines.reserve(n_lines);
|
|
|
|
for (size_t i = 0; i <= src.holes.size(); ++ i) {
|
|
|
|
const Polygon &poly = (i == 0) ? src.contour : src.holes[i - 1];
|
|
|
|
for (Points::const_iterator it = poly.points.begin(); it != poly.points.end()-1; ++it)
|
|
|
|
lines.push_back(Line(*it, *(it + 1)));
|
|
|
|
lines.push_back(Line(poly.points.back(), poly.points.front()));
|
|
|
|
}
|
|
|
|
return lines;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline Lines to_lines(const ExPolygons &src)
|
|
|
|
{
|
|
|
|
size_t n_lines = 0;
|
|
|
|
for (ExPolygons::const_iterator it_expoly = src.begin(); it_expoly != src.end(); ++ it_expoly) {
|
|
|
|
n_lines += it_expoly->contour.points.size();
|
|
|
|
for (size_t i = 0; i < it_expoly->holes.size(); ++ i)
|
|
|
|
n_lines += it_expoly->holes[i].points.size();
|
|
|
|
}
|
|
|
|
Lines lines;
|
|
|
|
lines.reserve(n_lines);
|
|
|
|
for (ExPolygons::const_iterator it_expoly = src.begin(); it_expoly != src.end(); ++ it_expoly) {
|
|
|
|
for (size_t i = 0; i <= it_expoly->holes.size(); ++ i) {
|
|
|
|
const Points &points = ((i == 0) ? it_expoly->contour : it_expoly->holes[i - 1]).points;
|
|
|
|
for (Points::const_iterator it = points.begin(); it != points.end()-1; ++it)
|
|
|
|
lines.push_back(Line(*it, *(it + 1)));
|
|
|
|
lines.push_back(Line(points.back(), points.front()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return lines;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline Polylines to_polylines(const ExPolygon &src)
|
|
|
|
{
|
|
|
|
Polylines polylines;
|
|
|
|
polylines.assign(src.holes.size() + 1, Polyline());
|
|
|
|
size_t idx = 0;
|
|
|
|
Polyline &pl = polylines[idx ++];
|
|
|
|
pl.points = src.contour.points;
|
|
|
|
pl.points.push_back(pl.points.front());
|
|
|
|
for (Polygons::const_iterator ith = src.holes.begin(); ith != src.holes.end(); ++ith) {
|
|
|
|
Polyline &pl = polylines[idx ++];
|
|
|
|
pl.points = ith->points;
|
|
|
|
pl.points.push_back(ith->points.front());
|
|
|
|
}
|
|
|
|
assert(idx == polylines.size());
|
|
|
|
return polylines;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline Polylines to_polylines(const ExPolygons &src)
|
|
|
|
{
|
|
|
|
Polylines polylines;
|
|
|
|
polylines.assign(number_polygons(src), Polyline());
|
|
|
|
size_t idx = 0;
|
|
|
|
for (ExPolygons::const_iterator it = src.begin(); it != src.end(); ++it) {
|
|
|
|
Polyline &pl = polylines[idx ++];
|
|
|
|
pl.points = it->contour.points;
|
|
|
|
pl.points.push_back(pl.points.front());
|
|
|
|
for (Polygons::const_iterator ith = it->holes.begin(); ith != it->holes.end(); ++ith) {
|
|
|
|
Polyline &pl = polylines[idx ++];
|
|
|
|
pl.points = ith->points;
|
|
|
|
pl.points.push_back(ith->points.front());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(idx == polylines.size());
|
|
|
|
return polylines;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if SLIC3R_CPPVER >= 11
|
|
|
|
inline Polylines to_polylines(ExPolygon &&src)
|
|
|
|
{
|
|
|
|
Polylines polylines;
|
|
|
|
polylines.assign(src.holes.size() + 1, Polyline());
|
|
|
|
size_t idx = 0;
|
|
|
|
Polyline &pl = polylines[idx ++];
|
|
|
|
pl.points = std::move(src.contour.points);
|
|
|
|
pl.points.push_back(pl.points.front());
|
|
|
|
for (Polygons::const_iterator ith = src.holes.begin(); ith != src.holes.end(); ++ith) {
|
|
|
|
Polyline &pl = polylines[idx ++];
|
|
|
|
pl.points = std::move(ith->points);
|
|
|
|
pl.points.push_back(ith->points.front());
|
|
|
|
}
|
|
|
|
assert(idx == polylines.size());
|
|
|
|
return polylines;
|
|
|
|
}
|
|
|
|
inline Polylines to_polylines(ExPolygons &&src)
|
|
|
|
{
|
|
|
|
Polylines polylines;
|
|
|
|
polylines.assign(number_polygons(src), Polyline());
|
|
|
|
size_t idx = 0;
|
|
|
|
for (ExPolygons::const_iterator it = src.begin(); it != src.end(); ++it) {
|
|
|
|
Polyline &pl = polylines[idx ++];
|
|
|
|
pl.points = std::move(it->contour.points);
|
|
|
|
pl.points.push_back(pl.points.front());
|
|
|
|
for (Polygons::const_iterator ith = it->holes.begin(); ith != it->holes.end(); ++ith) {
|
|
|
|
Polyline &pl = polylines[idx ++];
|
|
|
|
pl.points = std::move(ith->points);
|
|
|
|
pl.points.push_back(ith->points.front());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(idx == polylines.size());
|
|
|
|
return polylines;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
inline Polygons to_polygons(const ExPolygon &src)
|
|
|
|
{
|
|
|
|
Polygons polygons;
|
|
|
|
polygons.reserve(src.holes.size() + 1);
|
|
|
|
polygons.push_back(src.contour);
|
|
|
|
polygons.insert(polygons.end(), src.holes.begin(), src.holes.end());
|
|
|
|
return polygons;
|
|
|
|
}
|
|
|
|
|
2016-04-10 17:12:32 +00:00
|
|
|
inline Polygons to_polygons(const ExPolygons &src)
|
|
|
|
{
|
|
|
|
Polygons polygons;
|
2016-11-07 21:49:11 +00:00
|
|
|
polygons.reserve(number_polygons(src));
|
2016-04-10 17:12:32 +00:00
|
|
|
for (ExPolygons::const_iterator it = src.begin(); it != src.end(); ++it) {
|
|
|
|
polygons.push_back(it->contour);
|
2016-11-07 21:49:11 +00:00
|
|
|
polygons.insert(polygons.end(), it->holes.begin(), it->holes.end());
|
2016-04-10 17:12:32 +00:00
|
|
|
}
|
|
|
|
return polygons;
|
|
|
|
}
|
|
|
|
|
2016-11-02 09:47:00 +00:00
|
|
|
#if SLIC3R_CPPVER >= 11
|
2016-11-07 21:49:11 +00:00
|
|
|
inline Polygons to_polygons(ExPolygon &&src)
|
|
|
|
{
|
|
|
|
Polygons polygons;
|
|
|
|
polygons.reserve(src.holes.size() + 1);
|
|
|
|
polygons.push_back(std::move(src.contour));
|
|
|
|
std::move(std::begin(src.holes), std::end(src.holes), std::back_inserter(polygons));
|
|
|
|
return polygons;
|
|
|
|
}
|
2016-04-10 17:12:32 +00:00
|
|
|
inline Polygons to_polygons(ExPolygons &&src)
|
|
|
|
{
|
|
|
|
Polygons polygons;
|
2016-11-07 21:49:11 +00:00
|
|
|
polygons.reserve(number_polygons(src));
|
2016-04-10 17:12:32 +00:00
|
|
|
for (ExPolygons::const_iterator it = src.begin(); it != src.end(); ++it) {
|
|
|
|
polygons.push_back(std::move(it->contour));
|
2016-11-07 21:49:11 +00:00
|
|
|
std::move(std::begin(it->holes), std::end(it->holes), std::back_inserter(polygons));
|
2016-04-10 17:12:32 +00:00
|
|
|
}
|
|
|
|
return polygons;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2016-11-07 21:49:11 +00:00
|
|
|
inline void polygons_append(Polygons &dst, const ExPolygon &src)
|
|
|
|
{
|
|
|
|
dst.reserve(dst.size() + src.holes.size() + 1);
|
|
|
|
dst.push_back(src.contour);
|
|
|
|
dst.insert(dst.end(), src.holes.begin(), src.holes.end());
|
2016-11-02 09:47:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline void polygons_append(Polygons &dst, const ExPolygons &src)
|
|
|
|
{
|
|
|
|
dst.reserve(dst.size() + number_polygons(src));
|
|
|
|
for (ExPolygons::const_iterator it = src.begin(); it != src.end(); ++ it) {
|
|
|
|
dst.push_back(it->contour);
|
|
|
|
dst.insert(dst.end(), it->holes.begin(), it->holes.end());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if SLIC3R_CPPVER >= 11
|
2016-11-07 21:49:11 +00:00
|
|
|
inline void polygons_append(Polygons &dst, ExPolygon &&src)
|
|
|
|
{
|
|
|
|
dst.reserve(dst.size() + src.holes.size() + 1);
|
|
|
|
dst.push_back(std::move(src.contour));
|
|
|
|
std::move(std::begin(src.holes), std::end(src.holes), std::back_inserter(dst));
|
|
|
|
}
|
|
|
|
|
2016-11-02 20:29:59 +00:00
|
|
|
inline void polygons_append(Polygons &dst, ExPolygons &&src)
|
2016-11-02 09:47:00 +00:00
|
|
|
{
|
|
|
|
dst.reserve(dst.size() + number_polygons(src));
|
2016-11-02 20:29:59 +00:00
|
|
|
for (ExPolygons::const_iterator it = src.begin(); it != src.end(); ++ it) {
|
2016-11-02 09:47:00 +00:00
|
|
|
dst.push_back(std::move(it->contour));
|
2016-11-02 20:29:59 +00:00
|
|
|
std::move(std::begin(it->holes), std::end(it->holes), std::back_inserter(dst));
|
2016-11-02 09:47:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2016-11-23 14:51:47 +00:00
|
|
|
inline void expolygons_append(ExPolygons &dst, const ExPolygons &src)
|
|
|
|
{
|
|
|
|
dst.insert(dst.end(), src.begin(), src.end());
|
|
|
|
}
|
|
|
|
|
|
|
|
#if SLIC3R_CPPVER >= 11
|
|
|
|
inline void expolygons_append(ExPolygons &dst, ExPolygons &&src)
|
|
|
|
{
|
|
|
|
std::move(std::begin(src), std::end(src), std::back_inserter(dst));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2016-11-07 21:49:11 +00:00
|
|
|
inline void expolygons_rotate(ExPolygons &expolys, double angle)
|
|
|
|
{
|
|
|
|
for (ExPolygons::iterator p = expolys.begin(); p != expolys.end(); ++p)
|
|
|
|
p->rotate(angle);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool expolygons_contain(ExPolygons &expolys, const Point &pt)
|
|
|
|
{
|
|
|
|
for (ExPolygons::iterator p = expolys.begin(); p != expolys.end(); ++p)
|
|
|
|
if (p->contains(pt))
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
extern BoundingBox get_extents(const ExPolygon &expolygon);
|
|
|
|
extern BoundingBox get_extents(const ExPolygons &expolygons);
|
2016-11-07 21:49:11 +00:00
|
|
|
extern BoundingBox get_extents_rotated(const ExPolygon &poly, double angle);
|
|
|
|
extern BoundingBox get_extents_rotated(const ExPolygons &polygons, double angle);
|
2016-11-09 14:39:12 +00:00
|
|
|
extern std::vector<BoundingBox> get_extents_vector(const ExPolygons &polygons);
|
2016-09-13 11:30:00 +00:00
|
|
|
|
2016-10-20 11:04:23 +00:00
|
|
|
extern bool remove_sticks(ExPolygon &poly);
|
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
} // namespace Slic3r
|
2013-07-07 20:36:14 +00:00
|
|
|
|
2014-04-24 11:43:24 +00:00
|
|
|
// start Boost
|
|
|
|
#include <boost/polygon/polygon.hpp>
|
|
|
|
namespace boost { namespace polygon {
|
|
|
|
template <>
|
|
|
|
struct polygon_traits<ExPolygon> {
|
|
|
|
typedef coord_t coordinate_type;
|
|
|
|
typedef Points::const_iterator iterator_type;
|
|
|
|
typedef Point point_type;
|
|
|
|
|
|
|
|
// Get the begin iterator
|
|
|
|
static inline iterator_type begin_points(const ExPolygon& t) {
|
|
|
|
return t.contour.points.begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the end iterator
|
|
|
|
static inline iterator_type end_points(const ExPolygon& t) {
|
|
|
|
return t.contour.points.end();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the number of sides of the polygon
|
|
|
|
static inline std::size_t size(const ExPolygon& t) {
|
|
|
|
return t.contour.points.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the winding direction of the polygon
|
|
|
|
static inline winding_direction winding(const ExPolygon& t) {
|
|
|
|
return unknown_winding;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template <>
|
|
|
|
struct polygon_mutable_traits<ExPolygon> {
|
|
|
|
//expects stl style iterators
|
|
|
|
template <typename iT>
|
|
|
|
static inline ExPolygon& set_points(ExPolygon& expolygon, iT input_begin, iT input_end) {
|
|
|
|
expolygon.contour.points.assign(input_begin, input_end);
|
|
|
|
// skip last point since Boost will set last point = first point
|
|
|
|
expolygon.contour.points.pop_back();
|
|
|
|
return expolygon;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
struct geometry_concept<ExPolygon> { typedef polygon_with_holes_concept type; };
|
|
|
|
|
|
|
|
template <>
|
|
|
|
struct polygon_with_holes_traits<ExPolygon> {
|
|
|
|
typedef Polygons::const_iterator iterator_holes_type;
|
|
|
|
typedef Polygon hole_type;
|
|
|
|
static inline iterator_holes_type begin_holes(const ExPolygon& t) {
|
|
|
|
return t.holes.begin();
|
|
|
|
}
|
|
|
|
static inline iterator_holes_type end_holes(const ExPolygon& t) {
|
|
|
|
return t.holes.end();
|
|
|
|
}
|
|
|
|
static inline unsigned int size_holes(const ExPolygon& t) {
|
2016-09-26 11:56:24 +00:00
|
|
|
return (int)t.holes.size();
|
2014-04-24 11:43:24 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template <>
|
|
|
|
struct polygon_with_holes_mutable_traits<ExPolygon> {
|
|
|
|
template <typename iT>
|
|
|
|
static inline ExPolygon& set_holes(ExPolygon& t, iT inputBegin, iT inputEnd) {
|
|
|
|
t.holes.assign(inputBegin, inputEnd);
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
//first we register CPolygonSet as a polygon set
|
|
|
|
template <>
|
|
|
|
struct geometry_concept<ExPolygons> { typedef polygon_set_concept type; };
|
|
|
|
|
|
|
|
//next we map to the concept through traits
|
|
|
|
template <>
|
|
|
|
struct polygon_set_traits<ExPolygons> {
|
|
|
|
typedef coord_t coordinate_type;
|
|
|
|
typedef ExPolygons::const_iterator iterator_type;
|
|
|
|
typedef ExPolygons operator_arg_type;
|
|
|
|
|
|
|
|
static inline iterator_type begin(const ExPolygons& polygon_set) {
|
|
|
|
return polygon_set.begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline iterator_type end(const ExPolygons& polygon_set) {
|
|
|
|
return polygon_set.end();
|
|
|
|
}
|
|
|
|
|
|
|
|
//don't worry about these, just return false from them
|
|
|
|
static inline bool clean(const ExPolygons& polygon_set) { return false; }
|
|
|
|
static inline bool sorted(const ExPolygons& polygon_set) { return false; }
|
|
|
|
};
|
|
|
|
|
|
|
|
template <>
|
|
|
|
struct polygon_set_mutable_traits<ExPolygons> {
|
|
|
|
template <typename input_iterator_type>
|
|
|
|
static inline void set(ExPolygons& expolygons, input_iterator_type input_begin, input_iterator_type input_end) {
|
|
|
|
expolygons.assign(input_begin, input_end);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
} }
|
|
|
|
// end Boost
|
|
|
|
|
2013-07-06 14:33:49 +00:00
|
|
|
#endif
|