File BoundaryVisitor.cpp
File List > algorithm > BoundaryVisitor.cpp
Go to the documentation of this file
// Copyright (c) 2012-2013, IGN France.
// Copyright (c) 2012-2022, Oslandia.
// SPDX-License-Identifier: LGPL-2.0-or-later
#include "SFCGAL/algorithm/BoundaryVisitor.h"
#include "SFCGAL/GeometryCollection.h"
#include "SFCGAL/LineString.h"
#include "SFCGAL/MultiLineString.h"
#include "SFCGAL/MultiPoint.h"
#include "SFCGAL/MultiPolygon.h"
#include "SFCGAL/MultiSolid.h"
#include "SFCGAL/Point.h"
#include "SFCGAL/Polygon.h"
#include "SFCGAL/PolyhedralSurface.h"
#include "SFCGAL/Solid.h"
#include "SFCGAL/Triangle.h"
#include "SFCGAL/TriangulatedSurface.h"
#include "SFCGAL/detail/ComplexComparator.h"
#include <complex>
#include <map>
#include <memory>
#include <utility>
namespace SFCGAL::algorithm {
void
BoundaryVisitor::visit(const Point & /*g*/)
{
_boundary.reset();
}
void
BoundaryVisitor::visit(const LineString &g)
{
if (g.isEmpty()) {
_boundary.reset();
return;
}
if (g.startPoint().coordinate() == g.endPoint().coordinate()) {
_boundary.reset();
} else {
std::unique_ptr<MultiPoint> boundary(new MultiPoint);
boundary->addGeometry(g.startPoint());
boundary->addGeometry(g.endPoint());
_boundary = std::move(boundary);
}
}
void
BoundaryVisitor::visit(const Polygon &g)
{
if (g.isEmpty()) {
_boundary.reset();
return;
}
if (!g.hasInteriorRings()) {
_boundary.reset(g.exteriorRing().clone());
} else {
std::unique_ptr<MultiLineString> boundary(new MultiLineString);
for (size_t i = 0; i < g.numRings(); i++) {
boundary->addGeometry(g.ringN(i));
}
_boundary = std::move(boundary);
}
}
void
BoundaryVisitor::visit(const Triangle &g)
{
if (g.isEmpty()) {
_boundary.reset();
return;
}
std::unique_ptr<LineString> boundary(new LineString);
for (size_t i = 0; i < 4; i++) {
boundary->addPoint(g.vertex(i));
}
_boundary = std::move(boundary);
}
void
BoundaryVisitor::visit(const Solid &g)
{
BOOST_THROW_EXCEPTION(
Exception((boost::format("unsupported type %1% in boundary operation") %
g.geometryType())
.str()));
}
void
BoundaryVisitor::visit(const MultiPoint & /*g*/)
{
_boundary.reset();
}
void
BoundaryVisitor::visit(const MultiLineString &g)
{
if (g.isEmpty()) {
_boundary.reset();
return;
}
/*
* create a GeometryGraph and rely on vertex degree (1 means boundary)
*/
graph::GeometryGraph graph;
graph::GeometryGraphBuilder graphBuilder(graph);
for (size_t i = 0; i < g.numGeometries(); i++) {
graphBuilder.addLineString(g.lineStringN(i));
}
getBoundaryFromLineStrings(graph);
}
void
BoundaryVisitor::visit(const MultiPolygon &g)
{
graph::GeometryGraph graph;
graph::GeometryGraphBuilder graphBuilder(graph);
for (size_t i = 0; i < g.numGeometries(); i++) {
graphBuilder.addPolygon(g.polygonN(i));
}
getBoundaryFromPolygons(graph);
}
void
BoundaryVisitor::visit(const MultiSolid &g)
{
BOOST_THROW_EXCEPTION(
Exception((boost::format("unsupported type %1% in boundary operation") %
g.geometryType())
.str()));
}
void
BoundaryVisitor::visit(const GeometryCollection &g)
{
BOOST_THROW_EXCEPTION(
Exception((boost::format("unsupported type %1% in boundary operation") %
g.geometryType())
.str()));
}
void
BoundaryVisitor::visit(const PolyhedralSurface &g)
{
graph::GeometryGraph graph;
graph::GeometryGraphBuilder graphBuilder(graph);
graphBuilder.addPolyhedralSurface(g);
getBoundaryFromPolygons(graph);
}
void
BoundaryVisitor::visit(const TriangulatedSurface &g)
{
graph::GeometryGraph graph;
graph::GeometryGraphBuilder graphBuilder(graph);
graphBuilder.addTriangulatedSurface(g);
getBoundaryFromPolygons(graph);
}
auto
BoundaryVisitor::releaseBoundary() -> Geometry *
{
if (_boundary != nullptr) {
return _boundary.release();
}
return new GeometryCollection();
}
void
BoundaryVisitor::getBoundaryFromLineStrings(const graph::GeometryGraph &graph)
{
using vertex_descriptor = graph::GeometryGraph::vertex_descriptor;
using vertex_iterator = graph::GeometryGraph::vertex_iterator;
std::vector<vertex_descriptor> vertices;
vertex_iterator it;
vertex_iterator end;
for (boost::tie(it, end) = graph.vertices(); it != end; ++it) {
vertex_descriptor vertex = *it;
if (graph.degree(vertex) == 1) {
vertices.push_back(vertex);
}
}
if (vertices.empty()) {
_boundary.reset();
} else if (vertices.size() == 1) {
_boundary = std::make_unique<Point>(graph[vertices[0]].coordinate);
} else {
std::unique_ptr<MultiPoint> boundary(new MultiPoint);
for (auto &vertice : vertices) {
boundary->addGeometry(new Point(graph[vertice].coordinate));
}
_boundary = std::move(boundary);
}
}
void
BoundaryVisitor::getBoundaryFromPolygons(const graph::GeometryGraph &g)
{
using vertex_descriptor = graph::GeometryGraph::vertex_descriptor;
// typedef graph::GeometryGraph::vertex_iterator vertex_iterator ;
using edge_descriptor = graph::GeometryGraph::edge_descriptor;
using edge_iterator = graph::GeometryGraph::edge_iterator;
std::vector<edge_descriptor> boundaryEdges;
edge_iterator it;
edge_iterator end;
for (boost::tie(it, end) = g.edges(); it != end; ++it) {
if (g.edges(g.source(*it), g.target(*it)).size() == 1U) {
boundaryEdges.push_back(*it);
}
}
if (boundaryEdges.empty()) {
_boundary.reset();
} else {
// TODO merge Line Segments into LineString
std::unique_ptr<MultiLineString> boundary(new MultiLineString);
for (auto &edge : boundaryEdges) {
vertex_descriptor source = g.source(edge);
vertex_descriptor target = g.target(edge);
boundary->addGeometry(new LineString(Point(g[source].coordinate),
Point(g[target].coordinate)));
}
_boundary = std::move(boundary);
}
}
} // namespace SFCGAL::algorithm