Skip to content

Latest commit

 

History

History
315 lines (261 loc) · 13.8 KB

DOCS.md

File metadata and controls

315 lines (261 loc) · 13.8 KB

Basic usage

To triangulate a polygon, you'll need to create a detria::Triangulation instance, and provide the list of points and the list of polylines:

// Create a square, and triangulate it

// List of points (positions)
std::vector<detria::PointD> points =
{
    { 0.0, 0.0 },
    { 1.0, 0.0 },
    { 1.0, 1.0 },
    { 0.0, 1.0 }
};

// List of point indices
std::vector<size_t> outline = { 0, 1, 2, 3 };

bool delaunay = true;

detria::Triangulation tri;
tri.setPoints(points);
tri.addOutline(outline);

bool success = tri.triangulate(delaunay);

if (success)
{
    // Iterate over the result triangles

    // Should the result triangles be in CW order?
    bool cwTriangles = true;

    tri.forEachTriangle([&](detria::Triangle<size_t> triangle)
    {
        // `triangle` contains the point indices

        detria::PointD firstPointOfTriangle = points[triangle.x];
        detria::PointD secondPointOfTriangle = points[triangle.y];
        detria::PointD thirdPointOfTriangle = points[triangle.z];
    }, cwTriangles);
}

Terminology

  • Polyline: List of edges that form a simple polygon. Outlines and holes are polylines.
  • Outline: A polyline that contains a "solid" part of the triangulation (will contain interior triangles).
  • Hole: A polyline that will be subtracted from the triangulation (will contain hole triangles).
  • Convex hull triangles: When creating the triangulation, the convex hull of the points is automatically calculated. This can cause triangles to be created that are outside every outline, but these triangles are usually not used.

See images below for a visual representation:

Outlines, holes, and Steiner points

You can add multiple outlines, holes, and Steiner points to the triangulation.
The orientation of the outlines and holes can be either clockwise or counter-clockwise, it doesn't matter.
Outlines and holes can be nested, but:

  • Every hole must be directly inside an outline
  • Every outline must either be inside nothing (top-level), or directly inside a hole

Outlines that are directly inside other outlines, or holes that are directly insode other holes are invalid.
To add a hole to the triangulation, you can use the addHole function.
If you don't know if a polyline is an outline or a hole, you can use the addPolylineAutoDetectType function, which will automatically decide it.
You can also add a constrained edge between any two vertices using the setConstrainedEdge. This ensures that the final triangulation will have an edge between those two vertices.
Points that are not part of any outlines or holes will be used as Steiner points.

Memory safety

The points and polylines are not copied when set, only a pointer is stored. You must make sure that the pointers are still valid during the triangulate call. For example:

// THIS IS BAD, DON'T DO THIS

detria::Triangulation tri;
if (something)
{
    std::vector<detria::PointD> points = getPoints(...);
    tri.setPoints(points);
}
else
{
    std::vector<detria::PointD> points = getSomeOtherPoints(...);
    tri.setPoints(points);
}

// Will crash, the pointer stored in the triangulation is invalid
tri.triangulate(true);
// THIS IS ALSO BAD, DON'T DO THIS EITHER

detria::Triangulation tri;
std::vector<detria::PointD> points = getPoints(...);
tri.setPoints(points);

points.push_back(getSomeOtherPoint()); // The vector might reallocate here

// Undefined behavior, the pointer inside `points` might have been reallocated, and might point to freed memory
tri.triangulate(true);

Using the result of the triangulation

Use the function forEachTriangle to iterate over every interior triangle in the result triangulation.
Use forEachHoleTriangle to iterate over hole triangles.
You can also use forEachTriangleOfEveryLocation to iterate over every single triangle, even the convex hull triangles.
These functions have the following parameters:

  • The first parameter is a callback, which will be called for each triangle that has the "correct" location - interior triangles for forEachTriangle, and hole triangles for forEachHoleTriangle
    The callback must have a single detria::Triangle<Idx> parameter (can also have a const reference).
  • The second parameter is a boolean value, which determines if the result triangles should be in clockwise orientation (true: clockwise, false: counter-clockwise).
    The default value for this parameter is true (clockwise triangles).

