234 lines
6.8 KiB
C#
234 lines
6.8 KiB
C#
// -----------------------------------------------------------------------
|
|
// <copyright file="GenericMesher.cs">
|
|
// Triangle.NET code by Christian Woltering, http://triangle.codeplex.com/
|
|
// </copyright>
|
|
// -----------------------------------------------------------------------
|
|
|
|
namespace TriangleNet.Meshing
|
|
{
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using TriangleNet.Geometry;
|
|
using TriangleNet.IO;
|
|
using TriangleNet.Meshing.Algorithm;
|
|
|
|
/// <summary>
|
|
/// Create meshes of point sets or polygons.
|
|
/// </summary>
|
|
public class GenericMesher
|
|
{
|
|
Configuration config;
|
|
ITriangulator triangulator;
|
|
|
|
public GenericMesher()
|
|
: this(new Dwyer(), new Configuration())
|
|
{
|
|
}
|
|
|
|
public GenericMesher(ITriangulator triangulator)
|
|
: this(triangulator, new Configuration())
|
|
{
|
|
}
|
|
|
|
public GenericMesher(Configuration config)
|
|
: this(new Dwyer(), config)
|
|
{
|
|
}
|
|
|
|
public GenericMesher(ITriangulator triangulator, Configuration config)
|
|
{
|
|
this.config = config;
|
|
this.triangulator = triangulator;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public IMesh Triangulate(IList<Vertex> points)
|
|
{
|
|
return triangulator.Triangulate(points, config);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public IMesh Triangulate(IPolygon polygon)
|
|
{
|
|
return Triangulate(polygon, null, null);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public IMesh Triangulate(IPolygon polygon, ConstraintOptions options)
|
|
{
|
|
return Triangulate(polygon, options, null);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public IMesh Triangulate(IPolygon polygon, QualityOptions quality)
|
|
{
|
|
return Triangulate(polygon, null, quality);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public IMesh Triangulate(IPolygon polygon, ConstraintOptions options, QualityOptions quality)
|
|
{
|
|
var mesh = (Mesh)triangulator.Triangulate(polygon.Points, config);
|
|
|
|
var cmesher = new ConstraintMesher(mesh, config);
|
|
var qmesher = new QualityMesher(mesh, config);
|
|
|
|
mesh.SetQualityMesher(qmesher);
|
|
|
|
// Insert segments.
|
|
cmesher.Apply(polygon, options);
|
|
|
|
// Refine mesh.
|
|
qmesher.Apply(quality);
|
|
|
|
return mesh;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Generates a structured mesh with bounds [0, 0, width, height].
|
|
/// </summary>
|
|
/// <param name="width">Width of the mesh (must be > 0).</param>
|
|
/// <param name="height">Height of the mesh (must be > 0).</param>
|
|
/// <param name="nx">Number of segments in x direction.</param>
|
|
/// <param name="ny">Number of segments in y direction.</param>
|
|
/// <returns>Mesh</returns>
|
|
public static IMesh StructuredMesh(double width, double height, int nx, int ny)
|
|
{
|
|
if (width <= 0.0)
|
|
{
|
|
throw new ArgumentException("width");
|
|
}
|
|
|
|
if (height <= 0.0)
|
|
{
|
|
throw new ArgumentException("height");
|
|
}
|
|
|
|
return StructuredMesh(new Rectangle(0.0, 0.0, width, height), nx, ny);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Generates a structured mesh.
|
|
/// </summary>
|
|
/// <param name="bounds">Bounds of the mesh.</param>
|
|
/// <param name="nx">Number of segments in x direction.</param>
|
|
/// <param name="ny">Number of segments in y direction.</param>
|
|
/// <returns>Mesh</returns>
|
|
public static IMesh StructuredMesh(Rectangle bounds, int nx, int ny)
|
|
{
|
|
var polygon = new Polygon((nx + 1) * (ny + 1));
|
|
|
|
double x, y, dx, dy, left, bottom;
|
|
|
|
dx = bounds.Width / nx;
|
|
dy = bounds.Height / ny;
|
|
|
|
left = bounds.Left;
|
|
bottom = bounds.Bottom;
|
|
|
|
int i, j, k, l, n = 0;
|
|
|
|
// Add vertices.
|
|
var points = new Vertex[(nx + 1) * (ny + 1)];
|
|
|
|
for (i = 0; i <= nx; i++)
|
|
{
|
|
x = left + i * dx;
|
|
|
|
for (j = 0; j <= ny; j++)
|
|
{
|
|
y = bottom + j * dy;
|
|
|
|
points[n++] = new Vertex(x, y);
|
|
}
|
|
}
|
|
|
|
polygon.Points.AddRange(points);
|
|
|
|
n = 0;
|
|
|
|
// Set vertex hash and id.
|
|
foreach (var v in points)
|
|
{
|
|
v.hash = v.id = n++;
|
|
}
|
|
|
|
// Add boundary segments.
|
|
var segments = polygon.Segments;
|
|
|
|
segments.Capacity = 2 * (nx + ny);
|
|
|
|
Vertex a, b;
|
|
|
|
for (j = 0; j < ny; j++)
|
|
{
|
|
// Left
|
|
a = points[j];
|
|
b = points[j + 1];
|
|
|
|
segments.Add(new Segment(a, b, 1));
|
|
|
|
a.Label = b.Label = 1;
|
|
|
|
// Right
|
|
a = points[nx * (ny + 1) + j];
|
|
b = points[nx * (ny + 1) + (j + 1)];
|
|
|
|
segments.Add(new Segment(a, b, 1));
|
|
|
|
a.Label = b.Label = 1;
|
|
}
|
|
|
|
for (i = 0; i < nx; i++)
|
|
{
|
|
// Bottom
|
|
a = points[(ny + 1) * i];
|
|
b = points[(ny + 1) * (i + 1)];
|
|
|
|
segments.Add(new Segment(a, b, 1));
|
|
|
|
a.Label = b.Label = 1;
|
|
|
|
// Top
|
|
a = points[ny + (ny + 1) * i];
|
|
b = points[ny + (ny + 1) * (i + 1)];
|
|
|
|
segments.Add(new Segment(a, b, 1));
|
|
|
|
a.Label = b.Label = 1;
|
|
}
|
|
|
|
// Add triangles.
|
|
var triangles = new InputTriangle[2 * nx * ny];
|
|
|
|
n = 0;
|
|
|
|
for (i = 0; i < nx; i++)
|
|
{
|
|
for (j = 0; j < ny; j++)
|
|
{
|
|
k = j + (ny + 1) * i;
|
|
l = j + (ny + 1) * (i + 1);
|
|
|
|
// Create 2 triangles in rectangle [k, l, l + 1, k + 1].
|
|
|
|
if ((i + j) % 2 == 0)
|
|
{
|
|
// Diagonal from bottom left to top right.
|
|
triangles[n++] = new InputTriangle(k, l, l + 1);
|
|
triangles[n++] = new InputTriangle(k, l + 1, k + 1);
|
|
}
|
|
else
|
|
{
|
|
// Diagonal from top left to bottom right.
|
|
triangles[n++] = new InputTriangle(k, l, k + 1);
|
|
triangles[n++] = new InputTriangle(l, l + 1, k + 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
return Converter.ToMesh(polygon, triangles);
|
|
}
|
|
}
|
|
}
|