mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-11-04 03:26:39 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			617 lines
		
	
	
	
		
			16 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			617 lines
		
	
	
	
		
			16 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
//
 | 
						|
// Copyright (c) 2009-2010 Mikko Mononen memon@inside.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.
 | 
						|
//
 | 
						|
 | 
						|
#define _USE_MATH_DEFINES
 | 
						|
#include <math.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <ctype.h>
 | 
						|
#include <string.h>
 | 
						|
#include <algorithm>
 | 
						|
#include "Recast.h"
 | 
						|
#include "InputGeom.h"
 | 
						|
#include "ChunkyTriMesh.h"
 | 
						|
#include "MeshLoaderObj.h"
 | 
						|
#include "DebugDraw.h"
 | 
						|
#include "RecastDebugDraw.h"
 | 
						|
#include "DetourNavMesh.h"
 | 
						|
#include "Sample.h"
 | 
						|
 | 
						|
static bool intersectSegmentTriangle(const float* sp, const float* sq,
 | 
						|
									 const float* a, const float* b, const float* c,
 | 
						|
									 float &t)
 | 
						|
{
 | 
						|
	float v, w;
 | 
						|
	float ab[3], ac[3], qp[3], ap[3], norm[3], e[3];
 | 
						|
	rcVsub(ab, b, a);
 | 
						|
	rcVsub(ac, c, a);
 | 
						|
	rcVsub(qp, sp, sq);
 | 
						|
	
 | 
						|
	// Compute triangle normal. Can be precalculated or cached if
 | 
						|
	// intersecting multiple segments against the same triangle
 | 
						|
	rcVcross(norm, ab, ac);
 | 
						|
	
 | 
						|
	// Compute denominator d. If d <= 0, segment is parallel to or points
 | 
						|
	// away from triangle, so exit early
 | 
						|
	float d = rcVdot(qp, norm);
 | 
						|
	if (d <= 0.0f) return false;
 | 
						|
	
 | 
						|
	// Compute intersection t value of pq with plane of triangle. A ray
 | 
						|
	// intersects iff 0 <= t. Segment intersects iff 0 <= t <= 1. Delay
 | 
						|
	// dividing by d until intersection has been found to pierce triangle
 | 
						|
	rcVsub(ap, sp, a);
 | 
						|
	t = rcVdot(ap, norm);
 | 
						|
	if (t < 0.0f) return false;
 | 
						|
	if (t > d) return false; // For segment; exclude this code line for a ray test
 | 
						|
	
 | 
						|
	// Compute barycentric coordinate components and test if within bounds
 | 
						|
	rcVcross(e, qp, ap);
 | 
						|
	v = rcVdot(ac, e);
 | 
						|
	if (v < 0.0f || v > d) return false;
 | 
						|
	w = -rcVdot(ab, e);
 | 
						|
	if (w < 0.0f || v + w > d) return false;
 | 
						|
	
 | 
						|
	// Segment/ray intersects triangle. Perform delayed division
 | 
						|
	t /= d;
 | 
						|
	
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
static char* parseRow(char* buf, char* bufEnd, char* row, int len)
 | 
						|
{
 | 
						|
	bool start = true;
 | 
						|
	bool done = false;
 | 
						|
	int n = 0;
 | 
						|
	while (!done && buf < bufEnd)
 | 
						|
	{
 | 
						|
		char c = *buf;
 | 
						|
		buf++;
 | 
						|
		// multirow
 | 
						|
		switch (c)
 | 
						|
		{
 | 
						|
			case '\n':
 | 
						|
				if (start) break;
 | 
						|
				done = true;
 | 
						|
				break;
 | 
						|
			case '\r':
 | 
						|
				break;
 | 
						|
			case '\t':
 | 
						|
			case ' ':
 | 
						|
				if (start) break;
 | 
						|
				// else falls through
 | 
						|
			default:
 | 
						|
				start = false;
 | 
						|
				row[n++] = c;
 | 
						|
				if (n >= len-1)
 | 
						|
					done = true;
 | 
						|
				break;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	row[n] = '\0';
 | 
						|
	return buf;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
InputGeom::InputGeom() :
 | 
						|
	m_chunkyMesh(0),
 | 
						|
	m_mesh(0),
 | 
						|
	m_hasBuildSettings(false),
 | 
						|
	m_offMeshConCount(0),
 | 
						|
	m_volumeCount(0)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
InputGeom::~InputGeom()
 | 
						|
{
 | 
						|
	delete m_chunkyMesh;
 | 
						|
	delete m_mesh;
 | 
						|
}
 | 
						|
		
 | 
						|
bool InputGeom::loadMesh(rcContext* ctx, const std::string& filepath)
 | 
						|
{
 | 
						|
	if (m_mesh)
 | 
						|
	{
 | 
						|
		delete m_chunkyMesh;
 | 
						|
		m_chunkyMesh = 0;
 | 
						|
		delete m_mesh;
 | 
						|
		m_mesh = 0;
 | 
						|
	}
 | 
						|
	m_offMeshConCount = 0;
 | 
						|
	m_volumeCount = 0;
 | 
						|
	
 | 
						|
	m_mesh = new rcMeshLoaderObj;
 | 
						|
	if (!m_mesh)
 | 
						|
	{
 | 
						|
		ctx->log(RC_LOG_ERROR, "loadMesh: Out of memory 'm_mesh'.");
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	if (!m_mesh->load(filepath))
 | 
						|
	{
 | 
						|
		ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Could not load '%s'", filepath.c_str());
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	rcCalcBounds(m_mesh->getVerts(), m_mesh->getVertCount(), m_meshBMin, m_meshBMax);
 | 
						|
 | 
						|
	m_chunkyMesh = new rcChunkyTriMesh;
 | 
						|
	if (!m_chunkyMesh)
 | 
						|
	{
 | 
						|
		ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Out of memory 'm_chunkyMesh'.");
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	if (!rcCreateChunkyTriMesh(m_mesh->getVerts(), m_mesh->getTris(), m_mesh->getTriCount(), 256, m_chunkyMesh))
 | 
						|
	{
 | 
						|
		ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Failed to build chunky mesh.");
 | 
						|
		return false;
 | 
						|
	}		
 | 
						|
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
bool InputGeom::loadGeomSet(rcContext* ctx, const std::string& filepath)
 | 
						|
{
 | 
						|
	char* buf = 0;
 | 
						|
	FILE* fp = fopen(filepath.c_str(), "rb");
 | 
						|
	if (!fp)
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	if (fseek(fp, 0, SEEK_END) != 0)
 | 
						|
	{
 | 
						|
		fclose(fp);
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	long bufSize = ftell(fp);
 | 
						|
	if (bufSize < 0)
 | 
						|
	{
 | 
						|
		fclose(fp);
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	if (fseek(fp, 0, SEEK_SET) != 0)
 | 
						|
	{
 | 
						|
		fclose(fp);
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	buf = new char[bufSize];
 | 
						|
	if (!buf)
 | 
						|
	{
 | 
						|
		fclose(fp);
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	size_t readLen = fread(buf, bufSize, 1, fp);
 | 
						|
	fclose(fp);
 | 
						|
	if (readLen != 1)
 | 
						|
	{
 | 
						|
		delete[] buf;
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	
 | 
						|
	m_offMeshConCount = 0;
 | 
						|
	m_volumeCount = 0;
 | 
						|
	delete m_mesh;
 | 
						|
	m_mesh = 0;
 | 
						|
 | 
						|
	char* src = buf;
 | 
						|
	char* srcEnd = buf + bufSize;
 | 
						|
	char row[512];
 | 
						|
	while (src < srcEnd)
 | 
						|
	{
 | 
						|
		// Parse one row
 | 
						|
		row[0] = '\0';
 | 
						|
		src = parseRow(src, srcEnd, row, sizeof(row)/sizeof(char));
 | 
						|
		if (row[0] == 'f')
 | 
						|
		{
 | 
						|
			// File name.
 | 
						|
			const char* name = row+1;
 | 
						|
			// Skip white spaces
 | 
						|
			while (*name && isspace(*name))
 | 
						|
				name++;
 | 
						|
			if (*name)
 | 
						|
			{
 | 
						|
				if (!loadMesh(ctx, name))
 | 
						|
				{
 | 
						|
					delete [] buf;
 | 
						|
					return false;
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		else if (row[0] == 'c')
 | 
						|
		{
 | 
						|
			// Off-mesh connection
 | 
						|
			if (m_offMeshConCount < MAX_OFFMESH_CONNECTIONS)
 | 
						|
			{
 | 
						|
				float* v = &m_offMeshConVerts[m_offMeshConCount*3*2];
 | 
						|
				int bidir, area = 0, flags = 0;
 | 
						|
				float rad;
 | 
						|
				sscanf(row+1, "%f %f %f  %f %f %f %f %d %d %d",
 | 
						|
					   &v[0], &v[1], &v[2], &v[3], &v[4], &v[5], &rad, &bidir, &area, &flags);
 | 
						|
				m_offMeshConRads[m_offMeshConCount] = rad;
 | 
						|
				m_offMeshConDirs[m_offMeshConCount] = (unsigned char)bidir;
 | 
						|
				m_offMeshConAreas[m_offMeshConCount] = (unsigned char)area;
 | 
						|
				m_offMeshConFlags[m_offMeshConCount] = (unsigned short)flags;
 | 
						|
				m_offMeshConCount++;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		else if (row[0] == 'v')
 | 
						|
		{
 | 
						|
			// Convex volumes
 | 
						|
			if (m_volumeCount < MAX_VOLUMES)
 | 
						|
			{
 | 
						|
				ConvexVolume* vol = &m_volumes[m_volumeCount++];
 | 
						|
				sscanf(row+1, "%d %d %f %f", &vol->nverts, &vol->area, &vol->hmin, &vol->hmax);
 | 
						|
				for (int i = 0; i < vol->nverts; ++i)
 | 
						|
				{
 | 
						|
					row[0] = '\0';
 | 
						|
					src = parseRow(src, srcEnd, row, sizeof(row)/sizeof(char));
 | 
						|
					sscanf(row, "%f %f %f", &vol->verts[i*3+0], &vol->verts[i*3+1], &vol->verts[i*3+2]);
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		else if (row[0] == 's')
 | 
						|
		{
 | 
						|
			// Settings
 | 
						|
			m_hasBuildSettings = true;
 | 
						|
			sscanf(row + 1, "%f %f %f %f %f %f %f %f %f %f %f %f %f %d %f %f %f %f %f %f %f",
 | 
						|
							&m_buildSettings.cellSize,
 | 
						|
							&m_buildSettings.cellHeight,
 | 
						|
							&m_buildSettings.agentHeight,
 | 
						|
							&m_buildSettings.agentRadius,
 | 
						|
							&m_buildSettings.agentMaxClimb,
 | 
						|
							&m_buildSettings.agentMaxSlope,
 | 
						|
							&m_buildSettings.regionMinSize,
 | 
						|
							&m_buildSettings.regionMergeSize,
 | 
						|
							&m_buildSettings.edgeMaxLen,
 | 
						|
							&m_buildSettings.edgeMaxError,
 | 
						|
							&m_buildSettings.vertsPerPoly,
 | 
						|
							&m_buildSettings.detailSampleDist,
 | 
						|
							&m_buildSettings.detailSampleMaxError,
 | 
						|
							&m_buildSettings.partitionType,
 | 
						|
							&m_buildSettings.navMeshBMin[0],
 | 
						|
							&m_buildSettings.navMeshBMin[1],
 | 
						|
							&m_buildSettings.navMeshBMin[2],
 | 
						|
							&m_buildSettings.navMeshBMax[0],
 | 
						|
							&m_buildSettings.navMeshBMax[1],
 | 
						|
							&m_buildSettings.navMeshBMax[2],
 | 
						|
							&m_buildSettings.tileSize);
 | 
						|
		}
 | 
						|
	}
 | 
						|
	
 | 
						|
	delete [] buf;
 | 
						|
	
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
bool InputGeom::load(rcContext* ctx, const std::string& filepath)
 | 
						|
{
 | 
						|
	size_t extensionPos = filepath.find_last_of('.');
 | 
						|
	if (extensionPos == std::string::npos)
 | 
						|
		return false;
 | 
						|
 | 
						|
	std::string extension = filepath.substr(extensionPos);
 | 
						|
	std::transform(extension.begin(), extension.end(), extension.begin(), tolower);
 | 
						|
 | 
						|
	if (extension == ".gset")
 | 
						|
		return loadGeomSet(ctx, filepath);
 | 
						|
	if (extension == ".obj")
 | 
						|
		return loadMesh(ctx, filepath);
 | 
						|
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
bool InputGeom::saveGeomSet(const BuildSettings* settings)
 | 
						|
{
 | 
						|
	if (!m_mesh) return false;
 | 
						|
	
 | 
						|
	// Change extension
 | 
						|
	std::string filepath = m_mesh->getFileName();
 | 
						|
	size_t extPos = filepath.find_last_of('.');
 | 
						|
	if (extPos != std::string::npos)
 | 
						|
		filepath = filepath.substr(0, extPos);
 | 
						|
 | 
						|
	filepath += ".gset";
 | 
						|
 | 
						|
	FILE* fp = fopen(filepath.c_str(), "w");
 | 
						|
	if (!fp) return false;
 | 
						|
	
 | 
						|
	// Store mesh filename.
 | 
						|
	fprintf(fp, "f %s\n", m_mesh->getFileName().c_str());
 | 
						|
 | 
						|
	// Store settings if any
 | 
						|
	if (settings)
 | 
						|
	{
 | 
						|
		fprintf(fp,
 | 
						|
			"s %f %f %f %f %f %f %f %f %f %f %f %f %f %d %f %f %f %f %f %f %f\n",
 | 
						|
			settings->cellSize,
 | 
						|
			settings->cellHeight,
 | 
						|
			settings->agentHeight,
 | 
						|
			settings->agentRadius,
 | 
						|
			settings->agentMaxClimb,
 | 
						|
			settings->agentMaxSlope,
 | 
						|
			settings->regionMinSize,
 | 
						|
			settings->regionMergeSize,
 | 
						|
			settings->edgeMaxLen,
 | 
						|
			settings->edgeMaxError,
 | 
						|
			settings->vertsPerPoly,
 | 
						|
			settings->detailSampleDist,
 | 
						|
			settings->detailSampleMaxError,
 | 
						|
			settings->partitionType,
 | 
						|
			settings->navMeshBMin[0],
 | 
						|
			settings->navMeshBMin[1],
 | 
						|
			settings->navMeshBMin[2],
 | 
						|
			settings->navMeshBMax[0],
 | 
						|
			settings->navMeshBMax[1],
 | 
						|
			settings->navMeshBMax[2],
 | 
						|
			settings->tileSize);
 | 
						|
	}
 | 
						|
	
 | 
						|
	// Store off-mesh links.
 | 
						|
	for (int i = 0; i < m_offMeshConCount; ++i)
 | 
						|
	{
 | 
						|
		const float* v = &m_offMeshConVerts[i*3*2];
 | 
						|
		const float rad = m_offMeshConRads[i];
 | 
						|
		const int bidir = m_offMeshConDirs[i];
 | 
						|
		const int area = m_offMeshConAreas[i];
 | 
						|
		const int flags = m_offMeshConFlags[i];
 | 
						|
		fprintf(fp, "c %f %f %f  %f %f %f  %f %d %d %d\n",
 | 
						|
				v[0], v[1], v[2], v[3], v[4], v[5], rad, bidir, area, flags);
 | 
						|
	}
 | 
						|
 | 
						|
	// Convex volumes
 | 
						|
	for (int i = 0; i < m_volumeCount; ++i)
 | 
						|
	{
 | 
						|
		ConvexVolume* vol = &m_volumes[i];
 | 
						|
		fprintf(fp, "v %d %d %f %f\n", vol->nverts, vol->area, vol->hmin, vol->hmax);
 | 
						|
		for (int j = 0; j < vol->nverts; ++j)
 | 
						|
			fprintf(fp, "%f %f %f\n", vol->verts[j*3+0], vol->verts[j*3+1], vol->verts[j*3+2]);
 | 
						|
	}
 | 
						|
	
 | 
						|
	fclose(fp);
 | 
						|
	
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
static bool isectSegAABB(const float* sp, const float* sq,
 | 
						|
						 const float* amin, const float* amax,
 | 
						|
						 float& tmin, float& tmax)
 | 
						|
{
 | 
						|
	static const float EPS = 1e-6f;
 | 
						|
	
 | 
						|
	float d[3];
 | 
						|
	d[0] = sq[0] - sp[0];
 | 
						|
	d[1] = sq[1] - sp[1];
 | 
						|
	d[2] = sq[2] - sp[2];
 | 
						|
	tmin = 0.0;
 | 
						|
	tmax = 1.0f;
 | 
						|
	
 | 
						|
	for (int i = 0; i < 3; i++)
 | 
						|
	{
 | 
						|
		if (fabsf(d[i]) < EPS)
 | 
						|
		{
 | 
						|
			if (sp[i] < amin[i] || sp[i] > amax[i])
 | 
						|
				return false;
 | 
						|
		}
 | 
						|
		else
 | 
						|
		{
 | 
						|
			const float ood = 1.0f / d[i];
 | 
						|
			float t1 = (amin[i] - sp[i]) * ood;
 | 
						|
			float t2 = (amax[i] - sp[i]) * ood;
 | 
						|
			if (t1 > t2) { float tmp = t1; t1 = t2; t2 = tmp; }
 | 
						|
			if (t1 > tmin) tmin = t1;
 | 
						|
			if (t2 < tmax) tmax = t2;
 | 
						|
			if (tmin > tmax) return false;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
bool InputGeom::raycastMesh(float* src, float* dst, float& tmin)
 | 
						|
{
 | 
						|
	float dir[3];
 | 
						|
	rcVsub(dir, dst, src);
 | 
						|
 | 
						|
	// Prune hit ray.
 | 
						|
	float btmin, btmax;
 | 
						|
	if (!isectSegAABB(src, dst, m_meshBMin, m_meshBMax, btmin, btmax))
 | 
						|
		return false;
 | 
						|
	float p[2], q[2];
 | 
						|
	p[0] = src[0] + (dst[0]-src[0])*btmin;
 | 
						|
	p[1] = src[2] + (dst[2]-src[2])*btmin;
 | 
						|
	q[0] = src[0] + (dst[0]-src[0])*btmax;
 | 
						|
	q[1] = src[2] + (dst[2]-src[2])*btmax;
 | 
						|
	
 | 
						|
	int cid[512];
 | 
						|
	const int ncid = rcGetChunksOverlappingSegment(m_chunkyMesh, p, q, cid, 512);
 | 
						|
	if (!ncid)
 | 
						|
		return false;
 | 
						|
	
 | 
						|
	tmin = 1.0f;
 | 
						|
	bool hit = false;
 | 
						|
	const float* verts = m_mesh->getVerts();
 | 
						|
	
 | 
						|
	for (int i = 0; i < ncid; ++i)
 | 
						|
	{
 | 
						|
		const rcChunkyTriMeshNode& node = m_chunkyMesh->nodes[cid[i]];
 | 
						|
		const int* tris = &m_chunkyMesh->tris[node.i*3];
 | 
						|
		const int ntris = node.n;
 | 
						|
 | 
						|
		for (int j = 0; j < ntris*3; j += 3)
 | 
						|
		{
 | 
						|
			float t = 1;
 | 
						|
			if (intersectSegmentTriangle(src, dst,
 | 
						|
										 &verts[tris[j]*3],
 | 
						|
										 &verts[tris[j+1]*3],
 | 
						|
										 &verts[tris[j+2]*3], t))
 | 
						|
			{
 | 
						|
				if (t < tmin)
 | 
						|
					tmin = t;
 | 
						|
				hit = true;
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
	
 | 
						|
	return hit;
 | 
						|
}
 | 
						|
 | 
						|
void InputGeom::addOffMeshConnection(const float* spos, const float* epos, const float rad,
 | 
						|
									 unsigned char bidir, unsigned char area, unsigned short flags)
 | 
						|
{
 | 
						|
	if (m_offMeshConCount >= MAX_OFFMESH_CONNECTIONS) return;
 | 
						|
	float* v = &m_offMeshConVerts[m_offMeshConCount*3*2];
 | 
						|
	m_offMeshConRads[m_offMeshConCount] = rad;
 | 
						|
	m_offMeshConDirs[m_offMeshConCount] = bidir;
 | 
						|
	m_offMeshConAreas[m_offMeshConCount] = area;
 | 
						|
	m_offMeshConFlags[m_offMeshConCount] = flags;
 | 
						|
	m_offMeshConId[m_offMeshConCount] = 1000 + m_offMeshConCount;
 | 
						|
	rcVcopy(&v[0], spos);
 | 
						|
	rcVcopy(&v[3], epos);
 | 
						|
	m_offMeshConCount++;
 | 
						|
}
 | 
						|
 | 
						|
void InputGeom::deleteOffMeshConnection(int i)
 | 
						|
{
 | 
						|
	m_offMeshConCount--;
 | 
						|
	float* src = &m_offMeshConVerts[m_offMeshConCount*3*2];
 | 
						|
	float* dst = &m_offMeshConVerts[i*3*2];
 | 
						|
	rcVcopy(&dst[0], &src[0]);
 | 
						|
	rcVcopy(&dst[3], &src[3]);
 | 
						|
	m_offMeshConRads[i] = m_offMeshConRads[m_offMeshConCount];
 | 
						|
	m_offMeshConDirs[i] = m_offMeshConDirs[m_offMeshConCount];
 | 
						|
	m_offMeshConAreas[i] = m_offMeshConAreas[m_offMeshConCount];
 | 
						|
	m_offMeshConFlags[i] = m_offMeshConFlags[m_offMeshConCount];
 | 
						|
}
 | 
						|
 | 
						|
void InputGeom::drawOffMeshConnections(duDebugDraw* dd, bool hilight)
 | 
						|
{
 | 
						|
	unsigned int conColor = duRGBA(192,0,128,192);
 | 
						|
	unsigned int baseColor = duRGBA(0,0,0,64);
 | 
						|
	dd->depthMask(false);
 | 
						|
 | 
						|
	dd->begin(DU_DRAW_LINES, 2.0f);
 | 
						|
	for (int i = 0; i < m_offMeshConCount; ++i)
 | 
						|
	{
 | 
						|
		float* v = &m_offMeshConVerts[i*3*2];
 | 
						|
 | 
						|
		dd->vertex(v[0],v[1],v[2], baseColor);
 | 
						|
		dd->vertex(v[0],v[1]+0.2f,v[2], baseColor);
 | 
						|
		
 | 
						|
		dd->vertex(v[3],v[4],v[5], baseColor);
 | 
						|
		dd->vertex(v[3],v[4]+0.2f,v[5], baseColor);
 | 
						|
		
 | 
						|
		duAppendCircle(dd, v[0],v[1]+0.1f,v[2], m_offMeshConRads[i], baseColor);
 | 
						|
		duAppendCircle(dd, v[3],v[4]+0.1f,v[5], m_offMeshConRads[i], baseColor);
 | 
						|
 | 
						|
		if (hilight)
 | 
						|
		{
 | 
						|
			duAppendArc(dd, v[0],v[1],v[2], v[3],v[4],v[5], 0.25f,
 | 
						|
						(m_offMeshConDirs[i]&1) ? 0.6f : 0.0f, 0.6f, conColor);
 | 
						|
		}
 | 
						|
	}	
 | 
						|
	dd->end();
 | 
						|
 | 
						|
	dd->depthMask(true);
 | 
						|
}
 | 
						|
 | 
						|
void InputGeom::addConvexVolume(const float* verts, const int nverts,
 | 
						|
								const float minh, const float maxh, unsigned char area)
 | 
						|
{
 | 
						|
	if (m_volumeCount >= MAX_VOLUMES) return;
 | 
						|
	ConvexVolume* vol = &m_volumes[m_volumeCount++];
 | 
						|
	memset(vol, 0, sizeof(ConvexVolume));
 | 
						|
	memcpy(vol->verts, verts, sizeof(float)*3*nverts);
 | 
						|
	vol->hmin = minh;
 | 
						|
	vol->hmax = maxh;
 | 
						|
	vol->nverts = nverts;
 | 
						|
	vol->area = area;
 | 
						|
}
 | 
						|
 | 
						|
void InputGeom::deleteConvexVolume(int i)
 | 
						|
{
 | 
						|
	m_volumeCount--;
 | 
						|
	m_volumes[i] = m_volumes[m_volumeCount];
 | 
						|
}
 | 
						|
 | 
						|
void InputGeom::drawConvexVolumes(struct duDebugDraw* dd, bool /*hilight*/)
 | 
						|
{
 | 
						|
	dd->depthMask(false);
 | 
						|
 | 
						|
	dd->begin(DU_DRAW_TRIS);
 | 
						|
	
 | 
						|
	for (int i = 0; i < m_volumeCount; ++i)
 | 
						|
	{
 | 
						|
		const ConvexVolume* vol = &m_volumes[i];
 | 
						|
		unsigned int col = duTransCol(dd->areaToCol(vol->area), 32);
 | 
						|
		for (int j = 0, k = vol->nverts-1; j < vol->nverts; k = j++)
 | 
						|
		{
 | 
						|
			const float* va = &vol->verts[k*3];
 | 
						|
			const float* vb = &vol->verts[j*3];
 | 
						|
 | 
						|
			dd->vertex(vol->verts[0],vol->hmax,vol->verts[2], col);
 | 
						|
			dd->vertex(vb[0],vol->hmax,vb[2], col);
 | 
						|
			dd->vertex(va[0],vol->hmax,va[2], col);
 | 
						|
			
 | 
						|
			dd->vertex(va[0],vol->hmin,va[2], duDarkenCol(col));
 | 
						|
			dd->vertex(va[0],vol->hmax,va[2], col);
 | 
						|
			dd->vertex(vb[0],vol->hmax,vb[2], col);
 | 
						|
 | 
						|
			dd->vertex(va[0],vol->hmin,va[2], duDarkenCol(col));
 | 
						|
			dd->vertex(vb[0],vol->hmax,vb[2], col);
 | 
						|
			dd->vertex(vb[0],vol->hmin,vb[2], duDarkenCol(col));
 | 
						|
		}
 | 
						|
	}
 | 
						|
	
 | 
						|
	dd->end();
 | 
						|
 | 
						|
	dd->begin(DU_DRAW_LINES, 2.0f);
 | 
						|
	for (int i = 0; i < m_volumeCount; ++i)
 | 
						|
	{
 | 
						|
		const ConvexVolume* vol = &m_volumes[i];
 | 
						|
		unsigned int col = duTransCol(dd->areaToCol(vol->area), 220);
 | 
						|
		for (int j = 0, k = vol->nverts-1; j < vol->nverts; k = j++)
 | 
						|
		{
 | 
						|
			const float* va = &vol->verts[k*3];
 | 
						|
			const float* vb = &vol->verts[j*3];
 | 
						|
			dd->vertex(va[0],vol->hmin,va[2], duDarkenCol(col));
 | 
						|
			dd->vertex(vb[0],vol->hmin,vb[2], duDarkenCol(col));
 | 
						|
			dd->vertex(va[0],vol->hmax,va[2], col);
 | 
						|
			dd->vertex(vb[0],vol->hmax,vb[2], col);
 | 
						|
			dd->vertex(va[0],vol->hmin,va[2], duDarkenCol(col));
 | 
						|
			dd->vertex(va[0],vol->hmax,va[2], col);
 | 
						|
		}
 | 
						|
	}
 | 
						|
	dd->end();
 | 
						|
 | 
						|
	dd->begin(DU_DRAW_POINTS, 3.0f);
 | 
						|
	for (int i = 0; i < m_volumeCount; ++i)
 | 
						|
	{
 | 
						|
		const ConvexVolume* vol = &m_volumes[i];
 | 
						|
		unsigned int col = duDarkenCol(duTransCol(dd->areaToCol(vol->area), 220));
 | 
						|
		for (int j = 0; j < vol->nverts; ++j)
 | 
						|
		{
 | 
						|
			dd->vertex(vol->verts[j*3+0],vol->verts[j*3+1]+0.1f,vol->verts[j*3+2], col);
 | 
						|
			dd->vertex(vol->verts[j*3+0],vol->hmin,vol->verts[j*3+2], col);
 | 
						|
			dd->vertex(vol->verts[j*3+0],vol->hmax,vol->verts[j*3+2], col);
 | 
						|
		}
 | 
						|
	}
 | 
						|
	dd->end();
 | 
						|
	
 | 
						|
	
 | 
						|
	dd->depthMask(true);
 | 
						|
}
 |