DotRecastNetSim/src/DotRecast.Recast/RecastContour.cs

1023 lines
35 KiB
C#
Raw Normal View History

2023-03-14 08:02:43 +03:00
/*
Copyright (c) 2009-2010 Mikko Mononen memon@inside.org
2023-03-15 17:00:29 +03:00
recast4j copyright (c) 2015-2019 Piotr Piastucki piotr@jtilia.org
DotRecast Copyright (c) 2023 Choi Ikpil ikpil@naver.com
2023-03-14 08:02:43 +03:00
This software is provided 'as-is', without any express or implied
warranty. In no event will the authors be held liable for any damages
arising from the use of this software.
Permission is granted to anyone to use this software for any purpose,
including commercial applications, and to alter it and redistribute it
freely, subject to the following restrictions:
1. The origin of this software must not be misrepresented; you must not
claim that you wrote the original software. If you use this software
in a product, an acknowledgment in the product documentation would be
appreciated but is not required.
2. Altered source versions must be plainly marked as such, and must not be
misrepresented as being the original software.
3. This notice may not be removed or altered from any source distribution.
*/
using System;
using System.Collections.Generic;
2023-03-16 19:09:10 +03:00
namespace DotRecast.Recast
{
2023-03-14 08:02:43 +03:00
using static RecastConstants;
public class RecastContour
{
private class ContourRegion
{
public Contour outline;
public ContourHole[] holes;
public int nholes;
}
private class ContourHole
{
public int leftmost;
public int minx;
public int minz;
public Contour contour;
}
private class PotentialDiagonal
{
public int dist;
public int vert;
}
private class CornerHeight
{
public readonly int height;
public readonly bool borderVertex;
public CornerHeight(int height, bool borderVertex)
{
this.height = height;
this.borderVertex = borderVertex;
}
}
private static CornerHeight getCornerHeight(int x, int y, int i, int dir, CompactHeightfield chf)
{
bool isBorderVertex = false;
CompactSpan s = chf.spans[i];
int ch = s.y;
int dirp = (dir + 1) & 0x3;
int[] regs =
{
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].reg | (chf.areas[i] << 16);
if (RecastCommon.GetCon(s, dir) != RC_NOT_CONNECTED)
{
int ax = x + RecastCommon.GetDirOffsetX(dir);
int ay = y + RecastCommon.GetDirOffsetY(dir);
int ai = chf.cells[ax + ay * chf.width].index + RecastCommon.GetCon(s, dir);
CompactSpan @as = chf.spans[ai];
ch = Math.Max(ch, @as.y);
regs[1] = chf.spans[ai].reg | (chf.areas[ai] << 16);
if (RecastCommon.GetCon(@as, dirp) != RC_NOT_CONNECTED)
{
int ax2 = ax + RecastCommon.GetDirOffsetX(dirp);
int ay2 = ay + RecastCommon.GetDirOffsetY(dirp);
int ai2 = chf.cells[ax2 + ay2 * chf.width].index + RecastCommon.GetCon(@as, dirp);
CompactSpan as2 = chf.spans[ai2];
ch = Math.Max(ch, as2.y);
regs[2] = chf.spans[ai2].reg | (chf.areas[ai2] << 16);
}
}
if (RecastCommon.GetCon(s, dirp) != RC_NOT_CONNECTED)
{
int ax = x + RecastCommon.GetDirOffsetX(dirp);
int ay = y + RecastCommon.GetDirOffsetY(dirp);
int ai = chf.cells[ax + ay * chf.width].index + RecastCommon.GetCon(s, dirp);
CompactSpan @as = chf.spans[ai];
ch = Math.Max(ch, @as.y);
regs[3] = chf.spans[ai].reg | (chf.areas[ai] << 16);
if (RecastCommon.GetCon(@as, dir) != RC_NOT_CONNECTED)
{
int ax2 = ax + RecastCommon.GetDirOffsetX(dir);
int ay2 = ay + RecastCommon.GetDirOffsetY(dir);
int ai2 = chf.cells[ax2 + ay2 * chf.width].index + RecastCommon.GetCon(@as, dir);
CompactSpan as2 = chf.spans[ai2];
ch = Math.Max(ch, as2.y);
regs[2] = chf.spans[ai2].reg | (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] & RC_BORDER_REG) != 0 && regs[a] == regs[b];
bool twoInts = ((regs[c] | regs[d]) & RC_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 new CornerHeight(ch, isBorderVertex);
}
private static void walkContour(int x, int y, int i, CompactHeightfield chf, 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 isAreaBorder = false;
CornerHeight cornerHeight = getCornerHeight(x, y, i, dir, chf);
bool isBorderVertex = cornerHeight.borderVertex;
int px = x;
int py = cornerHeight.height;
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 (RecastCommon.GetCon(s, dir) != RC_NOT_CONNECTED)
{
int ax = x + RecastCommon.GetDirOffsetX(dir);
int ay = y + RecastCommon.GetDirOffsetY(dir);
int ai = chf.cells[ax + ay * chf.width].index + RecastCommon.GetCon(s, dir);
r = chf.spans[ai].reg;
if (area != chf.areas[ai])
isAreaBorder = true;
}
if (isBorderVertex)
r |= RC_BORDER_VERTEX;
if (isAreaBorder)
r |= RC_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 + RecastCommon.GetDirOffsetX(dir);
int ny = y + RecastCommon.GetDirOffsetY(dir);
CompactSpan s = chf.spans[i];
if (RecastCommon.GetCon(s, dir) != RC_NOT_CONNECTED)
{
CompactCell nc = chf.cells[nx + ny * chf.width];
ni = nc.index + RecastCommon.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] & RC_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] & RC_CONTOUR_REG_MASK) != (points[ii * 4 + 3] & RC_CONTOUR_REG_MASK);
bool areaBorders = (points[i * 4 + 3] & RC_AREA_BORDER) != (points[ii * 4 + 3] & RC_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] & RC_CONTOUR_REG_MASK) == 0 || (points[ci * 4 + 3] & RC_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 & (RC_CONTOUR_TESS_WALL_EDGES | 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 & RC_CONTOUR_TESS_WALL_EDGES) != 0 && (points[ci * 4 + 3] & RC_CONTOUR_REG_MASK) == 0)
tess = true;
// Edges between areas.
if ((buildFlags & RC_CONTOUR_TESS_AREA_EDGES) != 0 && (points[ci * 4 + 3] & RC_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 regardles 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] & (RC_CONTOUR_REG_MASK | RC_AREA_BORDER))
| points[bi * 4 + 3] & RC_BORDER_VERTEX;
}
}
private static int calcAreaOfPolygon2D(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 intersectSegCountour(int d0, int d1, int i, int n, int[] verts, int[] d0verts,
int[] d1verts)
{
// For each edge (k,k+1) of P
int[] pverts = new 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 = RecastMesh.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 (RecastMesh.vequal(pverts, d0, p0) || RecastMesh.vequal(pverts, d1, p0)
|| RecastMesh.vequal(pverts, d0, p1) || RecastMesh.vequal(pverts, d1, p1))
continue;
if (RecastMesh.intersect(pverts, d0, d1, p0, p1))
return true;
}
return false;
}
private static bool inCone(int i, int n, int[] verts, int pj, int[] vertpj)
{
int pi = i * 4;
int pi1 = RecastMesh.next(i, n) * 4;
int pin1 = RecastMesh.prev(i, n) * 4;
int[] pverts = new 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 (RecastMesh.leftOn(pverts, pin1, pi, pi1))
return RecastMesh.left(pverts, pi, pj, pin1) && RecastMesh.left(pverts, pj, pi, pi1);
// Assume (i-1,i,i+1) not collinear.
// else P[i] is reflex.
return !(RecastMesh.leftOn(pverts, pi, pj, pi1) && RecastMesh.leftOn(pverts, pj, pi, pin1));
}
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 = RecastMesh.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--;
}
}
}
private static void mergeContours(Contour ca, Contour cb, int ia, int ib)
{
int maxVerts = ca.nverts + cb.nverts + 2;
int[] verts = new int[maxVerts * 4];
int nv = 0;
// Copy contour A.
for (int i = 0; i <= ca.nverts; ++i)
{
int dst = nv * 4;
int src = ((ia + i) % ca.nverts) * 4;
verts[dst + 0] = ca.verts[src + 0];
verts[dst + 1] = ca.verts[src + 1];
verts[dst + 2] = ca.verts[src + 2];
verts[dst + 3] = ca.verts[src + 3];
nv++;
}
// Copy contour B
for (int i = 0; i <= cb.nverts; ++i)
{
int dst = nv * 4;
int src = ((ib + i) % cb.nverts) * 4;
verts[dst + 0] = cb.verts[src + 0];
verts[dst + 1] = cb.verts[src + 1];
verts[dst + 2] = cb.verts[src + 2];
verts[dst + 3] = cb.verts[src + 3];
nv++;
}
ca.verts = verts;
ca.nverts = nv;
cb.verts = null;
cb.nverts = 0;
}
// Finds the lowest leftmost vertex of a contour.
private static int[] findLeftMostVertex(Contour contour)
{
int minx = contour.verts[0];
int minz = contour.verts[2];
int leftmost = 0;
for (int i = 1; i < contour.nverts; i++)
{
int x = contour.verts[i * 4 + 0];
int z = contour.verts[i * 4 + 2];
if (x < minx || (x == minx && z < minz))
{
minx = x;
minz = z;
leftmost = i;
}
}
return new int[] { minx, minz, leftmost };
}
private class CompareHoles : IComparer<ContourHole>
{
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 CompareDiagDist : IComparer<PotentialDiagonal>
{
public int Compare(PotentialDiagonal va, PotentialDiagonal vb)
{
PotentialDiagonal a = va;
PotentialDiagonal b = vb;
return a.dist.CompareTo(b.dist);
}
}
private static void mergeRegionHoles(Telemetry ctx, ContourRegion region)
{
// Sort holes from left to right.
for (int i = 0; i < region.nholes; i++)
{
int[] minleft = findLeftMostVertex(region.holes[i].contour);
region.holes[i].minx = minleft[0];
region.holes[i].minz = minleft[1];
region.holes[i].leftmost = minleft[2];
}
Array.Sort(region.holes, new CompareHoles());
int maxVerts = region.outline.nverts;
for (int i = 0; i < region.nholes; i++)
maxVerts += region.holes[i].contour.nverts;
PotentialDiagonal[] diags = new PotentialDiagonal[maxVerts];
for (int pd = 0; pd < maxVerts; pd++)
{
diags[pd] = new PotentialDiagonal();
}
Contour outline = region.outline;
// Merge holes into the outline one by one.
for (int i = 0; i < region.nholes; i++)
{
Contour hole = region.holes[i].contour;
int index = -1;
int bestVertex = region.holes[i].leftmost;
for (int iter = 0; iter < hole.nverts; iter++)
{
// Find potential diagonals.
// The 'best' vertex must be in the cone described by 3 cosequtive 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.nverts; j++)
{
if (inCone(j, outline.nverts, outline.verts, corner, hole.verts))
{
int dx = outline.verts[j * 4 + 0] - hole.verts[corner + 0];
int dz = outline.verts[j * 4 + 2] - hole.verts[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.
Array.Sort(diags, 0, ndiags, new CompareDiagDist());
// 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 = intersectSegCountour(pt, corner, diags[j].vert, outline.nverts, outline.verts,
outline.verts, hole.verts);
for (int k = i; k < region.nholes && !intersect; k++)
intersect |= intersectSegCountour(pt, corner, -1, region.holes[k].contour.nverts,
region.holes[k].contour.verts, outline.verts, hole.verts);
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.nverts;
}
if (index == -1)
{
ctx.warn("mergeHoles: Failed to find merge points for");
continue;
}
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, rcCompactHeightfield, rcContourSet, rcConfig
public static ContourSet buildContours(Telemetry ctx, CompactHeightfield chf, float maxError, int maxEdgeLen,
int buildFlags)
{
int w = chf.width;
int h = chf.height;
int borderSize = chf.borderSize;
ContourSet cset = new ContourSet();
ctx.startTimer("CONTOURS");
RecastVectors.copy(cset.bmin, chf.bmin, 0);
RecastVectors.copy(cset.bmax, chf.bmax, 0);
if (borderSize > 0)
{
// If the heightfield was build with bordersize, remove the offset.
float pad = borderSize * chf.cs;
cset.bmin[0] += pad;
cset.bmin[2] += pad;
cset.bmax[0] -= pad;
cset.bmax[2] -= pad;
}
cset.cs = chf.cs;
cset.ch = chf.ch;
cset.width = chf.width - chf.borderSize * 2;
cset.height = chf.height - chf.borderSize * 2;
cset.borderSize = chf.borderSize;
cset.maxError = maxError;
int[] flags = new int[chf.spanCount];
ctx.startTimer("CONTOURS_TRACE");
// 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].reg == 0 || (chf.spans[i].reg & RC_BORDER_REG) != 0)
{
flags[i] = 0;
continue;
}
for (int dir = 0; dir < 4; ++dir)
{
int r = 0;
if (RecastCommon.GetCon(s, dir) != RC_NOT_CONNECTED)
{
int ax = x + RecastCommon.GetDirOffsetX(dir);
int ay = y + RecastCommon.GetDirOffsetY(dir);
int ai = chf.cells[ax + ay * w].index + RecastCommon.GetCon(s, dir);
r = chf.spans[ai].reg;
}
if (r == chf.spans[i].reg)
res |= (1 << dir);
}
flags[i] = res ^ 0xf; // Inverse, mark non connected edges.
}
}
}
ctx.stopTimer("CONTOURS_TRACE");
2023-03-16 19:09:10 +03:00
List<int> verts = new List<int>(256);
List<int> simplified = new List<int>(64);
2023-03-14 08:02:43 +03:00
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].reg;
if (reg == 0 || (reg & RC_BORDER_REG) != 0)
continue;
int area = chf.areas[i];
verts.Clear();
simplified.Clear();
ctx.startTimer("CONTOURS_WALK");
walkContour(x, y, i, chf, flags, verts);
ctx.stopTimer("CONTOURS_WALK");
ctx.startTimer("CONTOURS_SIMPLIFY");
simplifyContour(verts, simplified, maxError, maxEdgeLen, buildFlags);
removeDegenerateSegments(simplified);
ctx.stopTimer("CONTOURS_SIMPLIFY");
// Store region->contour remap info.
// Create contour.
if (simplified.Count / 4 >= 3)
{
Contour cont = new Contour();
cset.conts.Add(cont);
cont.nverts = simplified.Count / 4;
cont.verts = new int[simplified.Count];
for (int l = 0; l < cont.verts.Length; l++)
{
cont.verts[l] = simplified[l];
}
if (borderSize > 0)
{
// If the heightfield was build with bordersize, remove the offset.
for (int j = 0; j < cont.nverts; ++j)
{
cont.verts[j * 4] -= borderSize;
cont.verts[j * 4 + 2] -= borderSize;
}
}
cont.nrverts = verts.Count / 4;
cont.rverts = new int[verts.Count];
for (int l = 0; l < cont.rverts.Length; l++)
{
cont.rverts[l] = verts[l];
}
if (borderSize > 0)
{
// If the heightfield was build with bordersize, remove the offset.
for (int j = 0; j < cont.nrverts; ++j)
{
cont.rverts[j * 4] -= borderSize;
cont.rverts[j * 4 + 2] -= borderSize;
}
}
cont.reg = reg;
cont.area = area;
}
}
}
}
// Merge holes if needed.
if (cset.conts.Count > 0)
{
// Calculate winding of all polygons.
int[] winding = new int[cset.conts.Count];
int nholes = 0;
for (int i = 0; i < cset.conts.Count; ++i)
{
Contour cont = cset.conts[i];
// If the contour is wound backwards, it is a hole.
winding[i] = calcAreaOfPolygon2D(cont.verts, cont.nverts) < 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;
ContourRegion[] regions = new ContourRegion[nregions];
for (int i = 0; i < nregions; i++)
{
regions[i] = new ContourRegion();
}
for (int i = 0; i < cset.conts.Count; ++i)
{
Contour cont = cset.conts[i];
// Positively would contours are outlines, negative holes.
if (winding[i] > 0)
{
if (regions[cont.reg].outline != null)
{
throw new Exception(
"rcBuildContours: Multiple outlines for region " + cont.reg + ".");
}
regions[cont.reg].outline = cont;
}
else
{
regions[cont.reg].nholes++;
}
}
for (int i = 0; i < nregions; i++)
{
if (regions[i].nholes > 0)
{
regions[i].holes = new ContourHole[regions[i].nholes];
for (int nh = 0; nh < regions[i].nholes; nh++)
{
regions[i].holes[nh] = new ContourHole();
}
regions[i].nholes = 0;
}
}
for (int i = 0; i < cset.conts.Count; ++i)
{
Contour cont = cset.conts[i];
ContourRegion reg = regions[cont.reg];
if (winding[i] < 0)
reg.holes[reg.nholes++].contour = cont;
}
// Finally merge each regions holes into the outline.
for (int i = 0; i < nregions; i++)
{
ContourRegion reg = regions[i];
if (reg.nholes == 0)
continue;
if (reg.outline != null)
{
mergeRegionHoles(ctx, reg);
}
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.");
}
}
}
}
ctx.stopTimer("CONTOURS");
return cset;
}
2023-03-16 19:09:10 +03:00
}
2023-03-14 08:02:43 +03:00
}