summaryrefslogtreecommitdiff
path: root/src/PolySetCGALRenderer.cc
diff options
context:
space:
mode:
authorMarius Kintel <marius@kintel.net>2010-03-29 01:31:47 (GMT)
committerMarius Kintel <marius@kintel.net>2010-10-31 00:42:35 (GMT)
commit184d0e041f6279dba5d1f6348f973478ae133a3a (patch)
treedccc0d26fd5b14e4f5ca834290e7f588febfcbe4 /src/PolySetCGALRenderer.cc
parent34a8206410f049f566535b1d795b4e89950c6a6f (diff)
Initial implementation of CGALRenderer
Diffstat (limited to 'src/PolySetCGALRenderer.cc')
-rw-r--r--src/PolySetCGALRenderer.cc384
1 files changed, 384 insertions, 0 deletions
diff --git a/src/PolySetCGALRenderer.cc b/src/PolySetCGALRenderer.cc
new file mode 100644
index 0000000..acd03eb
--- /dev/null
+++ b/src/PolySetCGALRenderer.cc
@@ -0,0 +1,384 @@
+#include "PolySetCGALRenderer.h"
+#include "polyset.h"
+#include "CGALRenderer.h"
+#include "projectionnode.h"
+#include "dxflinextrudenode.h"
+#include "dxfrotextrudenode.h"
+#include "dxfdata.h"
+#include "dxftess.h"
+#include "module.h"
+
+#include "printutils.h"
+#include "export.h" // void cgal_nef3_to_polyset()
+#include "openscad.h" // get_fragments_from_r()
+
+PolySet *PolySetCGALRenderer::renderPolySet(const ProjectionNode &node, AbstractPolyNode::render_mode_e)
+{
+ CGAL_Nef_polyhedron N = this->cgalrenderer.renderCGALMesh(node);
+
+ PolySet *ps = new PolySet();
+ ps->convexity = node.convexity;
+ ps->is2d = true;
+
+ if (node.cut_mode)
+ {
+ PolySet *cube = new PolySet();
+ double infval = 1e8, eps = 0.1;
+ double x1 = -infval, x2 = +infval, y1 = -infval, y2 = +infval, z1 = 0, z2 = eps;
+
+ cube->append_poly(); // top
+ cube->append_vertex(x1, y1, z2);
+ cube->append_vertex(x2, y1, z2);
+ cube->append_vertex(x2, y2, z2);
+ cube->append_vertex(x1, y2, z2);
+
+ cube->append_poly(); // bottom
+ cube->append_vertex(x1, y2, z1);
+ cube->append_vertex(x2, y2, z1);
+ cube->append_vertex(x2, y1, z1);
+ cube->append_vertex(x1, y1, z1);
+
+ cube->append_poly(); // side1
+ cube->append_vertex(x1, y1, z1);
+ cube->append_vertex(x2, y1, z1);
+ cube->append_vertex(x2, y1, z2);
+ cube->append_vertex(x1, y1, z2);
+
+ cube->append_poly(); // side2
+ cube->append_vertex(x2, y1, z1);
+ cube->append_vertex(x2, y2, z1);
+ cube->append_vertex(x2, y2, z2);
+ cube->append_vertex(x2, y1, z2);
+
+ cube->append_poly(); // side3
+ cube->append_vertex(x2, y2, z1);
+ cube->append_vertex(x1, y2, z1);
+ cube->append_vertex(x1, y2, z2);
+ cube->append_vertex(x2, y2, z2);
+
+ cube->append_poly(); // side4
+ cube->append_vertex(x1, y2, z1);
+ cube->append_vertex(x1, y1, z1);
+ cube->append_vertex(x1, y1, z2);
+ cube->append_vertex(x1, y2, z2);
+ CGAL_Nef_polyhedron Ncube = this->cgalrenderer.renderCGALMesh(*cube);
+ cube->unlink();
+
+ // N.p3 *= CGAL_Nef_polyhedron3(CGAL_Plane(0, 0, 1, 0), CGAL_Nef_polyhedron3::INCLUDED);
+ N.p3 *= Ncube.p3;
+ if (!N.p3.is_simple()) {
+ PRINTF("WARNING: Body of projection(cut = true) isn't valid 2-manifold! Modify your design..");
+ goto cant_project_non_simple_polyhedron;
+ }
+
+ PolySet *ps3 = new PolySet();
+ cgal_nef3_to_polyset(ps3, &N);
+ Grid2d<int> conversion_grid(GRID_COARSE);
+ for (int i = 0; i < ps3->polygons.size(); i++) {
+ for (int j = 0; j < ps3->polygons[i].size(); j++) {
+ double x = ps3->polygons[i][j].x;
+ double y = ps3->polygons[i][j].y;
+ double z = ps3->polygons[i][j].z;
+ if (z != 0)
+ goto next_ps3_polygon_cut_mode;
+ if (conversion_grid.align(x, y) == i+1)
+ goto next_ps3_polygon_cut_mode;
+ conversion_grid.data(x, y) = i+1;
+ }
+ ps->append_poly();
+ for (int j = 0; j < ps3->polygons[i].size(); j++) {
+ double x = ps3->polygons[i][j].x;
+ double y = ps3->polygons[i][j].y;
+ conversion_grid.align(x, y);
+ ps->insert_vertex(x, y);
+ }
+ next_ps3_polygon_cut_mode:;
+ }
+ ps3->unlink();
+ }
+ else
+ {
+ if (!N.p3.is_simple()) {
+ PRINTF("WARNING: Body of projection(cut = false) isn't valid 2-manifold! Modify your design..");
+ goto cant_project_non_simple_polyhedron;
+ }
+
+ PolySet *ps3 = new PolySet();
+ cgal_nef3_to_polyset(ps3, &N);
+ CGAL_Nef_polyhedron np;
+ np.dim = 2;
+ for (int i = 0; i < ps3->polygons.size(); i++)
+ {
+ int min_x_p = -1;
+ double min_x_val = 0;
+ for (int j = 0; j < ps3->polygons[i].size(); j++) {
+ double x = ps3->polygons[i][j].x;
+ if (min_x_p < 0 || x < min_x_val) {
+ min_x_p = j;
+ min_x_val = x;
+ }
+ }
+ int min_x_p1 = (min_x_p+1) % ps3->polygons[i].size();
+ int min_x_p2 = (min_x_p+ps3->polygons[i].size()-1) % ps3->polygons[i].size();
+ double ax = ps3->polygons[i][min_x_p1].x - ps3->polygons[i][min_x_p].x;
+ double ay = ps3->polygons[i][min_x_p1].y - ps3->polygons[i][min_x_p].y;
+ double at = atan2(ay, ax);
+ double bx = ps3->polygons[i][min_x_p2].x - ps3->polygons[i][min_x_p].x;
+ double by = ps3->polygons[i][min_x_p2].y - ps3->polygons[i][min_x_p].y;
+ double bt = atan2(by, bx);
+
+ double eps = 0.000001;
+ if (fabs(at - bt) < eps || (fabs(ax) < eps && fabs(ay) < eps) ||
+ (fabs(bx) < eps && fabs(by) < eps)) {
+ // this triangle is degenerated in projection
+ continue;
+ }
+
+ std::list<CGAL_Nef_polyhedron2::Point> plist;
+ for (int j = 0; j < ps3->polygons[i].size(); j++) {
+ double x = ps3->polygons[i][j].x;
+ double y = ps3->polygons[i][j].y;
+ CGAL_Nef_polyhedron2::Point p = CGAL_Nef_polyhedron2::Point(x, y);
+ if (at > bt)
+ plist.push_front(p);
+ else
+ plist.push_back(p);
+ }
+ np.p2 += CGAL_Nef_polyhedron2(plist.begin(), plist.end(),
+ CGAL_Nef_polyhedron2::INCLUDED);
+ }
+ DxfData dxf(np);
+ dxf_tesselate(ps, &dxf, 0, true, false, 0);
+ dxf_border_to_ps(ps, &dxf);
+ ps3->unlink();
+ }
+
+cant_project_non_simple_polyhedron:
+
+ return ps;
+}
+
+static void add_slice(PolySet *ps, DxfData::Path *pt, double rot1, double rot2, double h1, double h2)
+{
+ for (int j = 1; j < pt->points.count(); j++)
+ {
+ int k = j - 1;
+
+ double jx1 = pt->points[j]->x * cos(rot1*M_PI/180) + pt->points[j]->y * sin(rot1*M_PI/180);
+ double jy1 = pt->points[j]->x * -sin(rot1*M_PI/180) + pt->points[j]->y * cos(rot1*M_PI/180);
+
+ double jx2 = pt->points[j]->x * cos(rot2*M_PI/180) + pt->points[j]->y * sin(rot2*M_PI/180);
+ double jy2 = pt->points[j]->x * -sin(rot2*M_PI/180) + pt->points[j]->y * cos(rot2*M_PI/180);
+
+ double kx1 = pt->points[k]->x * cos(rot1*M_PI/180) + pt->points[k]->y * sin(rot1*M_PI/180);
+ double ky1 = pt->points[k]->x * -sin(rot1*M_PI/180) + pt->points[k]->y * cos(rot1*M_PI/180);
+
+ double kx2 = pt->points[k]->x * cos(rot2*M_PI/180) + pt->points[k]->y * sin(rot2*M_PI/180);
+ double ky2 = pt->points[k]->x * -sin(rot2*M_PI/180) + pt->points[k]->y * cos(rot2*M_PI/180);
+
+ double dia1_len_sq = (jy1-ky2)*(jy1-ky2) + (jx1-kx2)*(jx1-kx2);
+ double dia2_len_sq = (jy2-ky1)*(jy2-ky1) + (jx2-kx1)*(jx2-kx1);
+
+ if (dia1_len_sq > dia2_len_sq)
+ {
+ ps->append_poly();
+ if (pt->is_inner) {
+ ps->append_vertex(kx1, ky1, h1);
+ ps->append_vertex(jx1, jy1, h1);
+ ps->append_vertex(jx2, jy2, h2);
+ } else {
+ ps->insert_vertex(kx1, ky1, h1);
+ ps->insert_vertex(jx1, jy1, h1);
+ ps->insert_vertex(jx2, jy2, h2);
+ }
+
+ ps->append_poly();
+ if (pt->is_inner) {
+ ps->append_vertex(kx2, ky2, h2);
+ ps->append_vertex(kx1, ky1, h1);
+ ps->append_vertex(jx2, jy2, h2);
+ } else {
+ ps->insert_vertex(kx2, ky2, h2);
+ ps->insert_vertex(kx1, ky1, h1);
+ ps->insert_vertex(jx2, jy2, h2);
+ }
+ }
+ else
+ {
+ ps->append_poly();
+ if (pt->is_inner) {
+ ps->append_vertex(kx1, ky1, h1);
+ ps->append_vertex(jx1, jy1, h1);
+ ps->append_vertex(kx2, ky2, h2);
+ } else {
+ ps->insert_vertex(kx1, ky1, h1);
+ ps->insert_vertex(jx1, jy1, h1);
+ ps->insert_vertex(kx2, ky2, h2);
+ }
+
+ ps->append_poly();
+ if (pt->is_inner) {
+ ps->append_vertex(jx2, jy2, h2);
+ ps->append_vertex(kx2, ky2, h2);
+ ps->append_vertex(jx1, jy1, h1);
+ } else {
+ ps->insert_vertex(jx2, jy2, h2);
+ ps->insert_vertex(kx2, ky2, h2);
+ ps->insert_vertex(jx1, jy1, h1);
+ }
+ }
+ }
+}
+
+PolySet *PolySetCGALRenderer::renderPolySet(const DxfLinearExtrudeNode &node, AbstractPolyNode::render_mode_e)
+{
+ DxfData *dxf;
+
+ if (node.filename.isEmpty())
+ {
+ CGAL_Nef_polyhedron N = this->cgalrenderer.renderCGALMesh(node);
+ dxf = new DxfData(N);
+ } else {
+ dxf = new DxfData(node.fn, node.fs, node.fa, node.filename, node.layername, node.origin_x, node.origin_y, node.scale);
+ }
+
+ PolySet *ps = new PolySet();
+ ps->convexity = node.convexity;
+
+ double h1, h2;
+
+ if (node.center) {
+ h1 = -node.height/2.0;
+ h2 = +node.height/2.0;
+ } else {
+ h1 = 0;
+ h2 = node.height;
+ }
+
+ bool first_open_path = true;
+ for (int i = 0; i < dxf->paths.count(); i++)
+ {
+ if (dxf->paths[i].is_closed)
+ continue;
+ if (first_open_path) {
+ PRINTF("WARNING: Open paths in dxf_linear_extrude(file = \"%s\", layer = \"%s\"):",
+ node.filename.toAscii().data(), node.layername.toAscii().data());
+ first_open_path = false;
+ }
+ PRINTF(" %9.5f %10.5f ... %10.5f %10.5f",
+ dxf->paths[i].points.first()->x / node.scale + node.origin_x,
+ dxf->paths[i].points.first()->y / node.scale + node.origin_y,
+ dxf->paths[i].points.last()->x / node.scale + node.origin_x,
+ dxf->paths[i].points.last()->y / node.scale + node.origin_y);
+ }
+
+
+ if (node.has_twist)
+ {
+ dxf_tesselate(ps, dxf, 0, false, true, h1);
+ dxf_tesselate(ps, dxf, node.twist, true, true, h2);
+ for (int j = 0; j < node.slices; j++)
+ {
+ double t1 = node.twist*j / node.slices;
+ double t2 = node.twist*(j+1) / node.slices;
+ double g1 = h1 + (h2-h1)*j / node.slices;
+ double g2 = h1 + (h2-h1)*(j+1) / node.slices;
+ for (int i = 0; i < dxf->paths.count(); i++)
+ {
+ if (!dxf->paths[i].is_closed)
+ continue;
+ add_slice(ps, &dxf->paths[i], t1, t2, g1, g2);
+ }
+ }
+ }
+ else
+ {
+ dxf_tesselate(ps, dxf, 0, false, true, h1);
+ dxf_tesselate(ps, dxf, 0, true, true, h2);
+ for (int i = 0; i < dxf->paths.count(); i++)
+ {
+ if (!dxf->paths[i].is_closed)
+ continue;
+ add_slice(ps, &dxf->paths[i], 0, 0, h1, h2);
+ }
+ }
+
+ delete dxf;
+
+ return ps;
+}
+
+PolySet *PolySetCGALRenderer::renderPolySet(const DxfRotateExtrudeNode &node, AbstractPolyNode::render_mode_e)
+{
+ DxfData *dxf;
+
+ if (node.filename.isEmpty())
+ {
+ CGAL_Nef_polyhedron N = this->cgalrenderer.renderCGALMesh(node);
+ dxf = new DxfData(N);
+ } else {
+ dxf = new DxfData(node.fn, node.fs, node.fa, node.filename, node.layername, node.origin_x, node.origin_y, node.scale);
+ }
+
+ PolySet *ps = new PolySet();
+ ps->convexity = node.convexity;
+
+ for (int i = 0; i < dxf->paths.count(); i++)
+ {
+ double max_x = 0;
+ for (int j = 0; j < dxf->paths[i].points.count(); j++) {
+ max_x = fmax(max_x, dxf->paths[i].points[j]->x);
+ }
+
+ int fragments = get_fragments_from_r(max_x, node.fn, node.fs, node.fa);
+
+ double points[fragments][dxf->paths[i].points.count()][3];
+
+ for (int j = 0; j < fragments; j++) {
+ double a = (j*2*M_PI) / fragments;
+ for (int k = 0; k < dxf->paths[i].points.count(); k++) {
+ if (dxf->paths[i].points[k]->x == 0) {
+ points[j][k][0] = 0;
+ points[j][k][1] = 0;
+ } else {
+ points[j][k][0] = dxf->paths[i].points[k]->x * sin(a);
+ points[j][k][1] = dxf->paths[i].points[k]->x * cos(a);
+ }
+ points[j][k][2] = dxf->paths[i].points[k]->y;
+ }
+ }
+
+ for (int j = 0; j < fragments; j++) {
+ int j1 = j + 1 < fragments ? j + 1 : 0;
+ for (int k = 0; k < dxf->paths[i].points.count(); k++) {
+ int k1 = k + 1 < dxf->paths[i].points.count() ? k + 1 : 0;
+ if (points[j][k][0] != points[j1][k][0] ||
+ points[j][k][1] != points[j1][k][1] ||
+ points[j][k][2] != points[j1][k][2]) {
+ ps->append_poly();
+ ps->append_vertex(points[j ][k ][0],
+ points[j ][k ][1], points[j ][k ][2]);
+ ps->append_vertex(points[j1][k ][0],
+ points[j1][k ][1], points[j1][k ][2]);
+ ps->append_vertex(points[j ][k1][0],
+ points[j ][k1][1], points[j ][k1][2]);
+ }
+ if (points[j][k1][0] != points[j1][k1][0] ||
+ points[j][k1][1] != points[j1][k1][1] ||
+ points[j][k1][2] != points[j1][k1][2]) {
+ ps->append_poly();
+ ps->append_vertex(points[j ][k1][0],
+ points[j ][k1][1], points[j ][k1][2]);
+ ps->append_vertex(points[j1][k ][0],
+ points[j1][k ][1], points[j1][k ][2]);
+ ps->append_vertex(points[j1][k1][0],
+ points[j1][k1][1], points[j1][k1][2]);
+ }
+ }
+ }
+ }
+
+ delete dxf;
+
+ return ps;
+}
contact: Jan Huwald // Impressum