1#include "KartReject.hh"
3#include "game/kart/KartDynamics.hh"
4#include "game/kart/KartMove.hh"
5#include "game/kart/KartParam.hh"
6#include "game/kart/KartState.hh"
8#include "game/field/CollisionDirector.hh"
9#include "game/field/CourseColMgr.hh"
11#include <egg/math/Math.hh>
12#include <egg/math/Quat.hh>
17KartReject::KartReject() =
default;
20KartReject::~KartReject() =
default;
23void KartReject::reset() {
28void KartReject::calcRejectRoad() {
29 if (state()->isInAction()) {
33 if (state()->isRejectRoadTrigger()) {
36 f32 cos = down.
dot(move()->lastDir());
37 f32 sin = down.cross(move()->lastDir()).
length();
38 f32 angle = EGG::Mathf::atan2(sin, cos);
39 angle = angle > 0.0f ? angle : -angle;
45 if (move()->up().dot(EGG::Vector3f::ey) < -0.7f) {
50 if (angle > minAngle) {
51 angle = 0.05f * (angle - 60.0f);
54 (1.0f + angle * move()->speedRatio()) * dVar11 * DEG2RAD * m_rejectSign, 0.0f);
59 dynamics()->setAngVel0(EGG::Vector3f::zero);
60 dynamics()->setFullRot(local_78);
61 dynamics()->setMainRot(local_78);
64 state()->setHop(
false);
66 bool didReject = calcRejection();
68 if (!state()->isNoSparkInvisibleWall() && !didReject) {
69 state()->setRejectRoadTrigger(
false);
75 if (state()->isRejectRoad() && !state()->isZipperInvisibleWall() && !state()->isOverZipper() &&
76 !state()->isHalfPipeRamp()) {
80 if (upXZ.
length() > 0.0f && speed() > 0.0f) {
84 if (local_88.y > 0.0f) {
86 m_rejectSign = upCross.
dot(move()->up()) > 0.0f ? 1.0f : -1.0f;
88 state()->setHop(
false);
89 state()->setRejectRoadTrigger(
true);
96bool KartReject::calcRejection() {
98 Field::KCLTypeMask mask = KCL_NONE;
99 state()->setNoSparkInvisibleWall(
false);
101 f32 posScalar = 100.0f;
102 f32 radius = posScalar;
104 for (
size_t i = 0; i < 2; ++i) {
106 EGG::Vector3f worldPos = pos() + (-posScalar * move()->scale().y) * local_d0;
108 auto *colDir = Field::CollisionDirector::Instance();
109 if (!colDir->checkSphereFullPush(radius, worldPos, worldUpPos,
KCL_TYPE_B0E82DFF, &colInfo,
118 bool hasFloorCollision =
false;
119 bool hasRejectCollision =
false;
120 bool hasInvisibleWallCollision =
false;
124 hasInvisibleWallCollision =
128 const auto *closestColEntry = colDir->closestCollisionEntry();
129 if (hasInvisibleWallCollision &&
KCL_VARIANT_TYPE(closestColEntry->attribute) == 0) {
130 hasRejectCollision =
true;
131 tangentOff = colInfo.wallNrm;
132 state()->setNoSparkInvisibleWall(
true);
135 hasFloorCollision =
false;
140 closestColEntry = colDir->closestCollisionEntry();
141 if (hasFloorCollision && closestColEntry->attribute & 0x4000) {
142 hasRejectCollision =
true;
143 tangentOff = colInfo.floorNrm;
147 if (!hasRejectCollision) {
151 EGG::Vector3f tangentUp = (tangentOff - move()->up()) * 1.0f;
152 move()->setUp(move()->up() + tangentUp);
153 move()->setSmoothedUp(move()->up());
155 bool bVar15 = tangentOff.
dot(EGG::Vector3f::ey) < -0.17f;
156 if (bVar15 || extVel().y < 0.0f || state()->isNoSparkInvisibleWall()) {
158 colInfo.tangentOff += worldPos;
160 f32 yOffset = bsp().initialYPos * scale().y;
161 f32 speedScalar = bVar15 ?
163 static_cast<f32
>(
static_cast<f64
>(EGG::Mathf::abs(speed()) * 0.01f) - 0.3d);
164 speedScalar = std::min(1.0f, std::max(0.0f, speedScalar));
167 colInfo.tangentOff + radius * tangentOff + yOffset * tangentOff;
168 posOffset.y += move()->hopPosY();
170 setPos(pos() + posOffset * speedScalar);
174 move()->setDir(local_13c);
175 move()->setVel1Dir(local_13c);
#define KCL_VARIANT_TYPE(x)
Extracts the "Variant" portion of the 2 byte KCL flag. It's the 3 bits before the "Bast Type".
#define KCL_TYPE_DRIVER_FLOOR
0x20E80DFF - Any KCL that the player can drive on.
#define KCL_TYPE_INVISIBLE_WALL
0x80002000
#define KCL_TYPE_B0E82DFF
0xB0E82DFF
EGG::Vector3f bodyUp() const
Returns the second column of the rotation matrix, which is the "up" direction.
Pertains to kart-related functionality.
A quaternion, used to represent 3D rotation.
void normalise()
Scales the quaternion to a unit length.
Vector3f rotateVector(const Vector3f &vec) const
Rotates a vector based on the quat.
void setRPY(const Vector3f &rpy)
Sets roll, pitch, and yaw.
f32 normalise()
Normalizes the vector and returns the original length.
f32 dot(const Vector3f &rhs) const
The dot product between two vectors.
f32 length() const
The square root of the vector's dot product.
Vector3f perpInPlane(const EGG::Vector3f &rhs, bool normalise) const
Calculates the orthogonal vector, based on the plane defined by this vector and rhs.