See the "Basic usage" part for a code example.

The function forEachTriangleOfLocation is similar to the previous ones, but this one also takes a detria::TriangleLocation value, which decides which triangles the callback should be called on.
The callback passed to this function also must have a detria::TriangleLocation parameter as the second parameter.
Code example:

tri.forEachTriangleOfLocation(
    [](const detria::Triangle<Idx>& triangle, detria::TriangleLocation location)
    {
        // Do something with `triangle` and `location`
    },

    // You can specify multiple locations
    detria::TriangleLocation::Interior | detria::TriangleLocation::Hole,

    // Should the triangles be in clockwise order?
    true
);

Delaunay triangulation of a point set

It's possible to create the delaunay triangulation without any polylines/constrained edges:

detria::Triangulation tri;
// Only set points, don't add any polylines
tri.setPoints(...);
if (tri.triangulate(true))
{
    // Iterate over the triangles
    tri.forEachTriangleOfEveryLocation([](const detria::Triangle<size_t>& triangle)
    {
        // ...
    });
}

Other results of the triangulation

Convex hull

After triangulation, you can get the convex hull of the entire triangulation, using the forEachConvexHullVertex function:

tri.forEachConvexHullVertex([](Idx vertexIndex)
{
    std::cout << vertexIndex << std::endl;
});

The vertices are in clockwise order.

Polyline hierarchy

You can query the parent of each polyline (that is, which polyline contains the current polyline directly).
addOutline, addHole, and addPolylineAutoDetectType return the polyline's index, which can be used to get its parent, using getParentPolylineIndex:

detria::Triangulation tri;
tri.setPoints(...);
Idx outlineIndex = tri.addOutline(...);
if (tri.triangulate(true))
{
    // If the polyline is top-level (has no parent), then nullopt is returned
    // Otherwise, the parent index of the polyline is returned
    std::optional<Idx> parentIndex = tri.getParentPolylineIndex(outlineIndex);
    // `parentIndex` should be nullopt here, since it's the only polyline 
}

In the example image above (in the "Terminology" section, the letter R), let's say that the outline (green) is index 0, and the hole (red) is index 1.
In this case, tri.getParentPolylineIndex(0) returns nullopt, tri.getParentPolylineIndex(1) returns 0.

Template parameters and configuration

The detria::Triangulation class has multiple template parameters:

template <
    typename Point = PointD,
    typename Idx = uint32_t,
    typename Config = DefaultTriangulationConfig<Point>
>
class Triangulation
{
    // ...
};

template <typename Point, typename Idx>
struct DefaultTriangulationConfig
{
    using PointGetter = DefaultPointGetter<Point, Idx>;

    using Allocator = memory::DefaultAllocator;

    template <typename T, typename Allocator>
    using Collection = std::vector<T, Allocator>;

    // Etc...
};

Template parameters:

  • Point - The point type used in the triangulation, more details about this in the next section.
  • Idx - The index type, which will be used when referring to vertex indices. You can use types like size_t instead of uint32_t if you have the indices in that format.
    Note that because a half-edge data structure is used (which uses two indices per edge), the number of edges in the result triangulation must be less than half of the max value of type (so e.g. 32767 for 16-bit numbers).

