forked from mirror/DotRecast
149 lines
7.1 KiB
C#
149 lines
7.1 KiB
C#
/*
|
|
recast4j copyright (c) 2021 Piotr Piastucki piotr@jtilia.org
|
|
|
|
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;
|
|
using DotRecast.Core;
|
|
using DotRecast.Recast;
|
|
|
|
namespace DotRecast.Detour.Dynamic.Io;
|
|
|
|
public class VoxelFile {
|
|
|
|
public static readonly ByteOrder PREFERRED_BYTE_ORDER = ByteOrder.BIG_ENDIAN;
|
|
public const int MAGIC = 'V' << 24 | 'O' << 16 | 'X' << 8 | 'L';
|
|
public const int VERSION_EXPORTER_MASK = 0xF000;
|
|
public const int VERSION_COMPRESSION_MASK = 0x0F00;
|
|
public const int VERSION_EXPORTER_RECAST4J = 0x1000;
|
|
public const int VERSION_COMPRESSION_LZ4 = 0x0100;
|
|
public int version;
|
|
public PartitionType partitionType = PartitionType.WATERSHED;
|
|
public bool filterLowHangingObstacles = true;
|
|
public bool filterLedgeSpans = true;
|
|
public bool filterWalkableLowHeightSpans = true;
|
|
public float walkableRadius;
|
|
public float walkableHeight;
|
|
public float walkableClimb;
|
|
public float walkableSlopeAngle;
|
|
public float cellSize;
|
|
public float maxSimplificationError;
|
|
public float maxEdgeLen;
|
|
public float minRegionArea;
|
|
public float regionMergeArea;
|
|
public int vertsPerPoly;
|
|
public bool buildMeshDetail;
|
|
public float detailSampleDistance;
|
|
public float detailSampleMaxError;
|
|
public bool useTiles;
|
|
public int tileSizeX;
|
|
public int tileSizeZ;
|
|
public float[] rotation = new float[3];
|
|
public float[] bounds = new float[6];
|
|
public readonly List<VoxelTile> tiles = new();
|
|
|
|
public void addTile(VoxelTile tile) {
|
|
tiles.Add(tile);
|
|
}
|
|
|
|
public RecastConfig getConfig(VoxelTile tile, PartitionType partitionType, int maxPolyVerts, int regionMergeSize,
|
|
bool filterLowHangingObstacles, bool filterLedgeSpans, bool filterWalkableLowHeightSpans,
|
|
AreaModification walkbableAreaMod, bool buildMeshDetail, float detailSampleDist, float detailSampleMaxError) {
|
|
return new RecastConfig(useTiles, tileSizeX, tileSizeZ, tile.borderSize, partitionType, cellSize, tile.cellHeight,
|
|
walkableSlopeAngle, filterLowHangingObstacles, filterLedgeSpans, filterWalkableLowHeightSpans, walkableHeight,
|
|
walkableRadius, walkableClimb, minRegionArea, regionMergeArea, maxEdgeLen, maxSimplificationError, maxPolyVerts,
|
|
buildMeshDetail, detailSampleDist, detailSampleMaxError, walkbableAreaMod);
|
|
}
|
|
|
|
public static VoxelFile from(RecastConfig config, List<RecastBuilderResult> results) {
|
|
VoxelFile f = new VoxelFile();
|
|
f.version = 1;
|
|
f.partitionType = config.partitionType;
|
|
f.filterLowHangingObstacles = config.filterLowHangingObstacles;
|
|
f.filterLedgeSpans = config.filterLedgeSpans;
|
|
f.filterWalkableLowHeightSpans = config.filterWalkableLowHeightSpans;
|
|
f.walkableRadius = config.walkableRadiusWorld;
|
|
f.walkableHeight = config.walkableHeightWorld;
|
|
f.walkableClimb = config.walkableClimbWorld;
|
|
f.walkableSlopeAngle = config.walkableSlopeAngle;
|
|
f.cellSize = config.cs;
|
|
f.maxSimplificationError = config.maxSimplificationError;
|
|
f.maxEdgeLen = config.maxEdgeLenWorld;
|
|
f.minRegionArea = config.minRegionAreaWorld;
|
|
f.regionMergeArea = config.mergeRegionAreaWorld;
|
|
f.vertsPerPoly = config.maxVertsPerPoly;
|
|
f.buildMeshDetail = config.buildMeshDetail;
|
|
f.detailSampleDistance = config.detailSampleDist;
|
|
f.detailSampleMaxError = config.detailSampleMaxError;
|
|
f.useTiles = config.useTiles;
|
|
f.tileSizeX = config.tileSizeX;
|
|
f.tileSizeZ = config.tileSizeZ;
|
|
f.bounds = new float[] { float.PositiveInfinity, float.PositiveInfinity, float.PositiveInfinity,
|
|
float.NegativeInfinity, float.NegativeInfinity, float.NegativeInfinity };
|
|
foreach (RecastBuilderResult r in results) {
|
|
f.tiles.Add(new VoxelTile(r.tileX, r.tileZ, r.getSolidHeightfield()));
|
|
f.bounds[0] = Math.Min(f.bounds[0], r.getSolidHeightfield().bmin[0]);
|
|
f.bounds[1] = Math.Min(f.bounds[1], r.getSolidHeightfield().bmin[1]);
|
|
f.bounds[2] = Math.Min(f.bounds[2], r.getSolidHeightfield().bmin[2]);
|
|
f.bounds[3] = Math.Max(f.bounds[3], r.getSolidHeightfield().bmax[0]);
|
|
f.bounds[4] = Math.Max(f.bounds[4], r.getSolidHeightfield().bmax[1]);
|
|
f.bounds[5] = Math.Max(f.bounds[5], r.getSolidHeightfield().bmax[2]);
|
|
}
|
|
return f;
|
|
}
|
|
|
|
public static VoxelFile from(DynamicNavMesh mesh) {
|
|
VoxelFile f = new VoxelFile();
|
|
f.version = 1;
|
|
DynamicNavMeshConfig config = mesh.config;
|
|
f.partitionType = config.partitionType;
|
|
f.filterLowHangingObstacles = config.filterLowHangingObstacles;
|
|
f.filterLedgeSpans = config.filterLedgeSpans;
|
|
f.filterWalkableLowHeightSpans = config.filterWalkableLowHeightSpans;
|
|
f.walkableRadius = config.walkableRadius;
|
|
f.walkableHeight = config.walkableHeight;
|
|
f.walkableClimb = config.walkableClimb;
|
|
f.walkableSlopeAngle = config.walkableSlopeAngle;
|
|
f.cellSize = config.cellSize;
|
|
f.maxSimplificationError = config.maxSimplificationError;
|
|
f.maxEdgeLen = config.maxEdgeLen;
|
|
f.minRegionArea = config.minRegionArea;
|
|
f.regionMergeArea = config.regionMergeArea;
|
|
f.vertsPerPoly = config.vertsPerPoly;
|
|
f.buildMeshDetail = config.buildDetailMesh;
|
|
f.detailSampleDistance = config.detailSampleDistance;
|
|
f.detailSampleMaxError = config.detailSampleMaxError;
|
|
f.useTiles = config.useTiles;
|
|
f.tileSizeX = config.tileSizeX;
|
|
f.tileSizeZ = config.tileSizeZ;
|
|
f.bounds = new float[] { float.PositiveInfinity, float.PositiveInfinity, float.PositiveInfinity,
|
|
float.NegativeInfinity, float.NegativeInfinity, float.NegativeInfinity };
|
|
foreach (VoxelTile vt in mesh.voxelTiles()) {
|
|
Heightfield heightfield = vt.heightfield();
|
|
f.tiles.Add(new VoxelTile(vt.tileX, vt.tileZ, heightfield));
|
|
f.bounds[0] = Math.Min(f.bounds[0], vt.boundsMin[0]);
|
|
f.bounds[1] = Math.Min(f.bounds[1], vt.boundsMin[1]);
|
|
f.bounds[2] = Math.Min(f.bounds[2], vt.boundsMin[2]);
|
|
f.bounds[3] = Math.Max(f.bounds[3], vt.boundsMax[0]);
|
|
f.bounds[4] = Math.Max(f.bounds[4], vt.boundsMax[1]);
|
|
f.bounds[5] = Math.Max(f.bounds[5], vt.boundsMax[2]);
|
|
}
|
|
return f;
|
|
}
|
|
|
|
}
|