mirror of
				https://github.com/AxioDL/metaforce.git
				synced 2025-10-26 04:10:24 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			481 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			481 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "Runtime/World/CScriptMazeNode.hpp"
 | |
| 
 | |
| #include "Runtime/CStateManager.hpp"
 | |
| #include "Runtime/Character/CModelData.hpp"
 | |
| #include "Runtime/GameGlobalObjects.hpp"
 | |
| #include "Runtime/World/CActorParameters.hpp"
 | |
| 
 | |
| #include "TCastTo.hpp" // Generated file, do not modify include path
 | |
| 
 | |
| namespace metaforce {
 | |
| 
 | |
| std::array<s32, 300> sMazeSeeds;
 | |
| 
 | |
| std::array<zeus::CVector3f, skMazeRows * skMazeCols> sDebugCellPos;
 | |
| 
 | |
| CScriptMazeNode::CScriptMazeNode(TUniqueId uid, std::string_view name, const CEntityInfo& info,
 | |
|                                  const zeus::CTransform& xf, bool active, s32 col, s32 row, s32 side,
 | |
|                                  const zeus::CVector3f& actorPos, const zeus::CVector3f& triggerPos,
 | |
|                                  const zeus::CVector3f& effectPos)
 | |
| : CActor(uid, active, name, info, xf, CModelData::CModelDataNull(), CMaterialList(), CActorParameters::None(),
 | |
|          kInvalidUniqueId)
 | |
| , xe8_col(col)
 | |
| , xec_row(row)
 | |
| , xf0_side(static_cast<ESide>(side))
 | |
| , x100_actorPos(actorPos)
 | |
| , x110_triggerPos(triggerPos)
 | |
| , x120_effectPos(effectPos) {}
 | |
| 
 | |
| void CScriptMazeNode::Accept(IVisitor& visitor) { visitor.Visit(this); }
 | |
| 
 | |
| void CScriptMazeNode::AcceptScriptMsg(EScriptObjectMessage msg, TUniqueId uid, CStateManager& mgr) {
 | |
|   if (GetActive()) {
 | |
|     if (msg == EScriptObjectMessage::Action) {
 | |
|       if (auto* maze = mgr.GetCurrentMaze()) {
 | |
|         bool shouldGenObjs = false;
 | |
|         auto& cell = maze->GetCell(xe8_col, xec_row);
 | |
|         if (xf0_side == ESide::Top && cell.x0_24_openTop) {
 | |
|           if (cell.x0_28_gateTop) {
 | |
|             shouldGenObjs = true;
 | |
|             x13c_25_hasGate = true;
 | |
|           }
 | |
|         } else if (xf0_side == ESide::Right && cell.x0_25_openRight) {
 | |
|           if (cell.x0_29_gateRight) {
 | |
|             shouldGenObjs = true;
 | |
|             x13c_25_hasGate = true;
 | |
|           }
 | |
|         } else {
 | |
|           shouldGenObjs = true;
 | |
|         }
 | |
|         if (shouldGenObjs) {
 | |
|           GenerateObjects(mgr);
 | |
|         }
 | |
|         if (xf0_side == ESide::Right && cell.x1_24_puddle) {
 | |
|           x13c_24_hasPuddle = true;
 | |
|         }
 | |
|         if (x13c_25_hasGate) {
 | |
|           const auto origin = GetTranslation();
 | |
|           for (const auto& conn : GetConnectionList()) {
 | |
|             if (conn.x0_state != EScriptObjectState::Modify || conn.x4_msg != EScriptObjectMessage::Activate) {
 | |
|               continue;
 | |
|             }
 | |
| 
 | |
|             bool wasGeneratingObject = mgr.GetIsGeneratingObject();
 | |
|             mgr.SetIsGeneratingObject(true);
 | |
|             const auto genObj = mgr.GenerateObject(conn.x8_objId);
 | |
|             mgr.SetIsGeneratingObject(wasGeneratingObject);
 | |
| 
 | |
|             xf4_gateEffectId = genObj.second;
 | |
|             if (TCastToPtr<CActor> actor = mgr.ObjectById(genObj.second)) {
 | |
|               actor->SetTranslation(origin + x120_effectPos);
 | |
|               mgr.SendScriptMsg(actor, GetUniqueId(), EScriptObjectMessage::Activate);
 | |
|             }
 | |
|             break;
 | |
|           }
 | |
|         }
 | |
|         if (x13c_24_hasPuddle) {
 | |
|           size_t count = 0;
 | |
|           for (const auto& conn : GetConnectionList()) {
 | |
|             if ((conn.x0_state == EScriptObjectState::Closed || conn.x0_state == EScriptObjectState::DeactivateState) &&
 | |
|                 conn.x4_msg == EScriptObjectMessage::Activate) {
 | |
|               count++;
 | |
|             }
 | |
|           }
 | |
|           x12c_puddleObjectIds.reserve(count);
 | |
|           for (const auto& conn : GetConnectionList()) {
 | |
|             if ((conn.x0_state == EScriptObjectState::Closed || conn.x0_state == EScriptObjectState::DeactivateState) &&
 | |
|                 conn.x4_msg == EScriptObjectMessage::Activate) {
 | |
|               bool wasGeneratingObject = mgr.GetIsGeneratingObject();
 | |
|               mgr.SetIsGeneratingObject(true);
 | |
|               const auto genObj = mgr.GenerateObject(conn.x8_objId);
 | |
|               mgr.SetIsGeneratingObject(wasGeneratingObject);
 | |
| 
 | |
|               x12c_puddleObjectIds.push_back(genObj.second);
 | |
|               if (TCastToPtr<CActor> actor = mgr.ObjectById(genObj.second)) {
 | |
|                 actor->SetTransform(GetTransform());
 | |
|                 if (conn.x0_state == EScriptObjectState::Closed) {
 | |
|                   mgr.SendScriptMsg(actor, GetUniqueId(), EScriptObjectMessage::Activate);
 | |
|                 }
 | |
|               }
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     } else if (msg == EScriptObjectMessage::SetToZero) {
 | |
|       auto* maze = mgr.GetCurrentMaze();
 | |
|       if (x13c_24_hasPuddle && maze != nullptr &&
 | |
|           std::any_of(x12c_puddleObjectIds.cbegin(), x12c_puddleObjectIds.cend(), [=](auto v) { return v == uid; })) {
 | |
|         for (const auto& id : x12c_puddleObjectIds) {
 | |
|           if (auto* ent = mgr.ObjectById(id)) {
 | |
|             if (ent->GetActive()) {
 | |
|               mgr.SendScriptMsg(ent, GetUniqueId(), EScriptObjectMessage::Activate);
 | |
|             } else {
 | |
|               mgr.FreeScriptObject(ent->GetUniqueId());
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|         for (const auto ent : mgr.GetAllObjectList()) {
 | |
|           if (TCastToPtr<CScriptMazeNode> node = ent) {
 | |
|             if (node->xe8_col == xe8_col - 1 && node->xec_row == xec_row && node->xf0_side == ESide::Right) {
 | |
|               auto& cell = maze->GetCell(xe8_col - 1, xec_row);
 | |
|               if (!cell.x0_25_openRight) {
 | |
|                 cell.x0_25_openRight = true;
 | |
|                 node->Reset(mgr);
 | |
|                 node->x13c_25_hasGate = false;
 | |
|               }
 | |
|             }
 | |
|             if (node->xe8_col == xe8_col && node->xec_row == xec_row && node->xf0_side == ESide::Right) {
 | |
|               auto& cell = maze->GetCell(xe8_col, xec_row);
 | |
|               if (!cell.x0_25_openRight) {
 | |
|                 cell.x0_25_openRight = true;
 | |
|                 node->Reset(mgr);
 | |
|                 node->x13c_25_hasGate = false;
 | |
|               }
 | |
|             }
 | |
|             if (node->xe8_col == xe8_col && node->xec_row == xec_row && node->xf0_side == ESide::Top) {
 | |
|               auto& cell = maze->GetCell(xe8_col, xec_row);
 | |
|               if (!cell.x0_24_openTop) {
 | |
|                 cell.x0_24_openTop = true;
 | |
|                 node->Reset(mgr);
 | |
|                 node->x13c_25_hasGate = false;
 | |
|               }
 | |
|             }
 | |
|             if (node->xe8_col == xe8_col && node->xec_row == xec_row + 1 && node->xf0_side == ESide::Top) {
 | |
|               auto& cell = maze->GetCell(xe8_col, xec_row + 1);
 | |
|               if (!cell.x0_24_openTop) {
 | |
|                 cell.x0_24_openTop = true;
 | |
|                 node->Reset(mgr);
 | |
|                 node->x13c_25_hasGate = false;
 | |
|               }
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     } else if (msg == EScriptObjectMessage::Deactivate) {
 | |
|       Reset(mgr);
 | |
|     } else if (msg == EScriptObjectMessage::InitializedInArea) {
 | |
|       if (mgr.GetCurrentMaze() == nullptr) {
 | |
|         auto maze = std::make_unique<CMazeState>(skEnterCol, skEnterRow, skTargetCol, skTargetRow);
 | |
|         maze->Reset(sMazeSeeds[mgr.GetActiveRandom()->Next() % sMazeSeeds.size()]);
 | |
|         maze->Initialize();
 | |
|         maze->GenerateObstacles();
 | |
|         mgr.SetCurrentMaze(std::move(maze));
 | |
|       }
 | |
|       if (xf0_side == ESide::Right) {
 | |
|         sDebugCellPos[xe8_col + xec_row * skMazeCols] = GetTranslation();
 | |
|       } else if (xe8_col == skMazeCols - 1) {
 | |
|         // Last column does not have right nodes, but we can infer the position
 | |
|         sDebugCellPos[xe8_col + xec_row * skMazeCols] = GetTranslation() - zeus::CVector3f{1.1875f, -0.1215f, 1.2187f};
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   // URDE change: used to be in the above if branch
 | |
|   if (msg == EScriptObjectMessage::Deleted) {
 | |
|     mgr.ClearCurrentMaze();
 | |
|     Reset(mgr);
 | |
|   }
 | |
|   CActor::AcceptScriptMsg(msg, uid, mgr);
 | |
| }
 | |
| 
 | |
| void CScriptMazeNode::Think(float dt, CStateManager& mgr) {
 | |
|   if (!GetActive() || !x13c_25_hasGate) {
 | |
|     return;
 | |
|   }
 | |
|   xf8_msgTimer -= dt;
 | |
|   if (xf8_msgTimer <= 0.f) {
 | |
|     xf8_msgTimer = 1.f;
 | |
|     if (x13c_26_gateActive) {
 | |
|       x13c_26_gateActive = false;
 | |
|       SendScriptMsgs(mgr, EScriptObjectMessage::Deactivate);
 | |
|     } else {
 | |
|       x13c_26_gateActive = true;
 | |
|       SendScriptMsgs(mgr, EScriptObjectMessage::Activate);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CScriptMazeNode::LoadMazeSeeds() {
 | |
|   const SObjectTag* tag = g_ResFactory->GetResourceIdByName("DUMB_MazeSeeds");
 | |
|   const u32 resSize = g_ResFactory->ResourceSize(*tag);
 | |
|   const std::unique_ptr<u8[]> buf = g_ResFactory->LoadResourceSync(*tag);
 | |
|   CMemoryInStream in(buf.get(), resSize);
 | |
|   for (auto& seed : sMazeSeeds) {
 | |
|     seed = in.readInt32Big();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CScriptMazeNode::GenerateObjects(CStateManager& mgr) {
 | |
|   for (const auto& conn : GetConnectionList()) {
 | |
|     if (conn.x0_state != EScriptObjectState::MaxReached || conn.x4_msg != EScriptObjectMessage::Activate) {
 | |
|       continue;
 | |
|     }
 | |
|     const auto* ent = mgr.GetObjectById(mgr.GetIdForScript(conn.x8_objId));
 | |
|     TCastToConstPtr<CScriptEffect> scriptEffect{ent};
 | |
|     TCastToConstPtr<CScriptActor> scriptActor{ent};
 | |
|     TCastToConstPtr<CScriptTrigger> scriptTrigger{ent};
 | |
|     if ((scriptEffect || scriptActor || scriptTrigger) && (!scriptEffect || !x13c_25_hasGate)) {
 | |
|       bool wasGeneratingObject = mgr.GetIsGeneratingObject();
 | |
|       mgr.SetIsGeneratingObject(true);
 | |
|       const auto genObj = mgr.GenerateObject(conn.x8_objId);
 | |
|       mgr.SetIsGeneratingObject(wasGeneratingObject);
 | |
|       if (auto* actor = static_cast<CActor*>(mgr.ObjectById(genObj.second))) {
 | |
|         mgr.SendScriptMsg(actor, GetUniqueId(), EScriptObjectMessage::Activate);
 | |
|         if (scriptEffect) {
 | |
|           actor->SetTranslation(GetTranslation() + x120_effectPos);
 | |
|           x11c_effectId = genObj.second;
 | |
|         }
 | |
|         if (scriptActor) {
 | |
|           actor->SetTranslation(GetTranslation() + x100_actorPos);
 | |
|           xfc_actorId = genObj.second;
 | |
|         }
 | |
|         if (scriptTrigger) {
 | |
|           actor->SetTranslation(GetTranslation() + x110_triggerPos);
 | |
|           x10c_triggerId = genObj.second;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CScriptMazeNode::Reset(CStateManager& mgr) {
 | |
|   mgr.FreeScriptObject(x11c_effectId);
 | |
|   mgr.FreeScriptObject(xfc_actorId);
 | |
|   mgr.FreeScriptObject(x10c_triggerId);
 | |
|   mgr.FreeScriptObject(xf4_gateEffectId);
 | |
|   xf4_gateEffectId = kInvalidUniqueId;
 | |
|   xfc_actorId = kInvalidUniqueId;
 | |
|   x10c_triggerId = kInvalidUniqueId;
 | |
|   x11c_effectId = kInvalidUniqueId;
 | |
| }
 | |
| 
 | |
| void CScriptMazeNode::SendScriptMsgs(CStateManager& mgr, EScriptObjectMessage msg) {
 | |
|   mgr.SendScriptMsg(x11c_effectId, GetUniqueId(), msg);
 | |
|   mgr.SendScriptMsg(xfc_actorId, GetUniqueId(), msg);
 | |
|   mgr.SendScriptMsg(x10c_triggerId, GetUniqueId(), msg);
 | |
|   mgr.SendScriptMsg(xf4_gateEffectId, GetUniqueId(), msg);
 | |
| }
 | |
| 
 | |
| void CMazeState::Reset(s32 seed) {
 | |
|   x0_rand.SetSeed(seed);
 | |
|   x94_24_initialized = false;
 | |
|   x4_cells.fill({});
 | |
| 
 | |
|   std::array<ESide, 4> sides{};
 | |
|   s32 cellIdx = 0;
 | |
|   for (u32 i = skMazeCols * skMazeRows - 1; i != 0;) {
 | |
|     u32 acc = 0;
 | |
|     if (cellIdx - skMazeCols > 0 && !GetCell(cellIdx - skMazeCols).IsOpen()) {
 | |
|       sides[acc++] = ESide::Top;
 | |
|     }
 | |
|     if (cellIdx < x4_cells.size() - 2 && (cellIdx + 1) % skMazeCols != 0 && !GetCell(cellIdx + 1).IsOpen()) {
 | |
|       sides[acc++] = ESide::Right;
 | |
|     }
 | |
|     if (cellIdx + skMazeCols < x4_cells.size() && !GetCell(cellIdx + skMazeCols).IsOpen()) {
 | |
|       sides[acc++] = ESide::Bottom;
 | |
|     }
 | |
|     if (cellIdx > 0 && cellIdx % skMazeCols != 0 && !GetCell(cellIdx - 1).IsOpen()) {
 | |
|       sides[acc++] = ESide::Left;
 | |
|     }
 | |
| 
 | |
|     if (acc == 0) {
 | |
|       do {
 | |
|         cellIdx++;
 | |
|         if (cellIdx > x4_cells.size() - 1) {
 | |
|           cellIdx = 0;
 | |
|         }
 | |
|       } while (!GetCell(cellIdx).IsOpen());
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     i--;
 | |
|     ESide side = sides[x0_rand.Next() % acc];
 | |
|     if (side == ESide::Bottom) {
 | |
|       GetCell(cellIdx).x0_26_openBottom = true;
 | |
|       GetCell(cellIdx + skMazeCols).x0_24_openTop = true;
 | |
|       cellIdx += skMazeCols;
 | |
|     } else if (side == ESide::Top) {
 | |
|       GetCell(cellIdx).x0_24_openTop = true;
 | |
|       GetCell(cellIdx - skMazeCols).x0_26_openBottom = true;
 | |
|       cellIdx -= skMazeCols;
 | |
|     } else if (side == ESide::Right) {
 | |
|       GetCell(cellIdx).x0_25_openRight = true;
 | |
|       GetCell(cellIdx + 1).x0_27_openLeft = true;
 | |
|       cellIdx++;
 | |
|     } else if (side == ESide::Left) {
 | |
|       GetCell(cellIdx).x0_27_openLeft = true;
 | |
|       GetCell(cellIdx - 1).x0_25_openRight = true;
 | |
|       cellIdx--;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CMazeState::Initialize() {
 | |
|   std::array<s32, skMazeRows * skMazeCols> path{};
 | |
|   path[0] = x84_enterCol + x88_enterRow * skMazeCols;
 | |
|   GetCell(path[0]).x1_26_checked = true;
 | |
|   s32 pathLength = 1;
 | |
|   while (path[0] != x8c_targetCol + x90_targetRow * skMazeCols) {
 | |
|     if (GetCell(path[0]).x0_24_openTop) {
 | |
|       if (!GetCell(path[0] - skMazeCols).x1_26_checked) {
 | |
|         path[pathLength] = path[0] - skMazeCols;
 | |
|         pathLength++;
 | |
|       }
 | |
|     }
 | |
|     if (GetCell(path[0]).x0_25_openRight) {
 | |
|       if (!GetCell(path[0] + 1).x1_26_checked) {
 | |
|         path[pathLength] = path[0] + 1;
 | |
|         pathLength++;
 | |
|       }
 | |
|     }
 | |
|     if (GetCell(path[0]).x0_26_openBottom) {
 | |
|       if (!GetCell(path[0] + skMazeCols).x1_26_checked) {
 | |
|         path[pathLength] = path[0] + skMazeCols;
 | |
|         pathLength++;
 | |
|       }
 | |
|     }
 | |
|     if (GetCell(path[0]).x0_27_openLeft) {
 | |
|       if (!GetCell(path[0] - 1).x1_26_checked) {
 | |
|         path[pathLength] = path[0] - 1;
 | |
|         pathLength++;
 | |
|       }
 | |
|     }
 | |
|     if (path[0] == path[pathLength - 1]) {
 | |
|       pathLength--;
 | |
|     }
 | |
|     path[0] = path[pathLength - 1];
 | |
|     GetCell(path[0]).x1_26_checked = true;
 | |
|   }
 | |
|   s32* idx = &path[pathLength];
 | |
|   while (pathLength != 0) {
 | |
|     pathLength--;
 | |
|     idx--;
 | |
|     auto& cell = GetCell(*idx);
 | |
|     if (cell.x1_26_checked) {
 | |
|       cell.x1_25_onPath = true;
 | |
|       if (pathLength > 0) {
 | |
|         m_path.push_back(*idx);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   x94_24_initialized = true;
 | |
| }
 | |
| 
 | |
| void CMazeState::GenerateObstacles() {
 | |
|   if (!x94_24_initialized) {
 | |
|     Initialize();
 | |
|   }
 | |
| 
 | |
|   auto GetRandom = [this](s32 offset) {
 | |
|     s32 tmp = x0_rand.Next();
 | |
|     return tmp + ((tmp / 5) * -5) + offset;
 | |
|   };
 | |
|   s32 gate1Idx = GetRandom(9);
 | |
|   s32 gate2Idx = GetRandom(21);
 | |
|   s32 gate3Idx = GetRandom(33);
 | |
|   s32 puddle1Idx = GetRandom(13);
 | |
|   s32 puddle2Idx = GetRandom(29);
 | |
| 
 | |
|   ESide side = ESide::Invalid;
 | |
|   s32 idx = 0;
 | |
| 
 | |
|   s32 prevCol = x84_enterCol;
 | |
|   s32 prevRow = x88_enterRow;
 | |
|   s32 col = x84_enterCol;
 | |
|   s32 row = x88_enterRow;
 | |
| 
 | |
|   while (col != x8c_targetCol || row != x90_targetRow) {
 | |
|     if (idx == gate1Idx || idx == gate2Idx || idx == gate3Idx) {
 | |
|       if (side == ESide::Bottom) {
 | |
|         GetCell(col, row).x0_28_gateTop = true;
 | |
|         GetCell(prevCol, prevRow).x0_30_gateBottom = true;
 | |
|       } else if (side == ESide::Top) {
 | |
|         GetCell(col, row).x0_30_gateBottom = true;
 | |
|         GetCell(prevCol, prevRow).x0_28_gateTop = true;
 | |
|       } else if (side == ESide::Right) {
 | |
|         GetCell(col, row).x0_31_gateLeft = true;
 | |
|         GetCell(prevCol, prevRow).x0_29_gateRight = true;
 | |
|       } else if (side == ESide::Left) {
 | |
|         GetCell(col, row).x0_29_gateRight = true;
 | |
|         GetCell(prevCol, prevRow).x0_31_gateLeft = true;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     s32 nextCol = col;
 | |
|     s32 nextRow = -1;
 | |
|     if (row < 1 || side == ESide::Bottom || !GetCell(col, row).x0_24_openTop || !GetCell(col, row - 1).x1_25_onPath) {
 | |
|       if (row < skMazeRows - 1 && side != ESide::Top && GetCell(col, row).x0_26_openBottom &&
 | |
|           GetCell(col, row + 1).x1_25_onPath) {
 | |
|         side = ESide::Bottom;
 | |
|         nextRow = row + 1;
 | |
|       } else {
 | |
|         nextRow = row;
 | |
|         if (col < 1 || side == ESide::Right || !GetCell(col, row).x0_27_openLeft ||
 | |
|             !GetCell(col - 1, row).x1_25_onPath) {
 | |
|           if (col > skMazeRows || side == ESide::Left || !GetCell(col, row).x0_25_openRight ||
 | |
|               !GetCell(col + 1, row).x1_25_onPath) {
 | |
|             return;
 | |
|           }
 | |
|           side = ESide::Right;
 | |
|           nextCol = col + 1;
 | |
|         } else {
 | |
|           side = ESide::Left;
 | |
|           nextCol = col - 1;
 | |
|         }
 | |
|       }
 | |
|     } else {
 | |
|       side = ESide::Top;
 | |
|       nextRow = row - 1;
 | |
|     }
 | |
| 
 | |
|     if (idx == puddle1Idx || idx == puddle2Idx) {
 | |
|       if (col == 0 || row == 0 || col == skMazeCols - 1 || row == skMazeRows - 1) {
 | |
|         if (idx == puddle1Idx) {
 | |
|           puddle1Idx++;
 | |
|         } else {
 | |
|           puddle2Idx++;
 | |
|         }
 | |
|       } else {
 | |
|         auto& cell = GetCell(col, row);
 | |
|         cell.x1_24_puddle = true;
 | |
|         if (side == ESide::Bottom) {
 | |
|           GetCell(nextCol, nextRow).x0_24_openTop = false;
 | |
|           cell.x0_26_openBottom = false;
 | |
|         } else if (side == ESide::Top) {
 | |
|           GetCell(nextCol, nextRow).x0_26_openBottom = false;
 | |
|           cell.x0_24_openTop = false;
 | |
|         } else if (side == ESide::Right) {
 | |
|           GetCell(nextCol, nextRow).x0_27_openLeft = false;
 | |
|           cell.x0_25_openRight = false;
 | |
|         } else if (side == ESide::Left) {
 | |
|           GetCell(nextCol, nextRow).x0_25_openRight = false;
 | |
|           cell.x0_27_openLeft = false;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     idx++;
 | |
|     prevCol = col;
 | |
|     prevRow = row;
 | |
|     col = nextCol;
 | |
|     row = nextRow;
 | |
|   };
 | |
| }
 | |
| 
 | |
| void CMazeState::DebugRender() {
 | |
|   m_renderer.Reset();
 | |
|   m_renderer.AddVertex(sDebugCellPos[skEnterCol + skEnterRow * skMazeCols], zeus::skBlue, 2.f);
 | |
|   for (s32 i = m_path.size() - 1; i >= 0; --i) {
 | |
|     s32 idx = m_path[i];
 | |
|     zeus::CVector3f pos;
 | |
|     if (idx == skMazeCols - 1) {
 | |
|       // 8,0 has no node, infer from 8,1
 | |
|       pos = sDebugCellPos[idx + skMazeCols] + zeus::CVector3f{4.f, 0.f, 0.f};
 | |
|     } else {
 | |
|       pos = sDebugCellPos[idx];
 | |
|     }
 | |
|     m_renderer.AddVertex(pos, zeus::skBlue, 2.f);
 | |
|   }
 | |
|   m_renderer.Render();
 | |
| }
 | |
| } // namespace metaforce
 |