A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
ObjectDirector.cc
1#include "ObjectDirector.hh"
2
3#include "game/field/BoxColManager.hh"
4#include "game/field/ObjectDrivableDirector.hh"
5#include "game/field/obj/ObjectRegistry.hh"
6
7#include "game/kart/KartObject.hh"
8
9#include "game/system/CourseMap.hh"
10
11namespace Field {
12
14void ObjectDirector::init() {
15 for (auto *&obj : m_objects) {
16 obj->init();
17 obj->calcModel();
18 }
19
20 ObjectDrivableDirector::Instance()->init();
21}
22
24void ObjectDirector::calc() {
25 for (auto *&obj : m_calcObjects) {
26 obj->calc();
27 }
28
29 for (auto *&obj : m_calcObjects) {
30 obj->calcModel();
31 }
32
33 ObjectDrivableDirector::Instance()->calc();
34}
35
37void ObjectDirector::addObject(ObjectCollidable *obj) {
38 u32 loadFlags = obj->loadFlags();
39
40 if (loadFlags & 1) {
41 m_calcObjects.push_back(obj);
42 }
43
44 if (m_flowTable.set(m_flowTable.slot(obj->id()))->mode != 0) {
45 if (obj->collision()) {
46 m_collisionObjects.push_back(obj);
47 }
48 }
49
50 m_objects.push_back(obj);
51}
52
53void ObjectDirector::addObjectNoImpl(ObjectNoImpl *obj) {
54 m_objects.push_back(obj);
55}
56
58size_t ObjectDirector::checkKartObjectCollision(Kart::KartObject *kartObj,
59 ObjectCollisionConvexHull *convexHull) {
60 size_t count = 0;
61
62 while (ObjectCollidable *obj = BoxColManager::Instance()->getNextObject()) {
63 auto *objCollision = obj->collision();
64 if (!objCollision) {
65 continue;
66 }
67
68 obj->calcCollisionTransform();
69 if (!obj->checkCollision(convexHull, m_hitDepths[count])) {
70 continue;
71 }
72
73 // We have a collision, process it
74 // Assume that we are not in a star, mega, or bullet
75 Kart::Reaction reactionOnKart = m_hitTableKart.reaction(m_hitTableKart.slot(obj->id()));
76 Kart::Reaction reactionOnObj =
77 m_hitTableKartObject.reaction(m_hitTableKartObject.slot(obj->id()));
78
79 // The object might change the reaction states
80 obj->processKartReactions(kartObj, reactionOnKart, reactionOnObj);
81
82 Kart::Reaction reaction =
83 obj->onCollision(kartObj, reactionOnKart, reactionOnObj, m_hitDepths[count]);
84 m_reactions[count] = reaction;
85
86 if (reaction == Kart::Reaction::WallAllSpeed || reaction == Kart::Reaction::WallSpark) {
87 obj->onWallCollision(kartObj, m_hitDepths[count]);
88 } else {
89 obj->onObjectCollision(kartObj);
90 }
91
92 m_collidingObjects[count] = obj;
93 if (m_hitDepths[count].y < 0.0f) {
94 m_hitDepths[count].y = 0.0f;
95 }
96
97 ++count;
98 }
99
100 return count;
101}
102
104ObjectDirector *ObjectDirector::CreateInstance() {
105 ASSERT(!s_instance);
106 s_instance = new ObjectDirector;
107
108 ObjectDrivableDirector::CreateInstance();
109
110 s_instance->createObjects();
111
112 return s_instance;
113}
114
116void ObjectDirector::DestroyInstance() {
117 ASSERT(s_instance);
118 auto *instance = s_instance;
119 s_instance = nullptr;
120 delete instance;
121
122 ObjectDrivableDirector::DestroyInstance();
123}
124
126ObjectDirector::ObjectDirector()
127 : m_flowTable("ObjFlow.bin"), m_hitTableKart("GeoHitTableKart.bin"),
128 m_hitTableKartObject("GeoHitTableKartObj.bin") {}
129
131ObjectDirector::~ObjectDirector() {
132 if (s_instance) {
133 s_instance = nullptr;
134 WARN("ObjectDirector instance not explicitly handled!");
135 }
136
137 for (auto *&obj : m_objects) {
138 delete obj;
139 }
140}
141
143void ObjectDirector::createObjects() {
144 const auto *courseMap = System::CourseMap::Instance();
145 size_t objectCount = courseMap->getGeoObjCount();
146
147 // It's possible for the KMP to specify settings for objects that aren't tracked here
148 // MAX_UNIT_COUNT is the upper bound for tracked object count, so we reserve the minimum
149 size_t maxCount = std::min(objectCount, MAX_UNIT_COUNT);
150 m_objects.reserve(maxCount);
151 m_calcObjects.reserve(maxCount);
152 m_collisionObjects.reserve(maxCount);
153
154 for (size_t i = 0; i < objectCount; ++i) {
155 const auto *pObj = courseMap->getGeoObj(i);
156 ASSERT(pObj);
157
158 // Assume one player - if the presence flag isn't set, don't construct it
159 if (!(pObj->presenceFlag() & 1)) {
160 continue;
161 }
162
163 // Prevent construction of objects with disabled or no collision
164 if (IsObjectBlacklisted(pObj->id())) {
165 continue;
166 }
167
168 ObjectBase *object = createObject(*pObj);
169 object->load();
170 }
171}
172
174ObjectBase *ObjectDirector::createObject(const System::MapdataGeoObj &params) {
175 ObjectId id = static_cast<ObjectId>(params.id());
176 switch (id) {
177 case ObjectId::DokanSFC:
178 return new ObjectDokan(params);
179 case ObjectId::OilSFC:
180 return new ObjectOilSFC(params);
181 case ObjectId::ParasolR:
182 return new ObjectParasolR(params);
183 case ObjectId::PuchiPakkun:
184 return new ObjectPuchiPakkun(params);
185 case ObjectId::Aurora:
186 return new ObjectAurora(params);
187 // Non-specified objects are stock collidable objects by default
188 // However, we need to specify an impl, so we don't use default
189 case ObjectId::DummyPole:
190 case ObjectId::CastleTree1c:
191 case ObjectId::PalmTree:
192 case ObjectId::DKtreeA64c:
193 case ObjectId::TownTreeDsc:
194 return new ObjectCollidable(params);
195 case ObjectId::Mdush:
196 return new ObjectKCL(params);
197 default:
198 return new ObjectNoImpl(params);
199 }
200}
201
202ObjectDirector *ObjectDirector::s_instance = nullptr;
203
204} // namespace Field
std::vector< ObjectBase * > m_objects
All objects live here.
std::vector< ObjectBase * > m_collisionObjects
Objects having collision live here too.
std::vector< ObjectBase * > m_calcObjects
Objects needing calc() live here too.
std::array< ObjectBase *, MAX_UNIT_COUNT > m_collidingObjects
Objects we are currently colliding with.
The highest level abstraction for a kart.
Definition KartObject.hh:11
Pertains to collision.
ObjectDokan ObjectPuchiPakkun
This is just to help with readability. The rMR piranhas are really just pipes.