Files
sbox-public/engine/Sandbox.Engine/Game/Navigation/Generation/ContourBuilder.cs
s&box team 71f266059a Open source release
This commit imports the C# engine code and game files, excluding C++ source code.

[Source-Commit: ceb3d758046e50faa6258bc3b658a30c97743268]
2025-11-24 09:05:18 +00:00

946 lines
26 KiB
C#

using System.Buffers;
namespace Sandbox.Navigation.Generation;
[SkipHotload]
internal static class ContourBuilder
{
[SkipHotload]
private static class ContourBuildFlags
{
public const int RC_CONTOUR_TESS_WALL_EDGES = 0x01;
public const int RC_CONTOUR_TESS_AREA_EDGES = 0x02;
}
[SkipHotload]
private struct ContourHole
{
public int LeftMost;
public int MinX;
public int MinZ;
public Contour Contour;
}
[SkipHotload]
private struct ContourRegion
{
public Contour Outline;
public int HoleStartIndex;
public int HoleCount;
}
private class ContourHoleComparer : IComparer<ContourHole>
{
public static readonly ContourHoleComparer Shared = new();
private ContourHoleComparer()
{
}
public int Compare( ContourHole a, ContourHole b )
{
if ( a.MinX == b.MinX )
{
return a.MinZ.CompareTo( b.MinZ );
}
else
{
return a.MinX.CompareTo( b.MinX );
}
}
}
private class PotentialDiagonalComparer : IComparer<PotentialDiagonal>
{
public static readonly PotentialDiagonalComparer Shared = new();
private PotentialDiagonalComparer()
{
}
public int Compare( PotentialDiagonal va, PotentialDiagonal vb )
{
PotentialDiagonal a = va;
PotentialDiagonal b = vb;
return a.dist.CompareTo( b.dist );
}
}
private struct PotentialDiagonal
{
public int vert;
public int dist;
}
private static int GetCornerHeight( int x, int y, int i, int dir, CompactHeightfield chf, out bool isBorderVertex )
{
isBorderVertex = false;
CompactSpan s = chf.Spans[i];
int ch = s.StartY;
int dirp = (dir + 1) & 0x3;
Span<int> regs = stackalloc int[]
{
0, 0, 0, 0
};
// Combine region and area codes in order to prevent
// border vertices which are in between two areas to be removed.
regs[0] = chf.Spans[i].Region | (chf.Areas[i] << 16);
if ( Utils.GetCon( s, dir ) != Constants.NOT_CONNECTED )
{
int ax = x + Utils.GetDirOffsetX( dir );
int ay = y + Utils.GetDirOffsetZ( dir );
int ai = chf.Cells[ax + ay * chf.Width].Index + Utils.GetCon( s, dir );
CompactSpan @as = chf.Spans[ai];
ch = Math.Max( ch, @as.StartY );
regs[1] = chf.Spans[ai].Region | (chf.Areas[ai] << 16);
if ( Utils.GetCon( @as, dirp ) != Constants.NOT_CONNECTED )
{
int ax2 = ax + Utils.GetDirOffsetX( dirp );
int ay2 = ay + Utils.GetDirOffsetZ( dirp );
int ai2 = chf.Cells[ax2 + ay2 * chf.Width].Index + Utils.GetCon( @as, dirp );
CompactSpan as2 = chf.Spans[ai2];
ch = Math.Max( ch, as2.StartY );
regs[2] = chf.Spans[ai2].Region | (chf.Areas[ai2] << 16);
}
}
if ( Utils.GetCon( s, dirp ) != Constants.NOT_CONNECTED )
{
int ax = x + Utils.GetDirOffsetX( dirp );
int ay = y + Utils.GetDirOffsetZ( dirp );
int ai = chf.Cells[ax + ay * chf.Width].Index + Utils.GetCon( s, dirp );
CompactSpan @as = chf.Spans[ai];
ch = Math.Max( ch, @as.StartY );
regs[3] = chf.Spans[ai].Region | (chf.Areas[ai] << 16);
if ( Utils.GetCon( @as, dir ) != Constants.NOT_CONNECTED )
{
int ax2 = ax + Utils.GetDirOffsetX( dir );
int ay2 = ay + Utils.GetDirOffsetZ( dir );
int ai2 = chf.Cells[ax2 + ay2 * chf.Width].Index + Utils.GetCon( @as, dir );
CompactSpan as2 = chf.Spans[ai2];
ch = Math.Max( ch, as2.StartY );
regs[2] = chf.Spans[ai2].Region | (chf.Areas[ai2] << 16);
}
}
// Check if the vertex is special edge vertex, these vertices will be removed later.
for ( int j = 0; j < 4; ++j )
{
int a = j;
int b = (j + 1) & 0x3;
int c = (j + 2) & 0x3;
int d = (j + 3) & 0x3;
// The vertex is a border vertex there are two same exterior cells in a row,
// followed by two interior cells and none of the regions are out of bounds.
bool twoSameExts = (regs[a] & regs[b] & ContourRegionFlags.BORDER_REG) != 0 && regs[a] == regs[b];
bool twoInts = ((regs[c] | regs[d]) & ContourRegionFlags.BORDER_REG) == 0;
bool intsSameArea = (regs[c] >> 16) == (regs[d] >> 16);
bool noZeros = regs[a] != 0 && regs[b] != 0 && regs[c] != 0 && regs[d] != 0;
if ( twoSameExts && twoInts && intsSameArea && noZeros )
{
isBorderVertex = true;
break;
}
}
return ch;
}
private static void WalkContour( int x, int y, int i, CompactHeightfield chf, Span<int> flags, List<int> points )
{
// Choose the first non-connected edge
int dir = 0;
while ( (flags[i] & (1 << dir)) == 0 )
dir++;
int startDir = dir;
int starti = i;
int area = chf.Areas[i];
int iter = 0;
while ( ++iter < 40000 )
{
if ( (flags[i] & (1 << dir)) != 0 )
{
// Choose the edge corner
bool isBorderVertex = false;
bool isAreaBorder = false;
int px = x;
int py = GetCornerHeight( x, y, i, dir, chf, out isBorderVertex );
int pz = y;
switch ( dir )
{
case 0:
pz++;
break;
case 1:
px++;
pz++;
break;
case 2:
px++;
break;
}
int r = 0;
CompactSpan s = chf.Spans[i];
if ( Utils.GetCon( s, dir ) != Constants.NOT_CONNECTED )
{
int ax = x + Utils.GetDirOffsetX( dir );
int ay = y + Utils.GetDirOffsetZ( dir );
int ai = chf.Cells[ax + ay * chf.Width].Index + Utils.GetCon( s, dir );
r = chf.Spans[ai].Region;
if ( area != chf.Areas[ai] )
isAreaBorder = true;
}
if ( isBorderVertex )
r |= ContourRegionFlags.BORDER_VERTEX;
if ( isAreaBorder )
r |= ContourRegionFlags.AREA_BORDER;
points.Add( px );
points.Add( py );
points.Add( pz );
points.Add( r );
flags[i] &= ~(1 << dir); // Remove visited edges
dir = (dir + 1) & 0x3; // Rotate CW
}
else
{
int ni = -1;
int nx = x + Utils.GetDirOffsetX( dir );
int ny = y + Utils.GetDirOffsetZ( dir );
CompactSpan s = chf.Spans[i];
if ( Utils.GetCon( s, dir ) != Constants.NOT_CONNECTED )
{
CompactCell nc = chf.Cells[nx + ny * chf.Width];
ni = nc.Index + Utils.GetCon( s, dir );
}
if ( ni == -1 )
{
// Should not happen.
return;
}
x = nx;
y = ny;
i = ni;
dir = (dir + 3) & 0x3; // Rotate CCW
}
if ( starti == i && startDir == dir )
{
break;
}
}
}
private static float DistancePtSeg( int x, int z, int px, int pz, int qx, int qz )
{
float pqx = qx - px;
float pqz = qz - pz;
float dx = x - px;
float dz = z - pz;
float d = pqx * pqx + pqz * pqz;
float t = pqx * dx + pqz * dz;
if ( d > 0 )
t /= d;
if ( t < 0 )
t = 0;
else if ( t > 1 )
t = 1;
dx = px + t * pqx - x;
dz = pz + t * pqz - z;
return dx * dx + dz * dz;
}
private static void SimplifyContour( List<int> points, List<int> simplified, float maxError, int maxEdgeLen, int buildFlags )
{
// Add initial points.
bool hasConnections = false;
for ( int i = 0; i < points.Count; i += 4 )
{
if ( (points[i + 3] & ContourRegionFlags.CONTOUR_REG_MASK) != 0 )
{
hasConnections = true;
break;
}
}
if ( hasConnections )
{
// The contour has some portals to other regions.
// Add a new point to every location where the region changes.
for ( int i = 0, ni = points.Count / 4; i < ni; ++i )
{
int ii = (i + 1) % ni;
bool differentRegs = (points[i * 4 + 3] & ContourRegionFlags.CONTOUR_REG_MASK) != (points[ii * 4 + 3] & ContourRegionFlags.CONTOUR_REG_MASK);
bool areaBorders = (points[i * 4 + 3] & ContourRegionFlags.AREA_BORDER) != (points[ii * 4 + 3] & ContourRegionFlags.AREA_BORDER);
if ( differentRegs || areaBorders )
{
simplified.Add( points[i * 4 + 0] );
simplified.Add( points[i * 4 + 1] );
simplified.Add( points[i * 4 + 2] );
simplified.Add( i );
}
}
}
if ( simplified.Count == 0 )
{
// If there is no connections at all,
// create some initial points for the simplification process.
// Find lower-left and upper-right vertices of the contour.
int llx = points[0];
int lly = points[1];
int llz = points[2];
int lli = 0;
int urx = points[0];
int ury = points[1];
int urz = points[2];
int uri = 0;
for ( int i = 0; i < points.Count; i += 4 )
{
int x = points[i + 0];
int y = points[i + 1];
int z = points[i + 2];
if ( x < llx || (x == llx && z < llz) )
{
llx = x;
lly = y;
llz = z;
lli = i / 4;
}
if ( x > urx || (x == urx && z > urz) )
{
urx = x;
ury = y;
urz = z;
uri = i / 4;
}
}
simplified.Add( llx );
simplified.Add( lly );
simplified.Add( llz );
simplified.Add( lli );
simplified.Add( urx );
simplified.Add( ury );
simplified.Add( urz );
simplified.Add( uri );
}
// Add points until all raw points are within
// error tolerance to the simplified shape.
int pn = points.Count / 4;
for ( int i = 0; i < simplified.Count / 4; )
{
int ii = (i + 1) % (simplified.Count / 4);
int ax = simplified[i * 4 + 0];
int az = simplified[i * 4 + 2];
int ai = simplified[i * 4 + 3];
int bx = simplified[ii * 4 + 0];
int bz = simplified[ii * 4 + 2];
int bi = simplified[ii * 4 + 3];
// Find maximum deviation from the segment.
float maxd = 0;
int maxi = -1;
int ci, cinc, endi;
// Traverse the segment in lexilogical order so that the
// max deviation is calculated similarly when traversing
// opposite segments.
if ( bx > ax || (bx == ax && bz > az) )
{
cinc = 1;
ci = (ai + cinc) % pn;
endi = bi;
}
else
{
cinc = pn - 1;
ci = (bi + cinc) % pn;
endi = ai;
int temp = ax;
ax = bx;
bx = temp;
temp = az;
az = bz;
bz = temp;
}
// Tessellate only outer edges or edges between areas.
if ( (points[ci * 4 + 3] & ContourRegionFlags.CONTOUR_REG_MASK) == 0 || (points[ci * 4 + 3] & ContourRegionFlags.AREA_BORDER) != 0 )
{
while ( ci != endi )
{
float d = DistancePtSeg( points[ci * 4 + 0], points[ci * 4 + 2], ax, az, bx, bz );
if ( d > maxd )
{
maxd = d;
maxi = ci;
}
ci = (ci + cinc) % pn;
}
}
// If the max deviation is larger than accepted error,
// add new point, else continue to next segment.
if ( maxi != -1 && maxd > (maxError * maxError) )
{
// Add the point.
simplified.Insert( (i + 1) * 4 + 0, points[maxi * 4 + 0] );
simplified.Insert( (i + 1) * 4 + 1, points[maxi * 4 + 1] );
simplified.Insert( (i + 1) * 4 + 2, points[maxi * 4 + 2] );
simplified.Insert( (i + 1) * 4 + 3, maxi );
}
else
{
++i;
}
}
// Split too long edges.
if ( maxEdgeLen > 0 && (buildFlags & (ContourBuildFlags.RC_CONTOUR_TESS_WALL_EDGES | ContourBuildFlags.RC_CONTOUR_TESS_AREA_EDGES)) != 0 )
{
for ( int i = 0; i < simplified.Count / 4; )
{
int ii = (i + 1) % (simplified.Count / 4);
int ax = simplified[i * 4 + 0];
int az = simplified[i * 4 + 2];
int ai = simplified[i * 4 + 3];
int bx = simplified[ii * 4 + 0];
int bz = simplified[ii * 4 + 2];
int bi = simplified[ii * 4 + 3];
// Find maximum deviation from the segment.
int maxi = -1;
int ci = (ai + 1) % pn;
// Tessellate only outer edges or edges between areas.
bool tess = false;
// Wall edges.
if ( (buildFlags & ContourBuildFlags.RC_CONTOUR_TESS_WALL_EDGES) != 0 && (points[ci * 4 + 3] & ContourRegionFlags.CONTOUR_REG_MASK) == 0 )
{
tess = true;
}
// Edges between areas.
if ( (buildFlags & ContourBuildFlags.RC_CONTOUR_TESS_AREA_EDGES) != 0 && (points[ci * 4 + 3] & ContourRegionFlags.AREA_BORDER) != 0 )
{
tess = true;
}
if ( tess )
{
int dx = bx - ax;
int dz = bz - az;
if ( dx * dx + dz * dz > maxEdgeLen * maxEdgeLen )
{
// Round based on the segments in lexilogical order so that the
// max tesselation is consistent regardless in which direction
// segments are traversed.
int n = bi < ai ? (bi + pn - ai) : (bi - ai);
if ( n > 1 )
{
if ( bx > ax || (bx == ax && bz > az) )
maxi = (ai + n / 2) % pn;
else
maxi = (ai + (n + 1) / 2) % pn;
}
}
}
// If the max deviation is larger than accepted error,
// add new point, else continue to next segment.
if ( maxi != -1 )
{
// Add the point.
simplified.Insert( (i + 1) * 4 + 0, points[maxi * 4 + 0] );
simplified.Insert( (i + 1) * 4 + 1, points[maxi * 4 + 1] );
simplified.Insert( (i + 1) * 4 + 2, points[maxi * 4 + 2] );
simplified.Insert( (i + 1) * 4 + 3, maxi );
}
else
{
++i;
}
}
}
for ( int i = 0; i < simplified.Count / 4; ++i )
{
// The edge vertex flag is take from the current raw point,
// and the neighbour region is take from the next raw point.
int ai = (simplified[i * 4 + 3] + 1) % pn;
int bi = simplified[i * 4 + 3];
simplified[i * 4 + 3] = (points[ai * 4 + 3] & (ContourRegionFlags.CONTOUR_REG_MASK | ContourRegionFlags.AREA_BORDER))
| points[bi * 4 + 3] & ContourRegionFlags.BORDER_VERTEX;
}
}
private static int CalcAreaOfPolygon2D( Span<int> verts, int nverts )
{
int area = 0;
for ( int i = 0, j = nverts - 1; i < nverts; j = i++ )
{
int vi = i * 4;
int vj = j * 4;
area += verts[vi + 0] * verts[vj + 2] - verts[vj + 0] * verts[vi + 2];
}
return (area + 1) / 2;
}
private static bool IntersectSegContour( int d0, int d1, int i, int n, Span<int> verts, Span<int> d0verts, Span<int> d1verts )
{
// For each edge (k,k+1) of P
Span<int> pverts = stackalloc int[4 * 4];
for ( int g = 0; g < 4; g++ )
{
pverts[g] = d0verts[d0 + g];
pverts[4 + g] = d1verts[d1 + g];
}
d0 = 0;
d1 = 4;
for ( int k = 0; k < n; k++ )
{
int k1 = Utils.Next( k, n );
// Skip edges incident to i.
if ( i == k || i == k1 )
continue;
int p0 = k * 4;
int p1 = k1 * 4;
for ( int g = 0; g < 4; g++ )
{
pverts[8 + g] = verts[p0 + g];
pverts[12 + g] = verts[p1 + g];
}
p0 = 8;
p1 = 12;
if ( Utils.VEqual2D( pverts.Slice( d0, 3 ), pverts.Slice( p0, 3 ) ) || Utils.VEqual2D( pverts.Slice( d1, 3 ), pverts.Slice( p0, 3 ) ) ||
Utils.VEqual2D( pverts.Slice( d0, 3 ), pverts.Slice( p1, 3 ) ) || Utils.VEqual2D( pverts.Slice( d1, 3 ), pverts.Slice( p1, 3 ) ) )
continue;
if ( Utils.Intersect2D( pverts.Slice( d0, 3 ), pverts.Slice( d1, 3 ), pverts.Slice( p0, 3 ), pverts.Slice( p1, 3 ) ) )
return true;
}
return false;
}
private static bool InCone( int i, int n, Span<int> verts, int pj, Span<int> vertpj )
{
int pi = i * 4;
int pi1 = Utils.Next( i, n ) * 4;
int pin1 = Utils.Prev( i, n ) * 4;
Span<int> pverts = stackalloc int[4 * 4];
for ( int g = 0; g < 4; g++ )
{
pverts[g] = verts[pi + g];
pverts[4 + g] = verts[pi1 + g];
pverts[8 + g] = verts[pin1 + g];
pverts[12 + g] = vertpj[pj + g];
}
pi = 0;
pi1 = 4;
pin1 = 8;
pj = 12;
// If P[i] is a convex vertex [ i+1 left or on (i-1,i) ].
if ( Utils.LeftOn2D( pverts.Slice( pin1, 3 ), pverts.Slice( pi, 3 ), pverts.Slice( pi1, 3 ) ) )
return Utils.Left2D( pverts.Slice( pi, 3 ), pverts.Slice( pj, 3 ), pverts.Slice( pin1, 3 ) ) && Utils.Left2D( pverts.Slice( pj, 3 ), pverts.Slice( pi, 3 ), pverts.Slice( pi1, 3 ) );
// Assume (i-1,i,i+1) not collinear.
// else P[i] is reflex.
return !(Utils.LeftOn2D( pverts.Slice( pi, 3 ), pverts.Slice( pj, 3 ), pverts.Slice( pi1, 3 ) ) && Utils.LeftOn2D( pverts.Slice( pj, 3 ), pverts.Slice( pi, 3 ), pverts.Slice( pin1, 3 ) ));
}
private static void RemoveDegenerateSegments( List<int> simplified )
{
// Remove adjacent vertices which are equal on xz-plane,
// or else the triangulator will get confused.
int npts = simplified.Count / 4;
for ( int i = 0; i < npts; ++i )
{
int ni = Utils.Next( i, npts );
// if (Vequal(&simplified[i*4], &simplified[ni*4]))
if ( simplified[i * 4] == simplified[ni * 4]
&& simplified[i * 4 + 2] == simplified[ni * 4 + 2] )
{
// Degenerate segment, remove.
simplified.RemoveAt( i * 4 );
simplified.RemoveAt( i * 4 );
simplified.RemoveAt( i * 4 );
simplified.RemoveAt( i * 4 );
npts--;
}
}
}
// Finds the lowest leftmost vertex of a contour.
private static (int x, int z, int leftmost) FindLeftMostVertex( Contour contour )
{
int minx = contour.Vertices[0];
int minz = contour.Vertices[2];
int leftmost = 0;
for ( int i = 1; i < contour.VertexCount; i++ )
{
int x = contour.Vertices[i * 4 + 0];
int z = contour.Vertices[i * 4 + 2];
if ( x < minx || (x == minx && z < minz) )
{
minx = x;
minz = z;
leftmost = i;
}
}
return (minx, minz, leftmost);
}
private static void MergeRegionHoles( ContourRegion region, Span<ContourHole> regionHoles )
{
// Sort holes from left to right.
for ( int i = 0; i < region.HoleCount; i++ )
{
(int minx, int miny, int minleftmost) = FindLeftMostVertex( regionHoles[i].Contour );
regionHoles[i].MinX = minx;
regionHoles[i].MinZ = miny;
regionHoles[i].LeftMost = minleftmost;
}
regionHoles.Sort( ContourHoleComparer.Shared );
int maxVerts = region.Outline.VertexCount;
for ( int i = 0; i < region.HoleCount; i++ )
maxVerts += regionHoles[i].Contour.VertexCount;
using var pooledDiags = new PooledSpan<PotentialDiagonal>( maxVerts );
Span<PotentialDiagonal> diags = pooledDiags.Span;
Contour outline = region.Outline;
// Merge holes into the outline one by one.
for ( int i = 0; i < region.HoleCount; i++ )
{
Contour hole = regionHoles[i].Contour;
int index = -1;
int bestVertex = regionHoles[i].LeftMost;
for ( int iter = 0; iter < hole.VertexCount; iter++ )
{
// Find potential diagonals.
// The 'best' vertex must be in the cone described by 3 consecutive vertices of the outline.
// ..o j-1
// |
// | * best
// |
// j o-----o j+1
// :
int ndiags = 0;
int corner = bestVertex * 4;
for ( int j = 0; j < outline.VertexCount; j++ )
{
if ( InCone( j, outline.VertexCount, outline.Vertices, corner, hole.Vertices ) )
{
int dx = outline.Vertices[j * 4 + 0] - hole.Vertices[corner + 0];
int dz = outline.Vertices[j * 4 + 2] - hole.Vertices[corner + 2];
diags[ndiags].vert = j;
diags[ndiags].dist = dx * dx + dz * dz;
ndiags++;
}
}
// Sort potential diagonals by distance, we want to make the connection as short as possible.
diags.Slice( 0, ndiags ).Sort( PotentialDiagonalComparer.Shared );
// Find a diagonal that is not intersecting the outline not the remaining holes.
index = -1;
for ( int j = 0; j < ndiags; j++ )
{
int pt = diags[j].vert * 4;
bool intersect = IntersectSegContour( pt, corner, diags[j].vert, outline.VertexCount, outline.Vertices,
outline.Vertices, hole.Vertices );
for ( int k = i; k < region.HoleCount && !intersect; k++ )
intersect |= IntersectSegContour( pt, corner, -1, regionHoles[k].Contour.VertexCount,
regionHoles[k].Contour.Vertices, outline.Vertices, hole.Vertices );
if ( !intersect )
{
index = diags[j].vert;
break;
}
}
// If found non-intersecting diagonal, stop looking.
if ( index != -1 )
break;
// All the potential diagonals for the current vertex were intersecting, try next vertex.
bestVertex = (bestVertex + 1) % hole.VertexCount;
}
if ( index == -1 )
{
Log.Warning( "mergeHoles: Failed to find merge points for" );
continue;
}
Contour.MergeContours( region.Outline, hole, index, bestVertex );
}
}
/// @par
///
/// The raw contours will match the region outlines exactly. The @p maxError and @p maxEdgeLen
/// parameters control how closely the simplified contours will match the raw contours.
///
/// Simplified contours are generated such that the vertices for portals between areas match up.
/// (They are considered mandatory vertices.)
///
/// Setting @p maxEdgeLength to zero will disabled the edge length feature.
///
/// See the #rcConfig documentation for more information on the configuration parameters.
///
/// @see rcAllocContourSet, CompactHeightfield, ContourSet, rcConfig
public static ContourSet BuildContours( CompactHeightfield chf, float maxError, int maxEdgeLen, List<int> vertsCache, List<int> vertsSimplifiedCache, ContourSet cset, int buildFlags = ContourBuildFlags.RC_CONTOUR_TESS_WALL_EDGES )
{
int w = chf.Width;
int h = chf.Height;
int borderSize = chf.BorderSize;
cset.Contours.Clear();
cset.BMin = chf.BMin;
cset.BMax = chf.BMax;
if ( borderSize > 0 )
{
// If the heightfield was build with bordersize, remove the offset.
float pad = borderSize * chf.CellSize;
cset.BMin.x += pad;
cset.BMin.z += pad;
cset.BMax.x -= pad;
cset.BMax.z -= pad;
}
cset.CellSize = chf.CellSize;
cset.CellHeight = chf.CellHeight;
cset.Width = chf.Width - chf.BorderSize * 2;
cset.Height = chf.Height - chf.BorderSize * 2;
cset.BorderSize = chf.BorderSize;
cset.MaxError = maxError;
using var pooledFlags = new PooledSpan<int>( chf.SpanCount );
Span<int> flags = pooledFlags.Span;
// Mark boundaries.
for ( int y = 0; y < h; ++y )
{
for ( int x = 0; x < w; ++x )
{
CompactCell c = chf.Cells[x + y * w];
for ( int i = c.Index, ni = c.Index + c.Count; i < ni; ++i )
{
int res = 0;
CompactSpan s = chf.Spans[i];
if ( chf.Spans[i].Region == 0 || (chf.Spans[i].Region & ContourRegionFlags.BORDER_REG) != 0 )
{
flags[i] = 0;
continue;
}
for ( int dir = 0; dir < 4; ++dir )
{
int r = 0;
if ( Utils.GetCon( s, dir ) != Constants.NOT_CONNECTED )
{
int ax = x + Utils.GetDirOffsetX( dir );
int ay = y + Utils.GetDirOffsetZ( dir );
int ai = chf.Cells[ax + ay * w].Index + Utils.GetCon( s, dir );
r = chf.Spans[ai].Region;
}
if ( r == chf.Spans[i].Region )
res |= (1 << dir);
}
flags[i] = res ^ 0xf; // Inverse, mark non connected edges.
}
}
}
for ( int y = 0; y < h; ++y )
{
for ( int x = 0; x < w; ++x )
{
CompactCell c = chf.Cells[x + y * w];
for ( int i = c.Index, ni = c.Index + c.Count; i < ni; ++i )
{
if ( flags[i] == 0 || flags[i] == 0xf )
{
flags[i] = 0;
continue;
}
int reg = chf.Spans[i].Region;
if ( reg == 0 || (reg & ContourRegionFlags.BORDER_REG) != 0 )
continue;
int area = chf.Areas[i];
vertsCache.Clear();
vertsSimplifiedCache.Clear();
WalkContour( x, y, i, chf, flags, vertsCache );
SimplifyContour( vertsCache, vertsSimplifiedCache, maxError, maxEdgeLen, buildFlags );
RemoveDegenerateSegments( vertsSimplifiedCache );
// Store region->contour remap info.
// Create contour.
if ( vertsSimplifiedCache.Count / 4 >= 3 )
{
Contour cont = new Contour( vertsSimplifiedCache.Count / 4 );
cset.Contours.Add( cont );
for ( int l = 0; l < cont.Vertices.Length; l++ )
{
cont.Vertices[l] = vertsSimplifiedCache[l];
}
if ( borderSize > 0 )
{
// If the heightfield was build with bordersize, remove the offset.
for ( int j = 0; j < cont.VertexCount; ++j )
{
cont.Vertices[j * 4] -= borderSize;
cont.Vertices[j * 4 + 2] -= borderSize;
}
}
cont.Region = reg;
cont.Area = area;
}
}
}
}
// Merge holes if needed.
if ( cset.Contours.Count > 0 )
{
// Calculate winding of all polygons.
using var pooledWinding = new PooledSpan<int>( cset.Contours.Count );
Span<int> winding = pooledWinding.Span;
int nholes = 0;
for ( int i = 0; i < cset.Contours.Count; ++i )
{
Contour cont = cset.Contours[i];
// If the contour is wound backwards, it is a hole.
winding[i] = CalcAreaOfPolygon2D( cont.Vertices, cont.VertexCount ) < 0 ? -1 : 1;
if ( winding[i] < 0 )
nholes++;
}
if ( nholes > 0 )
{
// Collect outline contour and holes contours per region.
// We assume that there is one outline and multiple holes.
int nregions = chf.MaxRegions + 1;
using var pooledRegions = new PooledSpan<ContourRegion>( nregions );
Span<ContourRegion> regions = pooledRegions.Span;
regions.Clear();
using var pooledHoles = new PooledSpan<ContourHole>( nholes );
Span<ContourHole> holes = pooledHoles.Span;
for ( int i = 0; i < cset.Contours.Count; ++i )
{
Contour cont = cset.Contours[i];
// Positively wound contours are outlines, negative holes.
if ( winding[i] > 0 )
{
if ( regions[cont.Region].Outline != null )
{
throw new Exception(
"rcBuildContours: Multiple outlines for region " + cont.Region + "." );
}
regions[cont.Region].Outline = cont;
}
else
{
regions[cont.Region].HoleCount++;
}
}
var currentHoleIndex = 0;
for ( int i = 0; i < nregions; i++ )
{
if ( regions[i].HoleCount > 0 )
{
regions[i].HoleStartIndex = currentHoleIndex;
currentHoleIndex += regions[i].HoleCount;
// we increment this again in the next loop
regions[i].HoleCount = 0; // reuse as write cursor
}
}
Assert.Equals( currentHoleIndex, nholes );
for ( int i = 0; i < cset.Contours.Count; ++i )
{
Contour cont = cset.Contours[i];
if ( winding[i] < 0 )
{
ContourRegion reg = regions[cont.Region];
Assert.True( reg.HoleStartIndex + reg.HoleCount < nholes );
holes[reg.HoleStartIndex + reg.HoleCount].Contour = cont;
regions[cont.Region].HoleCount++;
}
}
// Finally merge each regions holes into the outline.
for ( int i = 0; i < nregions; i++ )
{
ContourRegion reg = regions[i];
if ( reg.HoleCount == 0 )
continue;
if ( reg.Outline != null )
{
MergeRegionHoles( reg, holes.Slice( reg.HoleStartIndex, reg.HoleCount ) );
}
else
{
// The region does not have an outline.
// This can happen if the contour becaomes selfoverlapping because of
// too aggressive simplification settings.
throw new Exception( "rcBuildContours: Bad outline for region " + i
+ ", contour simplification is likely too aggressive." );
}
}
}
}
return cset;
}
}