Configuration:
The configuration allows you to customize even more parameters of the triangulation.

  • PointGetter - Used to retrieve points using custom logic, or to convert between point types. More details about this in the next section.

  • Collection - You can use a custom collection instead of std::vector, if you need to.
    The collection must have the basic functions of a collection, e.g. size(), resize(), indexing operator, etc.

  • Allocator - You can use a custom allocator instead of the default allocator (which uses the global new operator). More details about this in the "Custom allocators" section.

  • Sorter - You can use a custom sorting algorithm (which will be used to sort the points in the initial triangulation step). The provided sorter class must have a function with the following signature:

    template <typename Iter, typename Cmp>
    static void sort(Iter begin, Iter end, Cmp comparer);

    (This is the same signature as std::sort with a custom comparer)

  • UseRobustOrientationTests - You can disable robust orientation tests if you don't need them, but this can cause triangulations to produce incorrect results (because of floating point inaccuracy).

  • UseRobustIncircleTests - Same as above, but for incircle tests (these are only used in delaunay triangulations).

    Important: When using the robust tests, compiler flags for floating-point operation optimizations must be disabled (e.g. -ffast-math or /fp:fast).

    Robust tests are not required when using integer points, so in that case these options are ignored.

  • IndexChecks - If enabled, then all user-provided indices are checked, and if anything is invalid (out-of-bounds or negative indices, or two consecutive duplicate indices), then an error is generated.
    If disabled, then no checks are done, so if the input has any invalid indices, then it's undefined behavior (might crash).

  • NaNChecks - If enabled, then all the input points are checked, and if any of the points have a NaN or infinity value, then an error is generated.
    If there are no such values in the list of input points, then this can be disabled.

The Point and Idx types are not part of the configuration, because these types are the most likely to be changed, and it would require a configuration class to be defined for each combination.

To create a custom configuration, inherit from DefaultTriangulationConfig, and override the values/types you want to change. For example:

struct MyTriangulationConfig : public DefaultTriangulationConfig<detria::PointD>
{
    using Allocator = MyCustomAllocatorType;

    // The rest are used from the default configuration
}

void doStuff()
{
    detria::Triangulation<detria::PointD, uint32_t, MyTriangulationConfig> tri(getMyCustomAllocator());
    ...
}

This allows you to only override what you want, and keep the rest of the configuration the same.
Alternatively, you can just create a new class, but then you'd need to set every parameter manually (instead of only overriding the ones you want to change).

Custom point types

You can use any point type:

  • If your point type has x and y fields, then it will automatically work.
  • Otherwise, you must declare a "point getter" class (or a lambda function), which retrieves a point from a point index. It must be able to convert your type to a type that has x and y fields.
    Let's say that we store points as std::array<float, 2>. Then we can define the following point getter class:
// Simple example
struct PointGetterFromStdArray
{
    PointGetterFromStdArray(const std::vector<std::array<float, 2>>& points) :
        _points(&points)
    {
    }
   
    detria::PointF operator()(Idx idx) const
    {
        const std::array<float, 2>& point = (*_points)[size_t(idx)];
        return detria::PointF{ .x = point[0], .y = point[1] };
    }

private:
    const std::vector<std::array<float, 2>>* _points;
};

Point getter classes must have a Point operator()(Idx) const function, that has a single parameter (the point index), and returns anything that has an x and an y field. You can also use a lambda function.
The input parameter and the returned value can both be const references.

Integer coordinates

You can use integer coordinates for your points. For calculations, the values are converted to 64-bit signed integers to avoid overflows. But overflows can still happen if the difference of the coordinates are too large, which can cause errors in the triangulation.
Integer overflows are only checked in debug mode.

Custom allocators

You can use custom allocators for memory management.
The allocator class must have the following function:

template <typename T>
StlAllocator<T> createStlAllocator()
{
    // Create C++ allocator for type `T`
    return StlAllocator<T>();
}

StlAllocator must be a C++ allocator type (so must have value_type, size_type, etc.)
The allocator object can be passed to the constructor of detria::Triangulation. To create allocators for the collections, the allocator.createStlAllocator<T>() function will be used.
See detria::DefaultAllocator for a code example.

Error handling

Not all inputs can be triangulated. The triangulate function will return true if the triangulation was successful, and false if it failed.
If it failed, then you can use the getError function to get the error type, and getErrorMessage to get a human-readable error message.

Thread safety

There is no global state, so each detria::Triangulation object can be used independently from each other, even on separate threads.
Using a single triangulation object from multiple threads is possible, but:

  • If a mutable (non-const) function is running, then no other functions must be running on any other threads

So, at any point in time, only ONE of the following must be true:

  • A single mutable function is running (e.g. triangulate)
  • Any number of const functions are running, on any number of threads (e.g. forEachTriangle)
  • No functions running