openscad/src/Polygon2d.cc

113 lines
2.9 KiB
C++
Raw Normal View History

#include "Polygon2d.h"
2014-01-28 10:16:02 +04:00
#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
2014-11-28 04:38:53 +03:00
have intersecting geometry. 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)
{
2014-01-28 10:16:02 +04:00
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;
}
}
}
2013-12-17 10:35:12 +04:00
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;
}