openscad/src/Polygon2d.cc

115 lines
3.0 KiB
C++

#include "Polygon2d.h"
#include "printutils.h"
#include <boost/foreach.hpp>
/*!
Class for holding 2D geometry.
This class will hold 2D geometry consisting of a number of closed
contours. A polygon can contain holes and islands, as well as
intersecting contours.
We can store sanitized vs. unsanitized polygons. Sanitized polygons
will have opposite winding order for holes and is guaranteed to not
have intersecting geometry. The winding order will be counter-clockwise
for positive outlines and clockwise for holes. Sanitization is typically
done by ClipperUtils, but if you create geometry which you know is sanitized,
the flag can be set manually.
*/
size_t Polygon2d::memsize() const
{
size_t mem = 0;
BOOST_FOREACH(const Outline2d &o, this->outlines()) {
mem += o.vertices.size() * sizeof(Vector2d) + sizeof(Outline2d);
}
mem += sizeof(Polygon2d);
return mem;
}
BoundingBox Polygon2d::getBoundingBox() const
{
BoundingBox bbox;
BOOST_FOREACH(const Outline2d &o, this->outlines()) {
BOOST_FOREACH(const Vector2d &v, o.vertices) {
bbox.extend(Vector3d(v[0], v[1], 0));
}
}
return bbox;
}
std::string Polygon2d::dump() const
{
std::stringstream out;
BOOST_FOREACH(const Outline2d &o, this->theoutlines) {
out << "contour:\n";
BOOST_FOREACH(const Vector2d &v, o.vertices) {
out << " " << v.transpose();
}
out << "\n";
}
return out.str();
}
bool Polygon2d::isEmpty() const
{
return this->theoutlines.empty();
}
void Polygon2d::transform(const Transform2d &mat)
{
if (mat.matrix().determinant() == 0) {
PRINT("WARNING: Scaling a 2D object with 0 - removing object");
this->theoutlines.clear();
return;
}
BOOST_FOREACH(Outline2d &o, this->theoutlines) {
BOOST_FOREACH(Vector2d &v, o.vertices) {
v = mat * v;
}
}
}
void Polygon2d::resize(Vector2d newsize, const Eigen::Matrix<bool,2,1> &autosize)
{
BoundingBox bbox = this->getBoundingBox();
// Find largest dimension
int maxdim = (newsize[1] && newsize[1] > newsize[0]) ? 1 : 0;
// Default scale (scale with 1 if the new size is 0)
Vector2d scale(newsize[0] > 0 ? newsize[0] / bbox.sizes()[0] : 1,
newsize[1] > 0 ? newsize[1] / bbox.sizes()[1] : 1);
// Autoscale where applicable
double autoscale = newsize[maxdim] > 0 ? newsize[maxdim] / bbox.sizes()[maxdim] : 1;
Vector2d newscale(!autosize[0] || (newsize[0] > 0) ? scale[0] : autoscale,
!autosize[1] || (newsize[1] > 0) ? scale[1] : autoscale);
Transform2d t;
t.matrix() <<
newscale[0], 0, 0,
0, newscale[1], 0,
0, 0, 1;
this->transform(t);
}
bool Polygon2d::is_convex() const {
if (theoutlines.size() > 1) return false;
if (theoutlines.empty()) return true;
std::vector<Vector2d> const& pts = theoutlines[0].vertices;
int N = pts.size();
// Check for a right turn. This assumes the polygon is simple.
for (int i = 0; i < N; i++) {
Vector2d d1 = pts[(i+1)%N] - pts[i];
Vector2d d2 = pts[(i+2)%N] - pts[(i+1)%N];
double zcross = d1[0]*d2[1]-d1[1]*d2[0];
if (zcross < 0) return false;
}
return true;
}