172 lines
5.1 KiB
C#
172 lines
5.1 KiB
C#
|
// -----------------------------------------------------------------------
|
|||
|
// <copyright file="SimpleSmoother.cs" company="">
|
|||
|
// Triangle.NET code by Christian Woltering, http://triangle.codeplex.com/
|
|||
|
// </copyright>
|
|||
|
// -----------------------------------------------------------------------
|
|||
|
|
|||
|
namespace TriangleNet.Smoothing
|
|||
|
{
|
|||
|
using System.Linq;
|
|||
|
using TriangleNet.Geometry;
|
|||
|
using TriangleNet.Meshing;
|
|||
|
using TriangleNet.Topology.DCEL;
|
|||
|
using TriangleNet.Voronoi;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Simple mesh smoother implementation.
|
|||
|
/// </summary>
|
|||
|
/// <remarks>
|
|||
|
/// Vertices wich should not move (e.g. segment vertices) MUST have a
|
|||
|
/// boundary mark greater than 0.
|
|||
|
/// </remarks>
|
|||
|
public class SimpleSmoother : ISmoother
|
|||
|
{
|
|||
|
TrianglePool pool;
|
|||
|
Configuration config;
|
|||
|
|
|||
|
IVoronoiFactory factory;
|
|||
|
|
|||
|
ConstraintOptions options;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Initializes a new instance of the <see cref="SimpleSmoother" /> class.
|
|||
|
/// </summary>
|
|||
|
public SimpleSmoother()
|
|||
|
: this(new VoronoiFactory())
|
|||
|
{
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Initializes a new instance of the <see cref="SimpleSmoother" /> class.
|
|||
|
/// </summary>
|
|||
|
public SimpleSmoother(IVoronoiFactory factory)
|
|||
|
{
|
|||
|
this.factory = factory;
|
|||
|
this.pool = new TrianglePool();
|
|||
|
|
|||
|
this.config = new Configuration(
|
|||
|
() => RobustPredicates.Default,
|
|||
|
() => pool.Restart());
|
|||
|
|
|||
|
this.options = new ConstraintOptions() { ConformingDelaunay = true };
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Initializes a new instance of the <see cref="SimpleSmoother" /> class.
|
|||
|
/// </summary>
|
|||
|
/// <param name="factory">Voronoi object factory.</param>
|
|||
|
/// <param name="config">Configuration.</param>
|
|||
|
public SimpleSmoother(IVoronoiFactory factory, Configuration config)
|
|||
|
{
|
|||
|
this.factory = factory;
|
|||
|
this.config = config;
|
|||
|
|
|||
|
this.options = new ConstraintOptions() { ConformingDelaunay = true };
|
|||
|
}
|
|||
|
|
|||
|
public void Smooth(IMesh mesh)
|
|||
|
{
|
|||
|
Smooth(mesh, 10);
|
|||
|
}
|
|||
|
|
|||
|
public void Smooth(IMesh mesh, int limit)
|
|||
|
{
|
|||
|
var smoothedMesh = (Mesh)mesh;
|
|||
|
|
|||
|
var mesher = new GenericMesher(config);
|
|||
|
var predicates = config.Predicates();
|
|||
|
|
|||
|
// The smoother should respect the mesh segment splitting behavior.
|
|||
|
this.options.SegmentSplitting = smoothedMesh.behavior.NoBisect;
|
|||
|
|
|||
|
// Take a few smoothing rounds (Lloyd's algorithm).
|
|||
|
for (int i = 0; i < limit; i++)
|
|||
|
{
|
|||
|
Step(smoothedMesh, factory, predicates);
|
|||
|
|
|||
|
// Actually, we only want to rebuild, if the mesh is no longer
|
|||
|
// Delaunay. Flipping edges could be the right choice instead
|
|||
|
// of re-triangulating...
|
|||
|
smoothedMesh = (Mesh)mesher.Triangulate(Rebuild(smoothedMesh), options);
|
|||
|
|
|||
|
factory.Reset();
|
|||
|
}
|
|||
|
|
|||
|
smoothedMesh.CopyTo((Mesh)mesh);
|
|||
|
}
|
|||
|
|
|||
|
private void Step(Mesh mesh, IVoronoiFactory factory, IPredicates predicates)
|
|||
|
{
|
|||
|
var voronoi = new BoundedVoronoi(mesh, factory, predicates);
|
|||
|
|
|||
|
double x, y;
|
|||
|
|
|||
|
foreach (var face in voronoi.Faces)
|
|||
|
{
|
|||
|
if (face.generator.label == 0)
|
|||
|
{
|
|||
|
Centroid(face, out x, out y);
|
|||
|
|
|||
|
face.generator.x = x;
|
|||
|
face.generator.y = y;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Calculate the centroid of a polygon.
|
|||
|
/// </summary>
|
|||
|
private void Centroid(Face face, out double x, out double y)
|
|||
|
{
|
|||
|
double ai, atmp = 0, xtmp = 0, ytmp = 0;
|
|||
|
|
|||
|
var edge = face.Edge;
|
|||
|
var first = edge.Next.ID;
|
|||
|
|
|||
|
Point p, q;
|
|||
|
|
|||
|
do
|
|||
|
{
|
|||
|
p = edge.Origin;
|
|||
|
q = edge.Twin.Origin;
|
|||
|
|
|||
|
ai = p.x * q.y - q.x * p.y;
|
|||
|
atmp += ai;
|
|||
|
xtmp += (q.x + p.x) * ai;
|
|||
|
ytmp += (q.y + p.y) * ai;
|
|||
|
|
|||
|
edge = edge.Next;
|
|||
|
|
|||
|
} while (edge.Next.ID != first);
|
|||
|
|
|||
|
x = xtmp / (3 * atmp);
|
|||
|
y = ytmp / (3 * atmp);
|
|||
|
|
|||
|
//area = atmp / 2;
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Rebuild the input geometry.
|
|||
|
/// </summary>
|
|||
|
private Polygon Rebuild(Mesh mesh)
|
|||
|
{
|
|||
|
var data = new Polygon(mesh.vertices.Count);
|
|||
|
|
|||
|
foreach (var v in mesh.vertices.Values)
|
|||
|
{
|
|||
|
// Reset to input vertex.
|
|||
|
v.type = VertexType.InputVertex;
|
|||
|
|
|||
|
data.Points.Add(v);
|
|||
|
}
|
|||
|
|
|||
|
data.Segments.AddRange(mesh.subsegs.Values.Cast<ISegment>());
|
|||
|
|
|||
|
data.Holes.AddRange(mesh.holes);
|
|||
|
data.Regions.AddRange(mesh.regions);
|
|||
|
|
|||
|
return data;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|