You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1645 lines
65 KiB
C#

3 weeks ago
3 weeks ago
using System;
using System.Collections.Generic;
using Unity.Profiling;
using UnityEngine;
namespace Fusion.Addons.SimpleKCC
{
[DisallowMultipleComponent]
[RequireComponent(typeof(Rigidbody))]
public sealed class KCCOffline : MonoBehaviour
{
internal const int CACHE_SIZE = 64;
internal const int HISTORY_SIZE = 60;
[SerializeField]
private KCCSettings _settings = new();
private Transform _transform;
private Rigidbody _rigidbody;
private bool _isSpawned;
private KCCCollider _collider = new();
private KCCData _fixedData = new();
private KCCData[] _historyData = new KCCData[60];
private KCCSettings _defaultSettings = new();
private KCCOverlapInfo _extendedOverlapInfo = new(64);
private KCCOverlapInfo _trackOverlapInfo = new(64);
private List<Collider> _childColliders = new List<Collider>();
private RaycastHit[] _raycastHits = new RaycastHit[64];
private Collider[] _hitColliders = new Collider[64];
private Collider[] _addColliders = new Collider[64];
private Collider[] _removeColliders = new Collider[64];
private KCCResolver _resolver = new(64);
private Vector3 _lastAntiJitterPosition;
private Vector3 _predictionError;
private static ProfilerMarker _fixedUpdateMarker = new ProfilerMarker("KCC.FixedUpdate");
public Func<KCCOffline, Collider, bool> ResolveCollision;
internal bool IsSpawned => _isSpawned;
public bool IsGrounded => Data.IsGrounded;
public Vector3 RealVelocity => Data.RealVelocity;
public Vector3 Position => Data.TargetPosition;
public Quaternion LookRotation => Data.LookRotation;
public Vector3 LookDirection => Data.LookDirection;
public Quaternion TransformRotation => Data.TransformRotation;
public Vector3 TransformDirection => Data.TransformDirection;
public float RealSpeed => Data.RealSpeed;
public bool HasJumped => Data.HasJumped;
public bool IsActive => Data.IsActive;
internal KCCData Data => _fixedData;
internal KCCData FixedData => _fixedData;
public KCCSettings Settings => _settings;
public Transform Transform => _transform;
public CapsuleCollider Collider => _collider.Collider;
public Rigidbody Rigidbody => _rigidbody;
internal bool IsInFixedUpdate => true;
internal Vector3 PredictionError => _predictionError;
private event Action<KCCOffline> _onSpawn;
internal void SynchronizeTransform(bool synchronizePosition, bool synchronizeRotation, bool allowAntiJitter = true)
{
if (IsInFixedUpdate)
{
allowAntiJitter = false;
}
SynchronizeTransform(Data, synchronizePosition, synchronizeRotation, allowAntiJitter);
}
public void RefreshChildColliders()
{
_childColliders.Clear();
GetComponentsInChildren(includeInactive: true, _childColliders);
int num = 0;
int num2 = _childColliders.Count - 1;
while (num <= num2)
{
Collider collider = _childColliders[num];
if (collider.isTrigger || collider == _collider.Collider)
{
_childColliders[num] = _childColliders[num2];
_childColliders.RemoveAt(num2);
num2--;
}
else
{
num++;
}
}
}
internal KCCData GetHistoryData(int tick)
{
if (tick < 0)
{
return null;
}
KCCData kCCData = _historyData[tick % 60];
if (kCCData != null && kCCData.Tick == tick)
{
return kCCData;
}
return null;
}
internal void InvokeOnSpawn(Action<KCCOffline> callback)
{
if (_isSpawned)
{
callback(this);
return;
}
_onSpawn -= callback;
_onSpawn += callback;
}
internal void ManualFixedUpdate()
{
if (_isSpawned)
{
_fixedUpdateMarker.Begin();
OnFixedUpdateInternal();
_fixedUpdateMarker.End();
}
}
private void Awake()
{
_transform = base.transform;
_rigidbody = GetComponent<Rigidbody>();
_rigidbody.isKinematic = true;
RefreshCollider();
Spawned();
}
private void OnDestroy()
{
SetDefaults(cleanup: true);
}
private void OnDrawGizmosSelected()
{
if (_settings != null)
{
float num = Mathf.Max(0.01f, _settings.Radius);
float num2 = Mathf.Max(num * 2f, _settings.Height);
Vector3 position = base.transform.position;
Color color = Gizmos.color;
Vector3 vector = position + Vector3.up * num;
Vector3 vector2 = position + Vector3.up * (num2 - num);
Vector3 vector3 = Vector3.forward * num;
Vector3 vector4 = Vector3.back * num;
Vector3 vector5 = Vector3.left * num;
Vector3 vector6 = Vector3.right * num;
Gizmos.color = Color.green;
Gizmos.DrawWireSphere(vector, num);
Gizmos.DrawWireSphere(vector2, num);
Gizmos.DrawLine(vector + vector3, vector2 + vector3);
Gizmos.DrawLine(vector + vector4, vector2 + vector4);
Gizmos.DrawLine(vector + vector5, vector2 + vector5);
Gizmos.DrawLine(vector + vector6, vector2 + vector6);
if (_settings.Extent > 0f)
{
float radius = num + _settings.Extent;
Gizmos.color = Color.yellow;
Gizmos.DrawWireSphere(vector, radius);
Gizmos.DrawWireSphere(vector2, radius);
}
Gizmos.color = color;
}
}
public void Spawned()
{
if (_isSpawned)
{
throw new InvalidOperationException("[" + base.name + "] KCC is already spawned!");
}
_defaultSettings.CopyFromOther(_settings);
SetDefaults(cleanup: false);
_isSpawned = true;
// _isInFixedUpdate = true;
KCCUtility.GetClampedLookRotationAngles(_transform.rotation, out var pitch, out var yaw);
_fixedData = new();
_fixedData.Frame = Time.frameCount;
// _fixedData.Tick = base.Runner.Tick.Raw;
// _fixedData.Time = base.Runner.SimulationTime;
_fixedData.DeltaTime = Time.deltaTime;
_fixedData.UpdateDeltaTime = _fixedData.DeltaTime;
_fixedData.Gravity = Physics.gravity;
_fixedData.MaxGroundAngle = 60f;
_fixedData.MaxWallAngle = 5f;
_fixedData.MaxHangAngle = 30f;
_fixedData.BasePosition = _transform.position;
_fixedData.DesiredPosition = _transform.position;
_fixedData.TargetPosition = _transform.position;
_fixedData.LookPitch = pitch;
_fixedData.LookYaw = yaw;
// if (!base.Object.HasStateAuthority)
// {
// ReadNetworkData();
SynchronizeTransform(_fixedData, synchronizePosition: true, synchronizeRotation: true, allowAntiJitter: false);
// }
// _renderData = new();
// _renderData.CopyFromOther(_fixedData);
// _lastRenderPosition = _renderData.TargetPosition;
// _lastAntiJitterPosition = _renderData.TargetPosition;
RefreshCollider();
RefreshChildColliders();
if (this._onSpawn != null)
{
try
{
this._onSpawn(this);
}
catch (Exception exception)
{
Debug.LogException(exception);
}
this._onSpawn = null;
}
}
private void OnFixedUpdateInternal()
{
if (!IsInFixedUpdate)
{
throw new InvalidOperationException("[" + base.name + "] KCC fixed update called from render update! This is not allowed.");
}
RefreshCollider();
MovePredicted(_fixedData);
if (_fixedData.IsActive)
{
SynchronizeTransform(_fixedData, synchronizePosition: true, synchronizeRotation: true, allowAntiJitter: false);
}
}
private void MovePredicted(KCCData data)
{
float time = data.Time;
float deltaTime = data.DeltaTime;
Vector3 targetPosition = data.TargetPosition;
Vector3 targetPosition2 = data.TargetPosition;
bool isGrounded = data.IsGrounded;
bool isSteppingUp = data.IsSteppingUp;
bool isSnappingToGround = data.IsSnappingToGround;
data.DeltaTime = deltaTime;
data.BasePosition = targetPosition;
data.DesiredPosition = targetPosition2;
if (!data.IsActive)
{
data.ClearTransientProperties();
ForceRemoveAllHits(data);
return;
}
SetBaseProperties(data);
if (!data.IsActive)
{
data.ClearTransientProperties();
ForceRemoveAllHits(data);
return;
}
deltaTime = data.DeltaTime;
targetPosition = data.BasePosition;
if (deltaTime < KCCSettings.ExtrapolationDeltaTimeThreshold)
{
Vector3 vector = data.DesiredVelocity;
if (data.RealVelocity.sqrMagnitude <= vector.sqrMagnitude)
{
vector = data.RealVelocity;
}
targetPosition2 = targetPosition + vector * deltaTime;
data.BasePosition = targetPosition;
data.DesiredPosition = targetPosition2;
data.TargetPosition = targetPosition2;
return;
}
EnvironmentProcessor.SetDynamicVelocity(this, data);
ForceRemoveAllHits(data);
float num = Mathf.Clamp01(deltaTime);
Vector3 vector2 = data.DesiredVelocity * num + data.ExternalDelta;
targetPosition2 = (data.DesiredPosition = data.BasePosition + vector2);
data.TargetPosition = data.BasePosition;
data.ExternalDelta = default(Vector3);
bool flag = false;
float num2 = Mathf.Clamp(_settings.CCDRadiusMultiplier, 0.25f, 0.75f);
float num3 = _settings.Radius * (num2 + 0.1f);
float num4 = _settings.Radius * num2;
Vector3 targetPosition3 = data.TargetPosition;
while (!flag && !data.HasTeleported)
{
data.BasePosition = data.TargetPosition;
float num5 = num;
Vector3 vector3 = vector2;
float magnitude = vector3.magnitude;
if (magnitude > num3)
{
float num6 = num4 / magnitude;
num5 *= num6;
vector3 *= num6;
}
else
{
flag = true;
}
num -= num5;
vector2 -= vector3;
if (num <= 0f)
{
num = 0f;
}
data.Time = time - num;
data.DeltaTime = num5;
data.DesiredPosition = data.BasePosition + vector3;
data.TargetPosition = data.DesiredPosition;
data.WasGrounded = data.IsGrounded;
data.WasSteppingUp = data.IsSteppingUp;
data.WasSnappingToGround = data.IsSnappingToGround;
ProcessMoveStep(data);
if (!data.HasTeleported)
{
targetPosition3 = data.TargetPosition;
}
if (data.HasTeleported)
{
UpdateHits(data, null, EKCCHitsOverlapQuery.New);
}
if (flag && !data.ExternalDelta.IsZero())
{
vector2 += data.ExternalDelta;
data.ExternalDelta = default(Vector3);
flag = false;
}
}
data.Time = time;
data.DeltaTime = deltaTime;
data.BasePosition = targetPosition;
data.DesiredPosition = targetPosition2;
data.WasGrounded = isGrounded;
data.WasSteppingUp = isSteppingUp;
data.WasSnappingToGround = isSnappingToGround;
data.RealVelocity = (targetPosition3 - data.BasePosition) / data.DeltaTime;
data.RealSpeed = data.RealVelocity.magnitude;
Vector3 targetPosition4 = data.TargetPosition;
if (!data.TargetPosition.IsEqual(targetPosition4))
{
UpdateHits(data, null, EKCCHitsOverlapQuery.New);
}
targetPosition4 = data.TargetPosition;
if (!data.TargetPosition.IsEqual(targetPosition4))
{
UpdateHits(data, null, EKCCHitsOverlapQuery.New);
}
}
private void SetBaseProperties(KCCData data)
{
data.HasTeleported = false;
data.MaxPenetrationSteps = _settings.MaxPenetrationSteps;
if (data.Frame == _fixedData.Frame)
{
data.JumpFrames = 0;
}
}
private void ProcessMoveStep(KCCData data)
{
data.IsGrounded = false;
data.IsSteppingUp = false;
data.IsSnappingToGround = false;
data.GroundNormal = default(Vector3);
data.GroundTangent = default(Vector3);
data.GroundPosition = default(Vector3);
data.GroundDistance = 0f;
data.GroundAngle = 0f;
ForceRemoveAllHits(data);
bool flag = data.JumpFrames > 0;
if ((int)_settings.CollisionLayerMask != 0 && _collider.IsSpawned)
{
float radius = _settings.Radius;
EKCCHitsOverlapQuery overlapQuery = EKCCHitsOverlapQuery.Default;
CapsuleOverlap(_extendedOverlapInfo, data, data.TargetPosition, _settings.Radius, _settings.Height, radius, _settings.CollisionLayerMask, QueryTriggerInteraction.Collide);
data.TargetPosition = ResolvePenetration(_extendedOverlapInfo, data, data.BasePosition, data.TargetPosition, !flag, data.MaxPenetrationSteps, 3, resolveTriggers: true);
UpdateHits(data, _extendedOverlapInfo, overlapQuery);
}
if (flag)
{
data.IsGrounded = false;
}
EnvironmentProcessor.AfterMoveStep(this, data);
StepUpProcessor.AfterMoveStep(this, data, _extendedOverlapInfo);
GroundSnapProcessor.AfterMoveStep(this, data, _extendedOverlapInfo);
}
private void SynchronizeTransform(KCCData data, bool synchronizePosition, bool synchronizeRotation, bool allowAntiJitter)
{
if (synchronizePosition)
{
Vector3 vector = data.TargetPosition;
_rigidbody.position = vector;
if (allowAntiJitter && !_settings.AntiJitterDistance.IsZero())
{
Vector3 vector2 = vector - _lastAntiJitterPosition;
if (vector2.sqrMagnitude < _settings.TeleportThreshold * _settings.TeleportThreshold)
{
vector = _lastAntiJitterPosition;
float num = Mathf.Abs(vector2.y);
if (num > 1E-06f)
{
vector.y += vector2.y * Mathf.Clamp01((num - _settings.AntiJitterDistance.y) / num);
}
Vector3 vector3 = vector2.OnlyXZ();
float num2 = Vector3.Magnitude(vector3);
if (num2 > 1E-06f)
{
vector += vector3 * Mathf.Clamp01((num2 - _settings.AntiJitterDistance.x) / num2);
}
}
_lastAntiJitterPosition = vector;
}
if (synchronizeRotation)
{
_transform.SetPositionAndRotation(vector, data.TransformRotation);
}
else
{
_transform.position = vector;
}
}
else if (synchronizeRotation)
{
_transform.rotation = data.TransformRotation;
}
}
private void RefreshCollider()
{
if (_settings.Shape == EKCCShape.None)
{
_collider.Destroy();
return;
}
_settings.Radius = Mathf.Max(0.01f, _settings.Radius);
_settings.Height = Mathf.Max(_settings.Radius * 2f, _settings.Height);
_settings.Extent = Mathf.Max(0f, _settings.Extent);
_collider.Update(this);
}
private void SetDefaults(bool cleanup)
{
_fixedData.Clear();
// _renderData.Clear();
Array.Clear(_historyData, 0, _historyData.Length);
_extendedOverlapInfo.Reset(deep: true);
_trackOverlapInfo.Reset(deep: true);
_childColliders.Clear();
Array.Clear(_raycastHits, 0, _raycastHits.Length);
Array.Clear(_hitColliders, 0, _hitColliders.Length);
Array.Clear(_addColliders, 0, _addColliders.Length);
Array.Clear(_removeColliders, 0, _removeColliders.Length);
// _raycastHits.Clear();
// _hitColliders.Clear();
// _addColliders.Clear();
// _removeColliders.Clear();
_rigidbody.isKinematic = true;
// _lastRenderTime = 0f;
// _lastRenderPosition = default(Vector3);
_lastAntiJitterPosition = default(Vector3);
_predictionError = default(Vector3);
if (cleanup)
{
_isSpawned = false;
// _isInFixedUpdate = false;
_settings.CopyFromOther(_defaultSettings);
_collider.Destroy();
this._onSpawn = null;
ResolveCollision = null;
}
}
internal Vector3 ResolvePenetration(KCCOverlapInfo overlapInfo, KCCData data, Vector3 basePosition, Vector3 targetPosition, bool probeGrounding, int maxSteps, int resolverIterations, bool resolveTriggers)
{
if (_settings.SuppressConvexMeshColliders)
{
overlapInfo.ToggleConvexMeshColliders(convex: false);
}
if (overlapInfo.ColliderHitCount == 1)
{
targetPosition = DepenetrateSingle(overlapInfo, data, basePosition, targetPosition, probeGrounding, maxSteps);
}
else if (overlapInfo.ColliderHitCount > 1)
{
targetPosition = DepenetrateMultiple(overlapInfo, data, basePosition, targetPosition, probeGrounding, maxSteps, resolverIterations);
}
RecalculateGroundProperties(data);
if (resolveTriggers)
{
for (int i = 0; i < overlapInfo.TriggerHitCount; i++)
{
KCCOverlapHit kCCOverlapHit = overlapInfo.TriggerHits[i];
kCCOverlapHit.Transform.GetPositionAndRotation(out kCCOverlapHit.CachedPosition, out kCCOverlapHit.CachedRotation);
kCCOverlapHit.CollisionType = ((kCCOverlapHit.IsWithinExtent = (kCCOverlapHit.HasPenetration = Physics.ComputePenetration(_collider.Collider, data.TargetPosition, Quaternion.identity, kCCOverlapHit.Collider, kCCOverlapHit.CachedPosition, kCCOverlapHit.CachedRotation, out var _, out var distance))) ? ECollisionType.Trigger : ECollisionType.None);
if (distance > kCCOverlapHit.MaxPenetration)
{
kCCOverlapHit.MaxPenetration = distance;
}
}
}
if (_settings.SuppressConvexMeshColliders)
{
overlapInfo.ToggleConvexMeshColliders(convex: true);
}
return targetPosition;
}
private Vector3 DepenetrateSingle(KCCOverlapInfo overlapInfo, KCCData data, Vector3 basePosition, Vector3 targetPosition, bool probeGrounding, int maxSteps)
{
float num = Mathf.Cos(Mathf.Clamp(data.MaxGroundAngle, 0f, 90f) * (MathF.PI / 180f));
float num2 = 0f - Mathf.Cos(Mathf.Clamp(90f - data.MaxWallAngle, 0f, 90f) * (MathF.PI / 180f));
float num3 = 0f - Mathf.Cos(Mathf.Clamp(90f - data.MaxHangAngle, 0f, 90f) * (MathF.PI / 180f));
Vector3 vector = Vector3.up;
float num4 = 0f;
KCCOverlapHit kCCOverlapHit = overlapInfo.ColliderHits[0];
kCCOverlapHit.UpDirectionDot = float.MinValue;
kCCOverlapHit.Transform.GetPositionAndRotation(out kCCOverlapHit.CachedPosition, out kCCOverlapHit.CachedRotation);
if (maxSteps > 1)
{
float num5 = 0.001f;
float num6 = Vector3.Distance(basePosition, targetPosition);
if (num6 < (float)maxSteps * num5)
{
maxSteps = Mathf.Max(1, (int)(num6 / num5));
}
}
if (maxSteps <= 1)
{
kCCOverlapHit.HasPenetration = Physics.ComputePenetration(_collider.Collider, targetPosition, Quaternion.identity, kCCOverlapHit.Collider, kCCOverlapHit.CachedPosition, kCCOverlapHit.CachedRotation, out var direction, out var distance);
if (kCCOverlapHit.HasPenetration)
{
kCCOverlapHit.IsWithinExtent = true;
if (distance > kCCOverlapHit.MaxPenetration)
{
kCCOverlapHit.MaxPenetration = distance;
}
float num7 = Vector3.Dot(direction, Vector3.up);
if (num7 > kCCOverlapHit.UpDirectionDot)
{
kCCOverlapHit.UpDirectionDot = num7;
if (num7 >= num)
{
kCCOverlapHit.CollisionType = ECollisionType.Ground;
data.IsGrounded = true;
vector = direction;
}
else if (num7 > 0f - num2)
{
kCCOverlapHit.CollisionType = ECollisionType.Slope;
}
else if (num7 >= num2)
{
kCCOverlapHit.CollisionType = ECollisionType.Wall;
}
else if (num7 >= num3)
{
kCCOverlapHit.CollisionType = ECollisionType.Hang;
}
else
{
kCCOverlapHit.CollisionType = ECollisionType.Top;
}
}
if (num7 > 0f && num7 < num && distance >= 1E-06f && data.DynamicVelocity.y <= 0f && Vector3.Dot((targetPosition - basePosition).OnlyXZ(), direction.OnlyXZ()) < 0f)
{
KCCPhysicsUtility.ProjectVerticalPenetration(ref direction, ref distance);
}
targetPosition += direction * distance;
}
}
else
{
Vector3 vector2 = (targetPosition - basePosition) / maxSteps;
Vector3 vector3 = basePosition;
int num8 = maxSteps;
while (num8 > 0)
{
num8--;
vector3 += vector2;
kCCOverlapHit.HasPenetration = Physics.ComputePenetration(_collider.Collider, vector3, Quaternion.identity, kCCOverlapHit.Collider, kCCOverlapHit.CachedPosition, kCCOverlapHit.CachedRotation, out var direction2, out var distance2);
if (!kCCOverlapHit.HasPenetration)
{
continue;
}
kCCOverlapHit.IsWithinExtent = true;
if (distance2 > kCCOverlapHit.MaxPenetration)
{
kCCOverlapHit.MaxPenetration = distance2;
}
float num9 = Vector3.Dot(direction2, Vector3.up);
if (num9 > kCCOverlapHit.UpDirectionDot)
{
kCCOverlapHit.UpDirectionDot = num9;
if (num9 >= num)
{
kCCOverlapHit.CollisionType = ECollisionType.Ground;
data.IsGrounded = true;
vector = direction2;
}
else if (num9 > 0f - num2)
{
kCCOverlapHit.CollisionType = ECollisionType.Slope;
}
else if (num9 >= num2)
{
kCCOverlapHit.CollisionType = ECollisionType.Wall;
}
else if (num9 >= num3)
{
kCCOverlapHit.CollisionType = ECollisionType.Hang;
}
else
{
kCCOverlapHit.CollisionType = ECollisionType.Top;
}
}
if (num9 > 0f && num9 < num && distance2 >= 1E-06f && data.DynamicVelocity.y <= 0f && Vector3.Dot(vector2.OnlyXZ(), direction2.OnlyXZ()) < 0f)
{
KCCPhysicsUtility.ProjectVerticalPenetration(ref direction2, ref distance2);
}
vector3 += direction2 * distance2;
}
targetPosition = vector3;
}
if (kCCOverlapHit.UpDirectionDot == float.MinValue)
{
kCCOverlapHit.UpDirectionDot = 0f;
}
if (probeGrounding && !data.IsGrounded)
{
if (KCCPhysicsUtility.CheckGround(_collider.Collider, targetPosition, kCCOverlapHit.Collider, kCCOverlapHit.CachedPosition, kCCOverlapHit.CachedRotation, _settings.Radius, _settings.Height, _settings.Extent, num, out var groundNormal, out var groundDistance, out var isWithinExtent))
{
data.IsGrounded = true;
vector = groundNormal;
num4 = groundDistance;
kCCOverlapHit.IsWithinExtent = true;
kCCOverlapHit.CollisionType = ECollisionType.Ground;
}
else if (isWithinExtent)
{
kCCOverlapHit.IsWithinExtent = true;
if (kCCOverlapHit.CollisionType == ECollisionType.None)
{
kCCOverlapHit.CollisionType = ECollisionType.Slope;
}
}
}
if (data.IsGrounded)
{
data.GroundNormal = vector;
data.GroundAngle = Vector3.Angle(vector, Vector3.up);
data.GroundPosition = targetPosition + new Vector3(0f, _settings.Radius, 0f) - vector * (_settings.Radius + num4);
data.GroundDistance = num4;
}
return targetPosition;
}
private Vector3 DepenetrateMultiple(KCCOverlapInfo overlapInfo, KCCData data, Vector3 basePosition, Vector3 targetPosition, bool probeGrounding, int maxSteps, int resolverIterations)
{
float num = Mathf.Cos(Mathf.Clamp(data.MaxGroundAngle, 0f, 90f) * (MathF.PI / 180f));
float num2 = 0f - Mathf.Cos(Mathf.Clamp(90f - data.MaxWallAngle, 0f, 90f) * (MathF.PI / 180f));
float num3 = 0f - Mathf.Cos(Mathf.Clamp(90f - data.MaxHangAngle, 0f, 90f) * (MathF.PI / 180f));
float num4 = 0f;
float num5 = 0f;
Vector3 other = default(Vector3);
Vector3 vector = default(Vector3);
Vector3 lhs = (targetPosition - basePosition).OnlyXZ();
for (int i = 0; i < overlapInfo.ColliderHitCount; i++)
{
KCCOverlapHit kCCOverlapHit = overlapInfo.ColliderHits[i];
kCCOverlapHit.UpDirectionDot = float.MinValue;
kCCOverlapHit.Transform.GetPositionAndRotation(out kCCOverlapHit.CachedPosition, out kCCOverlapHit.CachedRotation);
}
if (maxSteps > 1)
{
float num6 = 0.001f;
float num7 = Vector3.Distance(basePosition, targetPosition);
if (num7 < (float)maxSteps * num6)
{
maxSteps = Mathf.Max(1, (int)(num7 / num6));
}
}
if (maxSteps <= 1)
{
_resolver.Reset();
for (int j = 0; j < overlapInfo.ColliderHitCount; j++)
{
KCCOverlapHit kCCOverlapHit2 = overlapInfo.ColliderHits[j];
kCCOverlapHit2.HasPenetration = Physics.ComputePenetration(_collider.Collider, targetPosition, Quaternion.identity, kCCOverlapHit2.Collider, kCCOverlapHit2.CachedPosition, kCCOverlapHit2.CachedRotation, out var direction, out var distance);
if (!kCCOverlapHit2.HasPenetration)
{
continue;
}
kCCOverlapHit2.IsWithinExtent = true;
if (distance > kCCOverlapHit2.MaxPenetration)
{
kCCOverlapHit2.MaxPenetration = distance;
}
float num8 = Vector3.Dot(direction, Vector3.up);
if (num8 > kCCOverlapHit2.UpDirectionDot)
{
kCCOverlapHit2.UpDirectionDot = num8;
if (num8 >= num)
{
kCCOverlapHit2.CollisionType = ECollisionType.Ground;
data.IsGrounded = true;
if (num8 >= num5)
{
num5 = num8;
other = direction;
}
vector += direction * num8;
}
else if (num8 > 0f - num2)
{
kCCOverlapHit2.CollisionType = ECollisionType.Slope;
}
else if (num8 >= num2)
{
kCCOverlapHit2.CollisionType = ECollisionType.Wall;
}
else if (num8 >= num3)
{
kCCOverlapHit2.CollisionType = ECollisionType.Hang;
}
else
{
kCCOverlapHit2.CollisionType = ECollisionType.Top;
}
}
if (num8 > 0f && num8 < num && distance >= 1E-06f && data.DynamicVelocity.y <= 0f && Vector3.Dot(lhs, direction.OnlyXZ()) < 0f)
{
KCCPhysicsUtility.ProjectVerticalPenetration(ref direction, ref distance);
}
_resolver.AddCorrection(direction, distance);
}
int num9 = Mathf.Max(0, resolverIterations);
float num10 = 1f - (float)Mathf.Min(num9, 2) * 0.25f;
if (_resolver.Size == 2)
{
_resolver.GetCorrection(0, out var direction2);
_resolver.GetCorrection(1, out var direction3);
if (Vector3.Dot(direction2, direction3) >= 0f)
{
targetPosition += _resolver.CalculateMinMax() * num10;
}
else
{
targetPosition += _resolver.CalculateBinary() * num10;
}
}
else
{
targetPosition += _resolver.CalculateGradientDescent(12, 0.0001f) * num10;
}
while (num9 > 0)
{
num9--;
_resolver.Reset();
for (int k = 0; k < overlapInfo.ColliderHitCount; k++)
{
KCCOverlapHit kCCOverlapHit3 = overlapInfo.ColliderHits[k];
if (!Physics.ComputePenetration(_collider.Collider, targetPosition, Quaternion.identity, kCCOverlapHit3.Collider, kCCOverlapHit3.CachedPosition, kCCOverlapHit3.CachedRotation, out var direction4, out var distance2))
{
continue;
}
kCCOverlapHit3.IsWithinExtent = true;
kCCOverlapHit3.HasPenetration = true;
if (distance2 > kCCOverlapHit3.MaxPenetration)
{
kCCOverlapHit3.MaxPenetration = distance2;
}
float num11 = Vector3.Dot(direction4, Vector3.up);
if (num11 > kCCOverlapHit3.UpDirectionDot)
{
kCCOverlapHit3.UpDirectionDot = num11;
if (num11 >= num)
{
kCCOverlapHit3.CollisionType = ECollisionType.Ground;
data.IsGrounded = true;
if (num11 >= num5)
{
num5 = num11;
other = direction4;
}
vector += direction4 * num11;
}
else if (num11 > 0f - num2)
{
kCCOverlapHit3.CollisionType = ECollisionType.Slope;
}
else if (num11 >= num2)
{
kCCOverlapHit3.CollisionType = ECollisionType.Wall;
}
else if (num11 >= num3)
{
kCCOverlapHit3.CollisionType = ECollisionType.Hang;
}
else
{
kCCOverlapHit3.CollisionType = ECollisionType.Top;
}
}
if (num11 > 0f && num11 < num && distance2 >= 1E-06f && data.DynamicVelocity.y <= 0f && Vector3.Dot(lhs, direction4.OnlyXZ()) < 0f)
{
KCCPhysicsUtility.ProjectVerticalPenetration(ref direction4, ref distance2);
}
_resolver.AddCorrection(direction4, distance2);
}
if (_resolver.Size == 0)
{
break;
}
switch (num9)
{
case 0:
if (_resolver.Size == 2)
{
_resolver.GetCorrection(0, out var direction5);
_resolver.GetCorrection(1, out var direction6);
if (Vector3.Dot(direction5, direction6) >= 0f)
{
targetPosition += _resolver.CalculateGradientDescent(12, 0.0001f);
}
else
{
targetPosition += _resolver.CalculateBinary();
}
}
else
{
targetPosition += _resolver.CalculateGradientDescent(12, 0.0001f);
}
break;
case 1:
targetPosition += _resolver.CalculateMinMax() * 0.75f;
break;
default:
targetPosition += _resolver.CalculateMinMax() * 0.5f;
break;
}
}
}
else
{
Vector3 vector2 = (targetPosition - basePosition) / maxSteps;
Vector3 vector3 = basePosition;
int num12 = maxSteps;
while (num12 > 1)
{
num12--;
vector3 += vector2;
_resolver.Reset();
for (int l = 0; l < overlapInfo.ColliderHitCount; l++)
{
KCCOverlapHit kCCOverlapHit4 = overlapInfo.ColliderHits[l];
kCCOverlapHit4.HasPenetration = Physics.ComputePenetration(_collider.Collider, vector3, Quaternion.identity, kCCOverlapHit4.Collider, kCCOverlapHit4.CachedPosition, kCCOverlapHit4.CachedRotation, out var direction7, out var distance3);
if (!kCCOverlapHit4.HasPenetration)
{
continue;
}
kCCOverlapHit4.IsWithinExtent = true;
if (distance3 > kCCOverlapHit4.MaxPenetration)
{
kCCOverlapHit4.MaxPenetration = distance3;
}
float num13 = Vector3.Dot(direction7, Vector3.up);
if (num13 > kCCOverlapHit4.UpDirectionDot)
{
kCCOverlapHit4.UpDirectionDot = num13;
if (num13 >= num)
{
kCCOverlapHit4.CollisionType = ECollisionType.Ground;
data.IsGrounded = true;
if (num13 >= num5)
{
num5 = num13;
other = direction7;
}
vector += direction7 * num13;
}
else if (num13 > 0f - num2)
{
kCCOverlapHit4.CollisionType = ECollisionType.Slope;
}
else if (num13 >= num2)
{
kCCOverlapHit4.CollisionType = ECollisionType.Wall;
}
else if (num13 >= num3)
{
kCCOverlapHit4.CollisionType = ECollisionType.Hang;
}
else
{
kCCOverlapHit4.CollisionType = ECollisionType.Top;
}
}
if (num13 > 0f && num13 < num && distance3 >= 1E-06f && data.DynamicVelocity.y <= 0f && Vector3.Dot(vector2.OnlyXZ(), direction7.OnlyXZ()) < 0f)
{
KCCPhysicsUtility.ProjectVerticalPenetration(ref direction7, ref distance3);
}
_resolver.AddCorrection(direction7, distance3);
}
if (_resolver.Size == 2)
{
_resolver.GetCorrection(0, out var direction8);
_resolver.GetCorrection(1, out var direction9);
if (Vector3.Dot(direction8, direction9) >= 0f)
{
vector3 += _resolver.CalculateMinMax();
}
else
{
vector3 += _resolver.CalculateBinary();
}
}
else
{
vector3 += _resolver.CalculateMinMax();
}
}
num12--;
vector3 += vector2;
_resolver.Reset();
for (int m = 0; m < overlapInfo.ColliderHitCount; m++)
{
KCCOverlapHit kCCOverlapHit5 = overlapInfo.ColliderHits[m];
kCCOverlapHit5.HasPenetration = Physics.ComputePenetration(_collider.Collider, vector3, Quaternion.identity, kCCOverlapHit5.Collider, kCCOverlapHit5.CachedPosition, kCCOverlapHit5.CachedRotation, out var direction10, out var distance4);
if (!kCCOverlapHit5.HasPenetration)
{
continue;
}
kCCOverlapHit5.IsWithinExtent = true;
if (distance4 > kCCOverlapHit5.MaxPenetration)
{
kCCOverlapHit5.MaxPenetration = distance4;
}
float num14 = Vector3.Dot(direction10, Vector3.up);
if (num14 > kCCOverlapHit5.UpDirectionDot)
{
kCCOverlapHit5.UpDirectionDot = num14;
if (num14 >= num)
{
kCCOverlapHit5.CollisionType = ECollisionType.Ground;
data.IsGrounded = true;
if (num14 >= num5)
{
num5 = num14;
other = direction10;
}
vector += direction10 * num14;
}
else if (num14 > 0f - num2)
{
kCCOverlapHit5.CollisionType = ECollisionType.Slope;
}
else if (num14 >= num2)
{
kCCOverlapHit5.CollisionType = ECollisionType.Wall;
}
else if (num14 >= num3)
{
kCCOverlapHit5.CollisionType = ECollisionType.Hang;
}
else
{
kCCOverlapHit5.CollisionType = ECollisionType.Top;
}
}
if (num14 > 0f && num14 < num && distance4 >= 1E-06f && data.DynamicVelocity.y <= 0f && Vector3.Dot(vector2.OnlyXZ(), direction10.OnlyXZ()) < 0f)
{
KCCPhysicsUtility.ProjectVerticalPenetration(ref direction10, ref distance4);
}
_resolver.AddCorrection(direction10, distance4);
}
if (_resolver.Size == 2)
{
_resolver.GetCorrection(0, out var direction11);
_resolver.GetCorrection(1, out var direction12);
if (Vector3.Dot(direction11, direction12) >= 0f)
{
vector3 += _resolver.CalculateMinMax();
}
else
{
vector3 += _resolver.CalculateBinary();
}
}
else
{
vector3 += _resolver.CalculateGradientDescent(12, 0.0001f);
}
targetPosition = vector3;
}
for (int n = 0; n < overlapInfo.ColliderHitCount; n++)
{
KCCOverlapHit kCCOverlapHit6 = overlapInfo.ColliderHits[n];
if (kCCOverlapHit6.UpDirectionDot == float.MinValue)
{
kCCOverlapHit6.UpDirectionDot = 0f;
}
}
if (probeGrounding && !data.IsGrounded)
{
Vector3 vector4 = Vector3.up;
float num15 = 1000f;
for (int num16 = 0; num16 < overlapInfo.ColliderHitCount; num16++)
{
KCCOverlapHit kCCOverlapHit7 = overlapInfo.ColliderHits[num16];
if (KCCPhysicsUtility.CheckGround(_collider.Collider, targetPosition, kCCOverlapHit7.Collider, kCCOverlapHit7.CachedPosition, kCCOverlapHit7.CachedRotation, _settings.Radius, _settings.Height, _settings.Extent, num, out var groundNormal, out var groundDistance, out var isWithinExtent))
{
data.IsGrounded = true;
if (groundDistance < num15)
{
vector4 = groundNormal;
num15 = groundDistance;
}
kCCOverlapHit7.IsWithinExtent = true;
kCCOverlapHit7.CollisionType = ECollisionType.Ground;
}
else if (isWithinExtent)
{
kCCOverlapHit7.IsWithinExtent = true;
if (kCCOverlapHit7.CollisionType == ECollisionType.None)
{
kCCOverlapHit7.CollisionType = ECollisionType.Slope;
}
}
}
if (data.IsGrounded)
{
other = vector4;
vector = vector4;
num4 = num15;
}
}
if (data.IsGrounded)
{
if (!vector.IsEqual(other))
{
vector.Normalize();
}
data.GroundNormal = vector;
data.GroundAngle = Vector3.Angle(data.GroundNormal, Vector3.up);
data.GroundPosition = targetPosition + new Vector3(0f, _settings.Radius, 0f) - data.GroundNormal * (_settings.Radius + num4);
data.GroundDistance = num4;
}
return targetPosition;
}
private static void RecalculateGroundProperties(KCCData data)
{
if (data.IsGrounded)
{
Vector3 projectedVector2;
if (KCCPhysicsUtility.ProjectOnGround(data.GroundNormal, data.GroundNormal.OnlyXZ(), out var projectedVector))
{
data.GroundTangent = projectedVector.normalized;
}
else if (KCCPhysicsUtility.ProjectOnGround(data.GroundNormal, data.DesiredVelocity.OnlyXZ(), out projectedVector2))
{
data.GroundTangent = projectedVector2.normalized;
}
else
{
data.GroundTangent = data.TransformDirection;
}
}
}
public void SetActive(bool isActive) => _fixedData.IsActive = isActive;
public Vector2 GetLookRotation(bool pitch = true, bool yaw = true) => Data.GetLookRotation(pitch, yaw);
public void AddLookRotation(float pitchDelta, float yawDelta)
{
KCCData kCCData = _fixedData;
kCCData.AddLookRotation(pitchDelta, yawDelta);
SynchronizeTransform(kCCData, synchronizePosition: false, synchronizeRotation: true, allowAntiJitter: false);
}
public void AddLookRotation(float pitchDelta, float yawDelta, float minPitch, float maxPitch)
{
KCCData kCCData = _fixedData;
kCCData.AddLookRotation(pitchDelta, yawDelta, minPitch, maxPitch);
SynchronizeTransform(kCCData, synchronizePosition: false, synchronizeRotation: true, allowAntiJitter: false);
}
public void AddLookRotation(Vector2 lookRotationDelta)
{
AddLookRotation(lookRotationDelta.x, lookRotationDelta.y);
}
public void AddLookRotation(Vector2 lookRotationDelta, float minPitch, float maxPitch)
{
AddLookRotation(lookRotationDelta.x, lookRotationDelta.y, minPitch, maxPitch);
}
public void SetLookRotation(float pitch, float yaw)
{
KCCData kCCData = _fixedData;
kCCData.SetLookRotation(pitch, yaw);
SynchronizeTransform(kCCData, synchronizePosition: false, synchronizeRotation: true, allowAntiJitter: false);
}
public void SetLookRotation(Vector2 lookRotation) => SetLookRotation(lookRotation.x, lookRotation.y);
public void SetLookRotation(Quaternion lookRotation, bool preservePitch = false, bool preserveYaw = false)
{
KCCData kCCData = _fixedData;
kCCData.SetLookRotation(lookRotation, preservePitch, preserveYaw);
SynchronizeTransform(kCCData, synchronizePosition: false, synchronizeRotation: true, allowAntiJitter: false);
}
public void SetPosition(Vector3 position)
{
KCCData kCCData = _fixedData;
kCCData.BasePosition = position;
kCCData.DesiredPosition = position;
kCCData.TargetPosition = position;
kCCData.HasTeleported = true;
kCCData.IsSteppingUp = false;
kCCData.IsSnappingToGround = false;
SynchronizeTransform(kCCData, synchronizePosition: true, synchronizeRotation: false, allowAntiJitter: false);
}
public void SetShape(EKCCShape shape, float radius = 0f, float height = 0f)
{
_settings.Shape = shape;
if (radius > 0f)
{
_settings.Radius = radius;
}
if (height > 0f)
{
_settings.Height = height;
}
RefreshCollider();
}
public void SetTrigger(bool isTrigger)
{
_settings.IsTrigger = isTrigger;
RefreshCollider();
}
public void SetRadius(float radius)
{
if (!(radius <= 0f))
{
_settings.Radius = radius;
RefreshCollider();
}
}
public void SetHeight(float height)
{
if (!(height <= 0f))
{
_settings.Height = height;
RefreshCollider();
}
}
public void SetColliderLayer(int layer)
{
_settings.ColliderLayer = layer;
RefreshCollider();
}
public void SetCollisionLayerMask(LayerMask layerMask)
{
_settings.CollisionLayerMask = layerMask;
}
private void RestoreHistoryData(KCCData historyData)
{
if (_fixedData.IsGrounded)
{
_fixedData.IsGrounded = historyData.IsGrounded;
_fixedData.WasGrounded = historyData.WasGrounded;
}
}
internal bool SphereOverlap(KCCOverlapInfo overlapInfo, Vector3 position, float radius, QueryTriggerInteraction triggerInteraction)
{
return SphereOverlap(overlapInfo, Data, position, radius, 0f, _settings.CollisionLayerMask, triggerInteraction);
}
internal bool CapsuleOverlap(KCCOverlapInfo overlapInfo, Vector3 position, float radius, float height, QueryTriggerInteraction triggerInteraction)
{
return CapsuleOverlap(overlapInfo, Data, position, radius, height, 0f, _settings.CollisionLayerMask, triggerInteraction);
}
internal bool RayCast(KCCShapeCastInfo shapeCastInfo, Vector3 position, Vector3 direction, float maxDistance, QueryTriggerInteraction triggerInteraction)
{
return RayCast(shapeCastInfo, Data, position, direction, maxDistance, _settings.CollisionLayerMask, triggerInteraction);
}
internal bool SphereCast(KCCShapeCastInfo shapeCastInfo, Vector3 position, float radius, Vector3 direction, float maxDistance, QueryTriggerInteraction triggerInteraction, bool trackInitialOverlaps = true)
{
return SphereCast(shapeCastInfo, Data, position, radius, 0f, direction, maxDistance, _settings.CollisionLayerMask, triggerInteraction, trackInitialOverlaps);
}
internal bool CapsuleCast(KCCShapeCastInfo shapeCastInfo, Vector3 position, float radius, float height, Vector3 direction, float maxDistance, QueryTriggerInteraction triggerInteraction, bool trackInitialOverlaps = true)
{
return CapsuleCast(shapeCastInfo, Data, position, radius, height, 0f, direction, maxDistance, _settings.CollisionLayerMask, triggerInteraction, trackInitialOverlaps);
}
internal void UpdateHits(KCCOverlapInfo baseOverlapInfo, EKCCHitsOverlapQuery overlapQuery)
{
UpdateHits(Data, baseOverlapInfo, overlapQuery);
}
internal bool IsValidHitCollider(Collider hitCollider)
{
if (hitCollider == null)
{
return false;
}
return IsValidHitCollider(Data, hitCollider);
}
private bool SphereOverlap(KCCOverlapInfo overlapInfo, KCCData data, Vector3 position, float radius, float extent, LayerMask layerMask, QueryTriggerInteraction triggerInteraction)
{
overlapInfo.Reset(deep: false);
overlapInfo.Position = position;
overlapInfo.Radius = radius;
overlapInfo.Height = 0f;
overlapInfo.Extent = extent;
overlapInfo.LayerMask = layerMask;
overlapInfo.TriggerInteraction = triggerInteraction;
// Collider[] hitColliders = _hitColliders;
Collider[] hitColliders = Physics.OverlapSphere(position, radius + extent, layerMask, triggerInteraction);
for (int i = 0; i < hitColliders.Length; i++)
{
Collider overlapCollider = hitColliders[i];
if (IsValidHitColliderUnsafe(data, overlapCollider))
{
overlapInfo.AddHit(overlapCollider);
}
}
return overlapInfo.AllHitCount > 0;
}
private bool CapsuleOverlap(KCCOverlapInfo overlapInfo, KCCData data, Vector3 position, float radius, float height, float extent, LayerMask layerMask, QueryTriggerInteraction triggerInteraction)
{
overlapInfo.Reset(deep: false);
overlapInfo.Position = position;
overlapInfo.Radius = radius;
overlapInfo.Height = height;
overlapInfo.Extent = extent;
overlapInfo.LayerMask = layerMask;
overlapInfo.TriggerInteraction = triggerInteraction;
Vector3 point = position + new Vector3(0f, height - radius, 0f);
Vector3 point2 = position + new Vector3(0f, radius, 0f);
// Collider[] hitColliders = _hitColliders;
Collider[] hitColliders = Physics.OverlapCapsule(point2, point, radius + extent, layerMask, triggerInteraction);
for (int i = 0; i < hitColliders.Length; i++)
{
Collider overlapCollider = hitColliders[i];
if (IsValidHitColliderUnsafe(data, overlapCollider))
{
overlapInfo.AddHit(overlapCollider);
}
}
return overlapInfo.AllHitCount > 0;
}
private bool RayCast(KCCShapeCastInfo shapeCastInfo, KCCData data, Vector3 position, Vector3 direction, float maxDistance, LayerMask layerMask, QueryTriggerInteraction triggerInteraction)
{
shapeCastInfo.Reset(deep: false);
shapeCastInfo.Position = position;
shapeCastInfo.Direction = direction;
shapeCastInfo.MaxDistance = maxDistance;
shapeCastInfo.LayerMask = layerMask;
shapeCastInfo.TriggerInteraction = triggerInteraction;
// RaycastHit[] raycastHits = _raycastHits;
// if(Physics.Raycast(position, direction, out RaycastHit hit, maxDistance, layerMask, triggerInteraction))
// // int num = base.Runner.GetPhysicsScene().Raycast(position, direction, raycastHits, maxDistance, layerMask, triggerInteraction);
// for (int i = 0; i < hit.; i++)
// {
// RaycastHit raycastHit = raycastHits[i];
// if (IsValidHitColliderUnsafe(data, raycastHit.collider))
// {
// shapeCastInfo.AddHit(raycastHit);
// }
// }
// return shapeCastInfo.AllHitCount > 0;
return Physics.Raycast(position, direction, maxDistance, layerMask, triggerInteraction);
}
private bool SphereCast(KCCShapeCastInfo shapeCastInfo, KCCData data, Vector3 position, float radius, float extent, Vector3 direction, float maxDistance, LayerMask layerMask, QueryTriggerInteraction triggerInteraction, bool trackInitialOverlaps)
{
shapeCastInfo.Reset(deep: false);
shapeCastInfo.Position = position;
shapeCastInfo.Radius = radius;
shapeCastInfo.Extent = extent;
shapeCastInfo.Direction = direction;
shapeCastInfo.MaxDistance = maxDistance;
shapeCastInfo.LayerMask = layerMask;
shapeCastInfo.TriggerInteraction = triggerInteraction;
RaycastHit[] raycastHits = _raycastHits;
return Physics.SphereCast(position, radius + extent, direction, out RaycastHit info, maxDistance, layerMask, triggerInteraction);
// int num = base.Runner.GetPhysicsScene().SphereCast(position, radius + extent, direction, raycastHits, maxDistance, layerMask, triggerInteraction);
// for (int i = 0; i < num; i++)
// {
// RaycastHit raycastHit = raycastHits[i];
// if ((trackInitialOverlaps || !(raycastHit.distance <= 0f) || !raycastHit.point.Equals(default(Vector3))) && IsValidHitColliderUnsafe(data, raycastHit.collider))
// {
// shapeCastInfo.AddHit(raycastHit);
// }
// }
// return shapeCastInfo.AllHitCount > 0;
}
private bool CapsuleCast(KCCShapeCastInfo shapeCastInfo, KCCData data, Vector3 position, float radius, float height, float extent, Vector3 direction, float maxDistance, LayerMask layerMask, QueryTriggerInteraction triggerInteraction, bool trackInitialOverlaps)
{
shapeCastInfo.Reset(deep: false);
shapeCastInfo.Position = position;
shapeCastInfo.Radius = radius;
shapeCastInfo.Height = height;
shapeCastInfo.Extent = extent;
shapeCastInfo.Position = position;
shapeCastInfo.Direction = direction;
shapeCastInfo.MaxDistance = maxDistance;
shapeCastInfo.LayerMask = layerMask;
shapeCastInfo.TriggerInteraction = triggerInteraction;
Vector3 point = position + new Vector3(0f, height - radius, 0f);
Vector3 point2 = position + new Vector3(0f, radius, 0f);
// RaycastHit[] raycastHits = _raycastHits;
return Physics.CapsuleCast(point2, point, radius + extent, direction, out RaycastHit info, maxDistance, layerMask, triggerInteraction);
// int num = base.Runner.GetPhysicsScene().CapsuleCast(point2, point, radius + extent, direction, raycastHits, maxDistance, layerMask, triggerInteraction);
// for (int i = 0; i < num; i++)
// {
// RaycastHit raycastHit = raycastHits[i];
// if ((trackInitialOverlaps || !(raycastHit.distance <= 0f) || !raycastHit.point.Equals(default(Vector3))) && IsValidHitColliderUnsafe(data, raycastHit.collider))
// {
// shapeCastInfo.AddHit(raycastHit);
// }
// }
// return shapeCastInfo.AllHitCount > 0;
}
private void UpdateHits(KCCData data, KCCOverlapInfo baseOverlapInfo, EKCCHitsOverlapQuery overlapQuery)
{
if (overlapQuery switch
{
EKCCHitsOverlapQuery.Default => baseOverlapInfo?.AllHitsWithinExtent() ?? false,
EKCCHitsOverlapQuery.Reuse => baseOverlapInfo != null,
EKCCHitsOverlapQuery.New => false,
_ => throw new NotImplementedException("overlapQuery"),
})
{
_trackOverlapInfo.CopyFromOther(baseOverlapInfo);
}
else
{
CapsuleOverlap(_trackOverlapInfo, data, data.TargetPosition, _settings.Radius, _settings.Height, _settings.Extent, _settings.CollisionLayerMask, QueryTriggerInteraction.Collide);
if (baseOverlapInfo != null)
{
for (int i = 0; i < _trackOverlapInfo.AllHitCount; i++)
{
KCCOverlapHit kCCOverlapHit = _trackOverlapInfo.AllHits[i];
for (int j = 0; j < baseOverlapInfo.AllHitCount; j++)
{
KCCOverlapHit kCCOverlapHit2 = baseOverlapInfo.AllHits[j];
if ((object)kCCOverlapHit.Collider == kCCOverlapHit2.Collider)
{
kCCOverlapHit.CopyFromOther(kCCOverlapHit2);
}
}
}
}
}
data.Hits.Clear();
int k = 0;
for (int allHitCount = _trackOverlapInfo.AllHitCount; k < allHitCount; k++)
{
data.Hits.Add(_trackOverlapInfo.AllHits[k]);
}
}
private void ForceRemoveAllHits(KCCData data)
{
_trackOverlapInfo.Reset(deep: false);
_extendedOverlapInfo.Reset(deep: false);
data.Hits.Clear();
}
private bool IsValidHitCollider(KCCData data, Collider hitCollider)
{
if (hitCollider == _collider.Collider)
{
return false;
}
int i = 0;
for (int count = _childColliders.Count; i < count; i++)
{
if (hitCollider == _childColliders[i])
{
return false;
}
}
int num = 1 << hitCollider.gameObject.layer;
if (((int)_settings.CollisionLayerMask & num) != num)
{
return false;
}
if (ResolveCollision != null)
{
try
{
return ResolveCollision(this, hitCollider);
}
catch (Exception exception)
{
Debug.LogException(exception);
}
}
return true;
}
private bool IsValidHitColliderUnsafe(KCCData data, Collider overlapCollider)
{
if ((object)overlapCollider == _collider.Collider)
{
return false;
}
int i = 0;
for (int count = _childColliders.Count; i < count; i++)
{
if ((object)overlapCollider == _childColliders[i])
{
return false;
}
}
if (ResolveCollision != null)
{
try
{
return ResolveCollision(this, overlapCollider);
}
catch (Exception exception)
{
Debug.LogException(exception);
}
}
return true;
}
public void SetMaxGroundAngle(float maxGroundAngle)
{
if (IsSpawned)
{
Data.MaxGroundAngle = maxGroundAngle;
}
}
public void SetGravity(float gravity)
{
if (IsSpawned)
{
Data.Gravity = Vector3.up * gravity;
}
}
public void Move(Vector3 kinematicVelocity = default(Vector3), float jumpImpulse = 0f)
{
if (IsSpawned)
{
if (!IsInFixedUpdate)
{
throw new InvalidOperationException("[" + base.name + "] Move can be invoked only from within fixed update!");
}
FixedData.KinematicVelocity = kinematicVelocity;
FixedData.JumpImpulse += Vector3.up * jumpImpulse;
ManualFixedUpdate();
}
}
public bool ProjectOnGround(Vector3 vector, out Vector3 projectedVector)
{
KCCData data = Data;
if (data.IsGrounded)
{
return KCCPhysicsUtility.ProjectOnGround(data.GroundNormal, vector, out projectedVector);
}
projectedVector = default(Vector3);
return false;
}
}
}