Working on adding "Shewshunk triangle" support for triangulating polygons
and for reliably determining "a" point inside a polygon.
This commit is contained in:
parent
07e2840980
commit
3c558ab7e6
6 changed files with 274 additions and 29 deletions
|
@ -5,4 +5,7 @@ libBuild_a_SOURCES = \
|
|||
trinodes.cxx trinodes.hxx \
|
||||
trisegs.cxx trisegs.hxx
|
||||
|
||||
INCLUDES += -I$(top_builddir)/src -I$(top_builddir)/src/Lib
|
||||
INCLUDES += \
|
||||
-I$(top_builddir)/src \
|
||||
-I$(top_builddir)/src/Lib \
|
||||
-I$(top_builddir)/src/Construct
|
||||
|
|
|
@ -23,24 +23,20 @@
|
|||
|
||||
|
||||
#include <simgear/compiler.h>
|
||||
#include <simgear/math/fg_types.hxx>
|
||||
|
||||
#include <Polygon/polygon.hxx>
|
||||
|
||||
#include "trinodes.hxx"
|
||||
|
||||
|
||||
// calculate some "arbitrary" point inside the specified contour for
|
||||
// assigning attribute areas. This requires data structures outside
|
||||
// of "FGPolygon" which is why it is living over here in "Lib/Build"
|
||||
|
||||
void calc_point_inside( const int contour, const FGTriNodes& trinodes );
|
||||
|
||||
|
||||
#include <simgear/constants.h>
|
||||
#include <simgear/math/fg_types.hxx>
|
||||
#include <simgear/math/point3d.hxx>
|
||||
|
||||
#include <Build/trisegs.hxx>
|
||||
#include <Polygon/polygon.hxx>
|
||||
#include <Triangulate/trieles.hxx>
|
||||
|
||||
#define REAL double
|
||||
extern "C" {
|
||||
#include <Triangle/triangle.h>
|
||||
}
|
||||
|
||||
#include "trinodes.hxx"
|
||||
|
||||
|
||||
// Given a line segment specified by two endpoints p1 and p2, return
|
||||
|
@ -227,3 +223,238 @@ Point3D calc_point_inside( const FGPolygon& p, const int contour,
|
|||
|
||||
return inside_pt;
|
||||
}
|
||||
|
||||
|
||||
// basic triangulation of a polygon with out adding points or
|
||||
// splitting edges
|
||||
static triele_list triangulate_poly( const point_list contour,
|
||||
const point_list holes ) {
|
||||
// triangle list
|
||||
triele_list elelist;
|
||||
struct triangulateio in, out, vorout;
|
||||
int counter;
|
||||
|
||||
// point list
|
||||
double max_x = contour[0].x();
|
||||
in.numberofpoints = contour.size();
|
||||
in.pointlist = (REAL *) malloc(in.numberofpoints * 2 * sizeof(REAL));
|
||||
|
||||
for ( int i = 0; i < in.numberofpoints; ++i ) {
|
||||
in.pointlist[2*i] = contour[i].x();
|
||||
in.pointlist[2*i + 1] = contour[i].y();
|
||||
if ( contour[i].x() > max_x ) {
|
||||
max_x = contour[i].x();
|
||||
}
|
||||
}
|
||||
|
||||
in.numberofpointattributes = 1;
|
||||
in.pointattributelist = (REAL *) malloc(in.numberofpoints *
|
||||
in.numberofpointattributes *
|
||||
sizeof(REAL));
|
||||
for ( int i = 0; i < in.numberofpoints * in.numberofpointattributes; ++i) {
|
||||
in.pointattributelist[i] = contour[i].z();
|
||||
}
|
||||
|
||||
in.pointmarkerlist = (int *) malloc(in.numberofpoints * sizeof(int));
|
||||
for ( int i = 0; i < in.numberofpoints; ++i) {
|
||||
in.pointmarkerlist[i] = 0;
|
||||
}
|
||||
|
||||
// triangle list
|
||||
in.numberoftriangles = 0;
|
||||
|
||||
// segment list
|
||||
in.numberofsegments = contour.size();
|
||||
in.segmentlist = (int *) malloc(in.numberofsegments * 2 * sizeof(int));
|
||||
in.segmentmarkerlist = (int *) malloc(in.numberofsegments * sizeof(int));
|
||||
counter = 0;
|
||||
for ( int i = 0; i < in.numberofsegments - 1; ++i ) {
|
||||
in.segmentlist[counter++] = i;
|
||||
in.segmentlist[counter++] = i + 1;
|
||||
in.segmentmarkerlist[i] = 0;
|
||||
}
|
||||
in.segmentlist[counter++] = in.numberofsegments - 1;
|
||||
in.segmentlist[counter++] = 0;
|
||||
in.segmentmarkerlist[in.numberofsegments - 1] = 0;
|
||||
|
||||
// hole list
|
||||
in.numberofholes = holes.size() + 1;
|
||||
in.holelist = (REAL *) malloc(in.numberofholes * 2 * sizeof(REAL));
|
||||
counter = 0;
|
||||
for ( int i = 0; i < (int)holes.size(); ++i ) {
|
||||
in.holelist[counter++] = holes[i].x();
|
||||
in.holelist[counter++] = holes[i].y();
|
||||
}
|
||||
// outside of polygon
|
||||
in.holelist[counter++] = max_x + 1.0;
|
||||
in.holelist[counter++] = 0.0;
|
||||
|
||||
// region list
|
||||
in.numberofregions = 0;
|
||||
in.regionlist = (REAL *) NULL;
|
||||
|
||||
// prep the output structures
|
||||
out.pointlist = (REAL *) NULL; // Not needed if -N switch used.
|
||||
// Not needed if -N switch used or number of point attributes is zero:
|
||||
out.pointattributelist = (REAL *) NULL;
|
||||
out.pointmarkerlist = (int *) NULL; // Not needed if -N or -B switch used.
|
||||
out.trianglelist = (int *) NULL; // Not needed if -E switch used.
|
||||
// Not needed if -E switch used or number of triangle attributes is zero:
|
||||
out.triangleattributelist = (REAL *) NULL;
|
||||
out.neighborlist = (int *) NULL; // Needed only if -n switch used.
|
||||
// Needed only if segments are output (-p or -c) and -P not used:
|
||||
out.segmentlist = (int *) NULL;
|
||||
// Needed only if segments are output (-p or -c) and -P and -B not used:
|
||||
out.segmentmarkerlist = (int *) NULL;
|
||||
out.edgelist = (int *) NULL; // Needed only if -e switch used.
|
||||
out.edgemarkerlist = (int *) NULL; // Needed if -e used and -B not used.
|
||||
|
||||
vorout.pointlist = (REAL *) NULL; // Needed only if -v switch used.
|
||||
// Needed only if -v switch used and number of attributes is not zero:
|
||||
vorout.pointattributelist = (REAL *) NULL;
|
||||
vorout.edgelist = (int *) NULL; // Needed only if -v switch used.
|
||||
vorout.normlist = (REAL *) NULL; // Needed only if -v switch used.
|
||||
|
||||
// TEMPORARY
|
||||
// write_out_data(&in);
|
||||
|
||||
// Triangulate the points. Switches are chosen to read and write
|
||||
// a PSLG (p), number everything from zero (z), and produce an
|
||||
// edge list (e), and a triangle neighbor list (n).
|
||||
// no new points on boundary (Y), no internal segment
|
||||
// splitting (YY), no quality refinement (q)
|
||||
|
||||
string tri_options;
|
||||
tri_options = "pzYYen";
|
||||
cout << "Triangulation with options = " << tri_options << endl;
|
||||
|
||||
triangulate( (char *)tri_options.c_str(), &in, &out, &vorout );
|
||||
|
||||
// TEMPORARY
|
||||
// write_out_data(&out);
|
||||
|
||||
// now copy the results back into the corresponding FGTriangle
|
||||
// structures
|
||||
|
||||
// triangles
|
||||
elelist.clear();
|
||||
int n1, n2, n3;
|
||||
double attribute;
|
||||
for ( int i = 0; i < out.numberoftriangles; ++i ) {
|
||||
n1 = out.trianglelist[i * 3];
|
||||
n2 = out.trianglelist[i * 3 + 1];
|
||||
n3 = out.trianglelist[i * 3 + 2];
|
||||
if ( out.numberoftriangleattributes > 0 ) {
|
||||
attribute = out.triangleattributelist[i];
|
||||
} else {
|
||||
attribute = 0.0;
|
||||
}
|
||||
// cout << "triangle = " << n1 << " " << n2 << " " << n3 << endl;
|
||||
|
||||
elelist.push_back( FGTriEle( n1, n2, n3, attribute ) );
|
||||
}
|
||||
|
||||
// free mem allocated to the "Triangle" structures
|
||||
free(in.pointlist);
|
||||
free(in.pointattributelist);
|
||||
free(in.pointmarkerlist);
|
||||
free(in.regionlist);
|
||||
free(out.pointlist);
|
||||
free(out.pointattributelist);
|
||||
free(out.pointmarkerlist);
|
||||
free(out.trianglelist);
|
||||
free(out.triangleattributelist);
|
||||
// free(out.trianglearealist);
|
||||
free(out.neighborlist);
|
||||
free(out.segmentlist);
|
||||
free(out.segmentmarkerlist);
|
||||
free(out.edgelist);
|
||||
free(out.edgemarkerlist);
|
||||
free(vorout.pointlist);
|
||||
free(vorout.pointattributelist);
|
||||
free(vorout.edgelist);
|
||||
free(vorout.normlist);
|
||||
|
||||
return elelist;
|
||||
}
|
||||
|
||||
|
||||
// Find a point inside the polygon without regard for holes
|
||||
static Point3D point_inside_hole( point_list contour ) {
|
||||
point_list holes;
|
||||
holes.clear();
|
||||
|
||||
triele_list elelist = triangulate_poly( contour, holes );
|
||||
if ( elelist.size() <= 0 ) {
|
||||
cout << "Error polygon triangulated to zero triangles!" << endl;
|
||||
exit(-1);
|
||||
}
|
||||
|
||||
FGTriEle t = elelist[0];
|
||||
Point3D p1 = contour[ t.get_n1() ];
|
||||
Point3D p2 = contour[ t.get_n2() ];
|
||||
Point3D p3 = contour[ t.get_n3() ];
|
||||
|
||||
Point3D m1 = ( p1 + p2 ) / 2;
|
||||
Point3D m2 = ( p1 + p3 ) / 2;
|
||||
|
||||
Point3D center = ( m1 + m2 ) / 2;
|
||||
|
||||
return center;
|
||||
}
|
||||
|
||||
|
||||
// Find a point inside the polygon without regard for holes
|
||||
static Point3D point_inside_contour( const FGPolygon p, int contour ) {
|
||||
point_list holes;
|
||||
holes.clear();
|
||||
|
||||
// build list of holes
|
||||
for ( int i = 0; i < p.contours(); ++i ) {
|
||||
if ( p.get_hole_flag( i ) ) {
|
||||
holes.push_back( p.get_point_inside( i ) );
|
||||
}
|
||||
}
|
||||
|
||||
triele_list elelist = triangulate_poly( p.get_contour( contour ), holes );
|
||||
if ( elelist.size() <= 0 ) {
|
||||
cout << "Error polygon triangulated to zero triangles!" << endl;
|
||||
exit(-1);
|
||||
}
|
||||
|
||||
FGTriEle t = elelist[0];
|
||||
Point3D p1 = p.get_pt( contour, t.get_n1() );
|
||||
Point3D p2 = p.get_pt( contour, t.get_n2() );
|
||||
Point3D p3 = p.get_pt( contour, t.get_n3() );
|
||||
|
||||
Point3D m1 = ( p1 + p2 ) / 2;
|
||||
Point3D m2 = ( p1 + p3 ) / 2;
|
||||
|
||||
Point3D center = ( m1 + m2 ) / 2;
|
||||
|
||||
return center;
|
||||
|
||||
}
|
||||
|
||||
|
||||
// calculate some "arbitrary" point inside the specified contour for
|
||||
// assigning attribute areas
|
||||
void calc_points_inside( FGPolygon& p ) {
|
||||
// first calculate an inside point for all holes
|
||||
for ( int i = 0; i < p.contours(); ++i ) {
|
||||
if ( p.get_hole_flag( i ) ) {
|
||||
Point3D hole_pt = point_inside_hole( p.get_contour( i ) );
|
||||
p.set_point_inside( i, hole_pt );
|
||||
}
|
||||
}
|
||||
|
||||
// next calculate an inside point for all non-hole contours taking
|
||||
// into consideration the holes
|
||||
for ( int i = 0; i < p.contours(); ++i ) {
|
||||
if ( ! p.get_hole_flag( i ) ) {
|
||||
Point3D inside_pt = point_inside_contour( p, i );
|
||||
p.set_point_inside( i, inside_pt );
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
@ -46,6 +46,8 @@
|
|||
Point3D calc_point_inside( const FGPolygon& p, const int contour,
|
||||
const FGTriNodes& trinodes );
|
||||
|
||||
void calc_points_inside( const FGPolygon& p );
|
||||
|
||||
|
||||
#endif // _POLY_SUPPORT_HXX
|
||||
|
||||
|
|
|
@ -33,6 +33,10 @@ AreaType get_area_type( string area ) {
|
|||
return SomeSortOfArea;
|
||||
} else if ( area == "Hole" ) {
|
||||
return HoleArea;
|
||||
} else if ( area == "Island" ) {
|
||||
return IslandArea;
|
||||
} else if ( area == "Pond" ) {
|
||||
return PondArea;
|
||||
} else if ( (area == "Swamp or Marsh")
|
||||
|| (area == "Marsh") ) {
|
||||
return MarshArea;
|
||||
|
@ -85,6 +89,10 @@ string get_area_name( AreaType area ) {
|
|||
return "Hole";
|
||||
} else if ( area == MarshArea ) {
|
||||
return "Marsh";
|
||||
} else if ( area == PondArea ) {
|
||||
return "Pond";
|
||||
} else if ( area == IslandArea ) {
|
||||
return "Island";
|
||||
} else if ( area == LakeArea ) {
|
||||
return "Lake";
|
||||
} else if ( area == DryLakeArea ) {
|
||||
|
|
|
@ -38,18 +38,20 @@ FG_USING_STD(string);
|
|||
enum AreaType {
|
||||
SomeSortOfArea = 0,
|
||||
HoleArea = 1,
|
||||
LakeArea = 2,
|
||||
DryLakeArea = 3,
|
||||
IntLakeArea = 4,
|
||||
ReservoirArea = 5,
|
||||
IntReservoirArea = 6,
|
||||
StreamArea = 7,
|
||||
CanalArea = 8,
|
||||
GlacierArea = 9,
|
||||
OceanArea = 10,
|
||||
UrbanArea = 11,
|
||||
MarshArea = 12,
|
||||
DefaultArea = 13,
|
||||
PondArea = 2,
|
||||
IslandArea = 3,
|
||||
LakeArea = 4,
|
||||
DryLakeArea = 5,
|
||||
IntLakeArea = 6,
|
||||
ReservoirArea = 7,
|
||||
IntReservoirArea = 8,
|
||||
StreamArea = 9,
|
||||
CanalArea = 10,
|
||||
GlacierArea = 11,
|
||||
OceanArea = 12,
|
||||
UrbanArea = 13,
|
||||
MarshArea = 14,
|
||||
DefaultArea = 15,
|
||||
VoidArea = 9997,
|
||||
NullArea = 9998,
|
||||
UnknownArea = 9999
|
||||
|
@ -65,4 +67,3 @@ string get_area_name( AreaType area );
|
|||
|
||||
#endif // _NAMES_HXX
|
||||
|
||||
|
||||
|
|
|
@ -103,7 +103,7 @@ public:
|
|||
return poly[contour].size();
|
||||
}
|
||||
|
||||
// return the ith polygon point index from the specified contour
|
||||
// return the ith point from the specified contour
|
||||
inline Point3D get_pt( int contour, int i ) const {
|
||||
return poly[contour][i];
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue