Skip to content

Commit

Permalink
Add Buffer3D Algorithm
Browse files Browse the repository at this point in the history
  • Loading branch information
lbartoletti committed Sep 13, 2024
1 parent 0942722 commit 3a49384
Show file tree
Hide file tree
Showing 13 changed files with 14,459 additions and 0 deletions.
17 changes: 17 additions & 0 deletions src/PolyhedralSurface.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,23 @@ PolyhedralSurface::PolyhedralSurface(const std::vector<Polygon> &polygons)
}
}

PolyhedralSurface::PolyhedralSurface(const std::unique_ptr<Geometry> &geometry)
{
if (geometry->is<PolyhedralSurface>()) {
*this = static_cast<const PolyhedralSurface &>(*geometry);
} else if (geometry->is<TriangulatedSurface>()) {
const TriangulatedSurface &triangulatedSurface =
geometry->as<TriangulatedSurface>();
for (size_t i = 0; i < triangulatedSurface.numTriangles(); ++i) {
this->addPolygon(triangulatedSurface.triangleN(i));
}
} else if (geometry->is<Polygon>()) {
this->addPolygon(geometry->as<Polygon>());
} else {
throw std::invalid_argument("Cannot convert geometry to PolyhedralSurface");
}
}

///
///
///
Expand Down
6 changes: 6 additions & 0 deletions src/PolyhedralSurface.h
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,12 @@ class SFCGAL_API PolyhedralSurface : public Surface {
* Constructor with a vector of polygons
*/
PolyhedralSurface(const std::vector<Polygon> &polygons);

/**
* Constructor with a Geometry
*/
PolyhedralSurface(const std::unique_ptr<Geometry> &geometry);

/**
* Constructor from a Polyhedron (detail::MarkedPolyhedron or
* CGAL::Polyhedron_3)
Expand Down
341 changes: 341 additions & 0 deletions src/algorithm/buffer3D.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,341 @@
#include "SFCGAL/algorithm/buffer3D.h"
#include "SFCGAL/Cylinder.h"
#include "SFCGAL/Kernel.h"
#include "SFCGAL/PolyhedralSurface.h"
#include "SFCGAL/Sphere.h"
#include "SFCGAL/algorithm/minkowskiSum3D.h"
#include "SFCGAL/algorithm/union.h"
#include "SFCGAL/detail/GeometrySet.h"
#include "SFCGAL/numeric.h"
#include "SFCGAL/triangulate/triangulatePolygon.h"
#include <CGAL/Nef_polyhedron_3.h>
#include <CGAL/Polygon_mesh_processing/corefinement.h>
#include <CGAL/Polygon_mesh_processing/repair.h>
#include <CGAL/Polygon_mesh_processing/triangulate_faces.h>
#include <CGAL/boost/graph/copy_face_graph.h>
#include <CGAL/minkowski_sum_3.h>

namespace PMP = CGAL::Polygon_mesh_processing;

namespace SFCGAL {
namespace algorithm {

Buffer3D::Buffer3D(const Geometry &inputGeometry, double radius, int segments)
: _radius(radius), _segments(segments)
{
if (inputGeometry.is<Point>()) {
_inputPoints.push_back(inputGeometry.as<Point>().toPoint_3());
} else if (inputGeometry.is<LineString>()) {
const LineString &ls = inputGeometry.as<LineString>();
for (size_t i = 0; i < ls.numPoints(); ++i) {
_inputPoints.push_back(ls.pointN(i).toPoint_3());
}
} else {
throw std::invalid_argument("Input geometry must be a Point or LineString");
}
}

std::unique_ptr<PolyhedralSurface>
Buffer3D::compute(BufferType type) const
{
if (_inputPoints.size() == 1) {
return computePointBuffer();
}

switch (type) {
case ROUND:
return computeRoundBuffer();
case CYLSPHERE:
return computeCylSphereBuffer();
case FLAT:
return computeFlatBuffer();
default:
throw std::invalid_argument("Invalid buffer type");
}
}

std::unique_ptr<PolyhedralSurface>
Buffer3D::computePointBuffer() const
{
Kernel::Point_3 center(_inputPoints[0].x(), _inputPoints[0].y(),
_inputPoints[0].z());
Sphere sphere(_radius, center, _segments / 2, _segments);
return std::make_unique<PolyhedralSurface>(sphere.generatePolyhedron());
}

std::unique_ptr<PolyhedralSurface>
Buffer3D::computeRoundBuffer() const
{
typedef Kernel::Point_3 Point_3;
typedef Point_3 *point_iterator;
typedef std::pair<point_iterator, point_iterator> point_range;
typedef std::list<point_range> polyline;
typedef CGAL::Nef_polyhedron_3<Kernel> Nef_polyhedron;

// Create sphere
Point_3 center(0, 0, 0);
SFCGAL::Sphere sphere(_radius, center, _segments / 2, _segments);

// Generate polyhedron from sphere
CGAL::Polyhedron_3<Kernel> spherePolyhedron = sphere.generatePolyhedron();

// Convert Polyhedron to a Nef_polyhedron
Nef_polyhedron N0(spherePolyhedron);

// Create a polyline from _inputPoints
polyline poly;
if (!_inputPoints.empty()) {
// Create a non-const copy of the points
std::vector<Point_3> points_copy(_inputPoints.begin(), _inputPoints.end());
poly.push_back(point_range(&points_copy.front(), &points_copy.back() + 1));

// Create a Nef_polyhedron from the polyline
Nef_polyhedron N1(poly.begin(), poly.end(),
Nef_polyhedron::Polylines_tag());

// Perform Minkowski sum
Nef_polyhedron result = CGAL::minkowski_sum_3(N0, N1);

// Convert result to SFCGAL::PolyhedralSurface
SFCGAL::detail::MarkedPolyhedron out;
result.convert_to_polyhedron(out);
return std::make_unique<PolyhedralSurface>(out);
} else {
// If _inputPoints is empty, return an empty PolyhedralSurface
return std::make_unique<PolyhedralSurface>();
}
}

std::unique_ptr<PolyhedralSurface>
Buffer3D::computeCylSphereBuffer() const
{
typedef CGAL::Nef_polyhedron_3<Kernel> Nef_polyhedron;
Nef_polyhedron result;
bool is_first = true;

// Add a sphere at the first point of the line
if (!_inputPoints.empty()) {
Kernel::Point_3 start_sphere_center(
_inputPoints[0].x(), _inputPoints[0].y(), _inputPoints[0].z());
Kernel::Vector_3 start_direction;
if (_inputPoints.size() > 1) {
start_direction = _inputPoints[1] - _inputPoints[0];
} else {
start_direction =
Kernel::Vector_3(0, 0, 1); // Default direction if only one point
}

Sphere start_sphere(_radius, start_sphere_center, _segments / 2, _segments,
start_direction);
CGAL::Polyhedron_3<Kernel> start_sphere_poly =
start_sphere.generatePolyhedron();
Nef_polyhedron start_sphere_nef(start_sphere_poly);

result = start_sphere_nef;
is_first = false;
}

// Create a cylinder and spheres for each segment of the line
for (size_t i = 0; i < _inputPoints.size() - 1; ++i) {
// Create a cylinder between each successive point
Kernel::Vector_3 axis(_inputPoints[i + 1].x() - _inputPoints[i].x(),
_inputPoints[i + 1].y() - _inputPoints[i].y(),
_inputPoints[i + 1].z() - _inputPoints[i].z());
Kernel::FT height = CGAL::sqrt(CGAL::to_double(axis.squared_length()));
Kernel::Point_3 base(_inputPoints[i].x(), _inputPoints[i].y(),
_inputPoints[i].z());
Cylinder cyl(base, axis, _radius, height, _segments);
CGAL::Polyhedron_3<Kernel> cyl_poly = cyl.generatePolyhedron();
Nef_polyhedron cyl_nef(cyl_poly);

result = result.join(cyl_nef);

// Add a sphere at the junctions (rounded corners)
if (i < _inputPoints.size() - 1) {
Kernel::Point_3 sphereCenter(_inputPoints[i + 1].x(),
_inputPoints[i + 1].y(),
_inputPoints[i + 1].z());
Kernel::Vector_3 sphere_direction;

if (i < _inputPoints.size() - 2) {
// For intermediate points, use the bisector of the two adjacent
// segments
Kernel::Vector_3 prev_dir = _inputPoints[i + 1] - _inputPoints[i];
Kernel::Vector_3 next_dir = _inputPoints[i + 2] - _inputPoints[i + 1];
prev_dir =
prev_dir / CGAL::sqrt(CGAL::to_double(prev_dir.squared_length()));
next_dir =
next_dir / CGAL::sqrt(CGAL::to_double(next_dir.squared_length()));
sphere_direction = prev_dir + next_dir;
} else {
// For the last point, use the direction of the last segment
sphere_direction = _inputPoints[i + 1] - _inputPoints[i];
}
sphere_direction =
sphere_direction /
CGAL::sqrt(CGAL::to_double(sphere_direction.squared_length()));

Sphere sphere(_radius, sphereCenter, _segments / 2, _segments,
sphere_direction);
CGAL::Polyhedron_3<Kernel> sphere_poly = sphere.generatePolyhedron();
Nef_polyhedron sphere_nef(sphere_poly);
result = result.join(sphere_nef);
}
}

// Convert the Nef_polyhedron to Polyhedron_3
CGAL::Polyhedron_3<Kernel> merged_mesh;
result.convert_to_polyhedron(merged_mesh);

// Clean up the geometry
PMP::remove_connected_components_of_negligible_size(merged_mesh);

// Convert the merged mesh to PolyhedralSurface and return
auto resultSurface = std::make_unique<PolyhedralSurface>();
resultSurface->addPolygons(merged_mesh);

return resultSurface;
}

std::unique_ptr<PolyhedralSurface>
Buffer3D::computeFlatBuffer() const
{
std::vector<Kernel::Point_3> line_points;
for (const auto &p : _inputPoints) {
line_points.emplace_back(p.x(), p.y(), p.z());
}

Cylinder::Surface_mesh buffer;
std::vector<std::vector<Cylinder::Surface_mesh::Vertex_index>> rings;

std::vector<Kernel::Plane_3> bisector_planes;
for (size_t i = 1; i < line_points.size() - 1; ++i) {
bisector_planes.push_back(compute_bisector_plane(
line_points[i - 1], line_points[i], line_points[i + 1]));
}

for (size_t i = 0; i < line_points.size() - 1; ++i) {
Kernel::Vector_3 axis =
normalizeVector(line_points[i + 1] - line_points[i]);
Kernel::Point_3 extended_start =
extend_point(line_points[i], -axis, _radius);
Kernel::Point_3 extended_end =
extend_point(line_points[i + 1], axis, _radius);

std::vector<Kernel::Point_3> start_circle =
create_circle_points(extended_start, axis, _radius, _segments);
std::vector<Kernel::Point_3> end_circle =
create_circle_points(extended_end, axis, _radius, _segments);

std::vector<Cylinder::Surface_mesh::Vertex_index> start_ring, end_ring;

for (size_t j = 0; j < _segments; ++j) {
Kernel::Point_3 start_point = start_circle[j];
Kernel::Point_3 end_point = end_circle[j];

if (i > 0) {
start_point = intersect_segment_plane(start_point, end_point,
bisector_planes[i - 1]);
}
if (i < line_points.size() - 2) {
end_point =
intersect_segment_plane(start_point, end_point, bisector_planes[i]);
}

Cylinder::Surface_mesh::Vertex_index v1 = buffer.add_vertex(start_point);
Cylinder::Surface_mesh::Vertex_index v2 = buffer.add_vertex(end_point);
start_ring.push_back(v1);
end_ring.push_back(v2);

if (j > 0) {
buffer.add_face(start_ring[j - 1], start_ring[j], end_ring[j],
end_ring[j - 1]);
}
}
buffer.add_face(start_ring.back(), start_ring.front(), end_ring.front(),
end_ring.back());

rings.push_back(start_ring);
if (i == line_points.size() - 2)
rings.push_back(end_ring);
}

// Add caps
Kernel::Point_3 start_center = extend_point(
line_points.front(),
normalizeVector(line_points[1] - line_points.front()), -_radius);
Kernel::Point_3 end_center = extend_point(
line_points.back(),
normalizeVector(line_points.back() - line_points[line_points.size() - 2]),
_radius);
Cylinder::Surface_mesh::Vertex_index start_center_index =
buffer.add_vertex(start_center);
Cylinder::Surface_mesh::Vertex_index end_center_index =
buffer.add_vertex(end_center);

for (size_t i = 0; i < _segments; ++i) {
buffer.add_face(start_center_index, rings.front()[(i + 1) % _segments],
rings.front()[i]);
buffer.add_face(end_center_index, rings.back()[i],
rings.back()[(i + 1) % _segments]);
}

return std::make_unique<PolyhedralSurface>(buffer);
}

Kernel::Point_3
Buffer3D::extend_point(const Kernel::Point_3 &point,
const Kernel::Vector_3 &direction, double distance) const
{
return point + direction * distance;
}

std::vector<Kernel::Point_3>
Buffer3D::create_circle_points(const Kernel::Point_3 &center,
const Kernel::Vector_3 &axis, double radius,
int segments) const
{
std::vector<Kernel::Point_3> points;
Kernel::Vector_3 perpendicular =
CGAL::cross_product(axis, Kernel::Vector_3(0, 0, 1));
if (perpendicular == CGAL::NULL_VECTOR) {
perpendicular = CGAL::cross_product(axis, Kernel::Vector_3(0, 1, 0));
}
perpendicular = normalizeVector(perpendicular);
Kernel::Vector_3 perpendicular2 =
normalizeVector(CGAL::cross_product(axis, perpendicular));

for (int i = 0; i < segments; ++i) {
double angle = 2.0 * M_PI * i / segments;
Kernel::Vector_3 offset = radius * (std::cos(angle) * perpendicular +
std::sin(angle) * perpendicular2);
points.push_back(center + offset);
}
return points;
}

Kernel::Plane_3
Buffer3D::compute_bisector_plane(const Kernel::Point_3 &p1,
const Kernel::Point_3 &p2,
const Kernel::Point_3 &p3) const
{
Kernel::Vector_3 v1 = normalizeVector(p2 - p1);
Kernel::Vector_3 v2 = normalizeVector(p3 - p2);
Kernel::Vector_3 bisector = v1 + v2;
return Kernel::Plane_3(p2, bisector);
}

Kernel::Point_3
Buffer3D::intersect_segment_plane(const Kernel::Point_3 &p1,
const Kernel::Point_3 &p2,
const Kernel::Plane_3 &plane) const
{
Kernel::Vector_3 v = p2 - p1;
Kernel::FT t =
-plane.a() * p1.x() - plane.b() * p1.y() - plane.c() * p1.z() - plane.d();
t /= plane.a() * v.x() + plane.b() * v.y() + plane.c() * v.z();
return p1 + t * v;
}

} // namespace algorithm
} // namespace SFCGAL
Loading

0 comments on commit 3a49384

Please sign in to comment.