forked from mirror/DotRecast
191 lines
7.1 KiB
C#
191 lines
7.1 KiB
C#
|
/*
|
||
|
Copyright (c) 2009-2010 Mikko Mononen memon@inside.org
|
||
|
recast4j copyright (c) 2015-2019 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;
|
||
|
|
||
|
namespace DotRecast.Detour.Crowd;
|
||
|
|
||
|
using static DetourCommon;
|
||
|
|
||
|
/// Represents an agent managed by a #dtCrowd object.
|
||
|
/// @ingroup crowd
|
||
|
public class CrowdAgent {
|
||
|
|
||
|
/// The type of navigation mesh polygon the agent is currently traversing.
|
||
|
/// @ingroup crowd
|
||
|
public enum CrowdAgentState {
|
||
|
DT_CROWDAGENT_STATE_INVALID, /// < The agent is not in a valid state.
|
||
|
DT_CROWDAGENT_STATE_WALKING, /// < The agent is traversing a normal navigation mesh polygon.
|
||
|
DT_CROWDAGENT_STATE_OFFMESH, /// < The agent is traversing an off-mesh connection.
|
||
|
};
|
||
|
|
||
|
public enum MoveRequestState {
|
||
|
DT_CROWDAGENT_TARGET_NONE,
|
||
|
DT_CROWDAGENT_TARGET_FAILED,
|
||
|
DT_CROWDAGENT_TARGET_VALID,
|
||
|
DT_CROWDAGENT_TARGET_REQUESTING,
|
||
|
DT_CROWDAGENT_TARGET_WAITING_FOR_QUEUE,
|
||
|
DT_CROWDAGENT_TARGET_WAITING_FOR_PATH,
|
||
|
DT_CROWDAGENT_TARGET_VELOCITY,
|
||
|
};
|
||
|
|
||
|
public readonly long idx;
|
||
|
/// The type of mesh polygon the agent is traversing. (See: #CrowdAgentState)
|
||
|
public CrowdAgentState state;
|
||
|
/// True if the agent has valid path (targetState == DT_CROWDAGENT_TARGET_VALID) and the path does not lead to the
|
||
|
/// requested position, else false.
|
||
|
public bool partial;
|
||
|
/// The path corridor the agent is using.
|
||
|
public PathCorridor corridor;
|
||
|
/// The local boundary data for the agent.
|
||
|
public LocalBoundary boundary;
|
||
|
/// Time since the agent's path corridor was optimized.
|
||
|
public float topologyOptTime;
|
||
|
/// The known neighbors of the agent.
|
||
|
public List<Crowd.CrowdNeighbour> neis = new();
|
||
|
/// The desired speed.
|
||
|
public float desiredSpeed;
|
||
|
|
||
|
public float[] npos = new float[3]; /// < The current agent position. [(x, y, z)]
|
||
|
public float[] disp = new float[3]; /// < A temporary value used to accumulate agent displacement during iterative
|
||
|
/// collision resolution. [(x, y, z)]
|
||
|
public float[] dvel = new float[3]; /// < The desired velocity of the agent. Based on the current path, calculated
|
||
|
/// from
|
||
|
/// scratch each frame. [(x, y, z)]
|
||
|
public float[] nvel = new float[3]; /// < The desired velocity adjusted by obstacle avoidance, calculated from scratch each
|
||
|
/// frame. [(x, y, z)]
|
||
|
public float[] vel = new float[3]; /// < The actual velocity of the agent. The change from nvel -> vel is
|
||
|
/// constrained by max acceleration. [(x, y, z)]
|
||
|
|
||
|
/// The agent's configuration parameters.
|
||
|
public CrowdAgentParams option;
|
||
|
/// The local path corridor corners for the agent.
|
||
|
public List<StraightPathItem> corners = new();
|
||
|
|
||
|
public MoveRequestState targetState; /// < State of the movement request.
|
||
|
public long targetRef; /// < Target polyref of the movement request.
|
||
|
public float[] targetPos = new float[3]; /// < Target position of the movement request (or velocity in case of
|
||
|
/// DT_CROWDAGENT_TARGET_VELOCITY).
|
||
|
public PathQueryResult targetPathQueryResult; /// < Path finder query
|
||
|
public bool targetReplan; /// < Flag indicating that the current path is being replanned.
|
||
|
public float targetReplanTime; /// <Time since the agent's target was replanned.
|
||
|
public float targetReplanWaitTime;
|
||
|
|
||
|
public CrowdAgentAnimation animation;
|
||
|
|
||
|
public CrowdAgent(int idx) {
|
||
|
this.idx = idx;
|
||
|
corridor = new PathCorridor();
|
||
|
boundary = new LocalBoundary();
|
||
|
animation = new CrowdAgentAnimation();
|
||
|
}
|
||
|
|
||
|
public void integrate(float dt) {
|
||
|
// Fake dynamic constraint.
|
||
|
float maxDelta = option.maxAcceleration * dt;
|
||
|
float[] dv = vSub(nvel, vel);
|
||
|
float ds = vLen(dv);
|
||
|
if (ds > maxDelta)
|
||
|
dv = vScale(dv, maxDelta / ds);
|
||
|
vel = vAdd(vel, dv);
|
||
|
|
||
|
// Integrate
|
||
|
if (vLen(vel) > 0.0001f)
|
||
|
npos = vMad(npos, vel, dt);
|
||
|
else
|
||
|
vSet(vel, 0, 0, 0);
|
||
|
}
|
||
|
|
||
|
public bool overOffmeshConnection(float radius) {
|
||
|
if (0 == corners.Count)
|
||
|
return false;
|
||
|
|
||
|
bool offMeshConnection = ((corners[corners.Count - 1].getFlags()
|
||
|
& NavMeshQuery.DT_STRAIGHTPATH_OFFMESH_CONNECTION) != 0) ? true : false;
|
||
|
if (offMeshConnection) {
|
||
|
float distSq = vDist2DSqr(npos, corners[corners.Count - 1].getPos());
|
||
|
if (distSq < radius * radius)
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
public float getDistanceToGoal(float range) {
|
||
|
if (0 == corners.Count)
|
||
|
return range;
|
||
|
|
||
|
bool endOfPath = ((corners[corners.Count - 1].getFlags() & NavMeshQuery.DT_STRAIGHTPATH_END) != 0) ? true : false;
|
||
|
if (endOfPath)
|
||
|
return Math.Min(vDist2D(npos, corners[corners.Count - 1].getPos()), range);
|
||
|
|
||
|
return range;
|
||
|
}
|
||
|
|
||
|
public float[] calcSmoothSteerDirection() {
|
||
|
float[] dir = new float[3];
|
||
|
if (0 < corners.Count) {
|
||
|
|
||
|
int ip0 = 0;
|
||
|
int ip1 = Math.Min(1, corners.Count - 1);
|
||
|
float[] p0 = corners[ip0].getPos();
|
||
|
float[] p1 = corners[ip1].getPos();
|
||
|
|
||
|
float[] dir0 = vSub(p0, npos);
|
||
|
float[] dir1 = vSub(p1, npos);
|
||
|
dir0[1] = 0;
|
||
|
dir1[1] = 0;
|
||
|
|
||
|
float len0 = vLen(dir0);
|
||
|
float len1 = vLen(dir1);
|
||
|
if (len1 > 0.001f)
|
||
|
dir1 = vScale(dir1, 1.0f / len1);
|
||
|
|
||
|
dir[0] = dir0[0] - dir1[0] * len0 * 0.5f;
|
||
|
dir[1] = 0;
|
||
|
dir[2] = dir0[2] - dir1[2] * len0 * 0.5f;
|
||
|
|
||
|
vNormalize(dir);
|
||
|
}
|
||
|
return dir;
|
||
|
}
|
||
|
|
||
|
public float[] calcStraightSteerDirection() {
|
||
|
float[] dir = new float[3];
|
||
|
if (0 < corners.Count) {
|
||
|
dir = vSub(corners[0].getPos(), npos);
|
||
|
dir[1] = 0;
|
||
|
vNormalize(dir);
|
||
|
}
|
||
|
return dir;
|
||
|
}
|
||
|
|
||
|
public void setTarget(long refs, float[] pos) {
|
||
|
targetRef = refs;
|
||
|
vCopy(targetPos, pos);
|
||
|
targetPathQueryResult = null;
|
||
|
if (targetRef != 0) {
|
||
|
targetState = MoveRequestState.DT_CROWDAGENT_TARGET_REQUESTING;
|
||
|
} else {
|
||
|
targetState = MoveRequestState.DT_CROWDAGENT_TARGET_FAILED;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|