#include "Runtime/Camera/CCameraManager.hpp"

#include <algorithm>

#include "Runtime/CStateManager.hpp"
#include "Runtime/GameGlobalObjects.hpp"
#include "Runtime/Camera/CBallCamera.hpp"
#include "Runtime/Camera/CCameraShakeData.hpp"
#include "Runtime/Camera/CCinematicCamera.hpp"
#include "Runtime/Camera/CFirstPersonCamera.hpp"
#include "Runtime/Camera/CInterpolationCamera.hpp"
#include "Runtime/Camera/CPathCamera.hpp"
#include "Runtime/World/CExplosion.hpp"
#include "Runtime/World/CPlayer.hpp"
#include "Runtime/World/CScriptCameraHint.hpp"
#include "Runtime/World/CScriptSpindleCamera.hpp"
#include "Runtime/World/CScriptWater.hpp"

#include "TCastTo.hpp" // Generated file, do not modify include path

namespace metaforce {
float CCameraManager::sFirstPersonFOV = 55.f;

CCameraManager::CCameraManager(TUniqueId curCameraId) : x0_curCameraId(curCameraId) {
  CSfxManager::AddListener(CSfxManager::ESfxChannels::Game, zeus::skZero3f, zeus::skZero3f, {1.f, 0.f, 0.f},
                           {0.f, 0.f, 1.f}, 50.f, 50.f, 1000.f, 1, 1.f);
  sFirstPersonFOV = g_tweakGame->GetFirstPersonFOV();
}

bool CCameraManager::IsInFirstPersonCamera() const { return x7c_fpCamera->GetUniqueId() == x0_curCameraId; }

zeus::CVector3f CCameraManager::GetGlobalCameraTranslation(const CStateManager& stateMgr) const {
  const CGameCamera* camera = GetCurrentCamera(stateMgr);
  return camera->GetTransform().rotate(x30_shakeOffset);
}

zeus::CTransform CCameraManager::GetCurrentCameraTransform(const CStateManager& stateMgr) const {
  const CGameCamera* camera = GetCurrentCamera(stateMgr);
  return camera->GetTransform() * zeus::CTransform::Translate(x30_shakeOffset);
}

void CCameraManager::RemoveCameraShaker(u32 id) {
  const auto iter = std::find_if(x14_shakers.cbegin(), x14_shakers.cend(),
                                 [id](const auto& shaker) { return shaker.xbc_shakerId == id; });
  if (iter == x14_shakers.cend()) {
    return;
  }
  x14_shakers.erase(iter);
}

int CCameraManager::AddCameraShaker(const CCameraShakeData& data, bool sfx) {
  x14_shakers.emplace_back(data).xbc_shakerId = ++x2c_lastShakeId;
  if (!xa0_24_pendingRumble) {
    xa0_24_pendingRumble = true;
    x90_rumbleCooldown = 0.5f;
  }
  if (sfx && data.x0_duration > 0.f) {
    float vol = zeus::clamp(100.f, std::max(data.GetMaxAMComponent(), data.GetMaxFMComponent()) * 9.f + 100.f, 127.f);
    CSfxHandle sfxHandle;
    if (data.xc0_flags & 0x1)
      sfxHandle = CSfxManager::AddEmitter(SFXamb_x_rumble_lp_00, data.xc4_sfxPos, zeus::skZero3f, vol / 127.f, false,
                                          false, 0x7f, kInvalidAreaId);
    else
      sfxHandle = CSfxManager::SfxStart(SFXamb_x_rumble_lp_00, vol / 127.f, 0.f, false, 0x7f, false, kInvalidAreaId);
    sfxHandle->SetTimeRemaining(data.x0_duration);
  }
  return x2c_lastShakeId;
}

void CCameraManager::EnterCinematic(CStateManager& mgr) {
  mgr.GetPlayer().GetPlayerGun()->CancelFiring(mgr);
  mgr.GetPlayer().UnFreeze(mgr);

  for (const CEntity* ent : mgr.GetAllObjectList()) {
    if (const TCastToConstPtr<CExplosion> explo = ent) {
      mgr.FreeScriptObject(explo->GetUniqueId());
    } else if (const TCastToConstPtr<CWeapon> weap = ent) {
      if (weap->GetActive()) {
        if (False(weap->GetAttribField() & EProjectileAttrib::KeepInCinematic)) {
          if (TCastToConstPtr<CAi>(mgr.GetObjectById(weap->GetOwnerId())) ||
              TCastToConstPtr<CPlayer>(mgr.GetObjectById(weap->GetOwnerId())))
            mgr.FreeScriptObject(weap->GetUniqueId());
        }
      }
    }
  }
}

void CCameraManager::AddCinemaCamera(TUniqueId id, CStateManager& stateMgr) {
  if (x4_cineCameras.empty()) {
    EnterCinematic(stateMgr);
  }

  RemoveCinemaCamera(id, stateMgr);
  x4_cineCameras.push_back(id);

  if (const TCastToPtr<CCinematicCamera> cam = stateMgr.ObjectById(id)) {
    // Into player eye
    if ((cam->GetFlags() & 0x4) != 0) {
      float time = 4.f;
      float delayTime = cam->GetDuration() - 4.f;
      if (delayTime < 0.f) {
        delayTime = 0.f;
        time = cam->GetDuration();
      }
      cam->SetFovInterpolation(cam->GetFov(), 55.f, time, delayTime);
    }
  }
}

void CCameraManager::SetInsideFluid(bool isInside, TUniqueId fluidId) {
  if (isInside) {
    ++x74_fluidCounter;
    x78_fluidId = fluidId;
  } else {
    --x74_fluidCounter;
  }
}

void CCameraManager::Update(float dt, CStateManager& stateMgr) {
  UpdateCameraHints(dt, stateMgr);
  ThinkCameras(dt, stateMgr);
  UpdateListener(stateMgr);
  UpdateRumble(dt, stateMgr);
  UpdateFog(dt, stateMgr);
}

CGameCamera* CCameraManager::GetCurrentCamera(CStateManager& stateMgr) const {
  CObjectList* camList = stateMgr.ObjectListById(EGameObjectList::GameCamera);
  return static_cast<CGameCamera*>(camList->GetObjectById(GetCurrentCameraId()));
}

const CGameCamera* CCameraManager::GetCurrentCamera(const CStateManager& stateMgr) const {
  const CObjectList* camList = stateMgr.GetObjectListById(EGameObjectList::GameCamera);
  return static_cast<const CGameCamera*>(camList->GetObjectById(GetCurrentCameraId()));
}

void CCameraManager::CreateStandardCameras(CStateManager& stateMgr) {
  TUniqueId fpId = stateMgr.AllocateUniqueId();
  x7c_fpCamera =
      new CFirstPersonCamera(fpId, zeus::CTransform(), stateMgr.Player()->GetUniqueId(),
                             g_tweakPlayer->GetOrbitCameraSpeed(), sFirstPersonFOV, NearPlane(), FarPlane(), Aspect());
  stateMgr.AddObject(x7c_fpCamera);
  stateMgr.Player()->SetCameraState(CPlayer::EPlayerCameraState::FirstPerson, stateMgr);
  SetCurrentCameraId(fpId, stateMgr);

  x80_ballCamera = new CBallCamera(stateMgr.AllocateUniqueId(), stateMgr.Player()->GetUniqueId(), zeus::CTransform(),
                                   ThirdPersonFOV(), NearPlane(), FarPlane(), Aspect());
  stateMgr.AddObject(x80_ballCamera);

  x88_interpCamera = new CInterpolationCamera(stateMgr.AllocateUniqueId(), zeus::CTransform());
  stateMgr.AddObject(x88_interpCamera);
}

void CCameraManager::SkipCinematic(CStateManager& stateMgr) {
  const TUniqueId camId = GetCurrentCameraId();
  auto* ent = static_cast<CCinematicCamera*>(stateMgr.ObjectById(camId));
  while (ent) {
    ent->SetActive(false);
    ent->WasDeactivated(stateMgr);
    ent = TCastToPtr<CCinematicCamera>(GetCurrentCamera(stateMgr)).GetPtr();
  }
  stateMgr.GetPlayer().UpdateCinematicState(stateMgr);
  x7c_fpCamera->SkipCinematic();
}

void CCameraManager::SetPathCamera(TUniqueId id, CStateManager& mgr) {
  xa4_pathCamId = id;
  if (const TCastToPtr<CPathCamera> cam = mgr.ObjectById(id)) {
    cam->Reset(GetCurrentCameraTransform(mgr), mgr);
    x80_ballCamera->TeleportCamera(cam->GetTransform(), mgr);
  }
}

void CCameraManager::SetSpindleCamera(TUniqueId id, CStateManager& mgr) {
  xa2_spindleCamId = id;
  if (const TCastToPtr<CScriptSpindleCamera> cam = mgr.ObjectById(id)) {
    cam->Reset(GetCurrentCameraTransform(mgr), mgr);
    x80_ballCamera->TeleportCamera(cam->GetTransform(), mgr);
  }
}

void CCameraManager::InterpolateToBallCamera(const zeus::CTransform& xf, TUniqueId camId,
                                             const zeus::CVector3f& lookPos, float maxTime, float positionSpeed,
                                             float rotationSpeed, bool sinusoidal, CStateManager& mgr) {
  if (!IsInFirstPersonCamera()) {
    x88_interpCamera->SetInterpolation(xf, lookPos, maxTime, positionSpeed, rotationSpeed, camId, sinusoidal, mgr);
    if (!ShouldBypassInterpolation())
      SetCurrentCameraId(x88_interpCamera->GetUniqueId(), mgr);
  }
}

void CCameraManager::RestoreHintlessCamera(CStateManager& mgr) {
  const TCastToConstPtr<CScriptCameraHint> hint = mgr.ObjectById(xa6_camHintId);
  const zeus::CTransform ballCamXf = x80_ballCamera->GetTransform();

  xa6_camHintId = kInvalidUniqueId;
  xa8_hintPriority = 1000;

  if (!hint) {
    return;
  }

  zeus::CVector3f camToPlayerFlat = mgr.GetPlayer().GetTranslation() - ballCamXf.origin;
  camToPlayerFlat.z() = 0.f;
  if (camToPlayerFlat.canBeNormalized()) {
    camToPlayerFlat.normalize();
  } else {
    camToPlayerFlat = mgr.GetPlayer().GetMoveDir();
  }

  x80_ballCamera->ResetToTweaks(mgr);
  x80_ballCamera->UpdateLookAtPosition(0.f, mgr);
  if (!mgr.GetPlayer().IsMorphBallTransitioning() &&
      hint->GetHint().GetBehaviourType() != CBallCamera::EBallCameraBehaviour::Default) {
    if ((hint->GetHint().GetOverrideFlags() & 0x1000) != 0) {
      x80_ballCamera->SetClampVelRange(hint->GetHint().GetClampVelRange());
      x80_ballCamera->SetClampVelTimer(hint->GetHint().GetClampVelTime());
    } else {
      x80_ballCamera->TeleportCamera(x80_ballCamera->UpdateLookDirection(camToPlayerFlat, mgr), mgr);
      InterpolateToBallCamera(ballCamXf, x80_ballCamera->GetUniqueId(), x80_ballCamera->GetLookPos(),
                              hint->GetHint().GetClampVelTime(), hint->GetHint().GetClampVelRange(),
                              hint->GetHint().GetClampRotRange(), (hint->GetHint().GetOverrideFlags() & 0x800) != 0,
                              mgr);
    }
  }
}

void CCameraManager::SkipBallCameraCinematic(CStateManager& mgr) {
  if (IsInCinematicCamera()) {
    x80_ballCamera->TeleportCamera(GetLastCineCamera(mgr)->GetTransform(), mgr);
    x80_ballCamera->SetFovInterpolation(GetLastCineCamera(mgr)->GetFov(), x80_ballCamera->GetFov(), 1.f, 0.f);
    SkipCinematic(mgr);
    SetCurrentCameraId(x80_ballCamera->GetUniqueId(), mgr);
  }
}

void CCameraManager::ApplyCameraHint(const CScriptCameraHint& hint, CStateManager& mgr) {
  if (x80_ballCamera->GetState() == CBallCamera::EBallCameraState::ToBall) {
    x80_ballCamera->SetState(CBallCamera::EBallCameraState::Default, mgr);
    mgr.GetPlayer().SetCameraState(CPlayer::EPlayerCameraState::Ball, mgr);
  }

  const TCastToConstPtr<CScriptCameraHint> oldHint = mgr.ObjectById(xa6_camHintId);
  xa6_camHintId = hint.GetUniqueId();
  xa8_hintPriority = hint.GetPriority();

  const zeus::CTransform camXf = GetCurrentCameraTransform(mgr);
  x80_ballCamera->ApplyCameraHint(mgr);

  if ((hint.GetHint().GetOverrideFlags() & 0x20) != 0) {
    x80_ballCamera->ResetPosition(mgr);
  }

  switch (hint.GetHint().GetBehaviourType()) {
  case CBallCamera::EBallCameraBehaviour::PathCameraDesiredPos:
  case CBallCamera::EBallCameraBehaviour::PathCamera:
    SetPathCamera(hint.GetDelegatedCamera(), mgr);
    break;
  case CBallCamera::EBallCameraBehaviour::SpindleCamera:
    SetSpindleCamera(hint.GetDelegatedCamera(), mgr);
    break;
  default:
    SetPathCamera(kInvalidUniqueId, mgr);
    SetSpindleCamera(kInvalidUniqueId, mgr);
    break;
  }

  if ((hint.GetHint().GetOverrideFlags() & 0x2000) != 0) {
    SkipBallCameraCinematic(mgr);
  }

  x80_ballCamera->UpdateLookAtPosition(0.f, mgr);

  if ((hint.GetHint().GetOverrideFlags() & 0x20) == 0 &&
      (hint.GetHint().GetBehaviourType() != CBallCamera::EBallCameraBehaviour::Default ||
       (oldHint && oldHint->GetHint().GetBehaviourType() != CBallCamera::EBallCameraBehaviour::Default))) {
    InterpolateToBallCamera(camXf, x80_ballCamera->GetUniqueId(), x80_ballCamera->GetLookPos(),
                            hint.GetHint().GetInterpolateTime(), hint.GetHint().GetClampVelRange(),
                            hint.GetHint().GetClampRotRange(), (hint.GetHint().GetOverrideFlags() & 0x400) != 0, mgr);
  }
}

void CCameraManager::UpdateCameraHints(float, CStateManager& mgr) {
  bool invalidHintRemoved = false;
  for (auto it = xac_cameraHints.begin(); it != xac_cameraHints.end();) {
    if (!TCastToConstPtr<CScriptCameraHint>(mgr.ObjectById(it->second))) {
      invalidHintRemoved = true;
      it = xac_cameraHints.erase(it);
      continue;
    }
    ++it;
  }

  bool inactiveHintRemoved = false;
  for (const auto& id : x2b0_inactiveCameraHints) {
    if (const TCastToConstPtr<CScriptCameraHint> hint = mgr.GetObjectById(id)) {
      if (hint->GetHelperCount() == 0 || hint->GetInactive()) {
        for (auto it = xac_cameraHints.begin(); it != xac_cameraHints.end(); ++it) {
          if (it->second == id) {
            xac_cameraHints.erase(it);
            if (xa6_camHintId == id) {
              inactiveHintRemoved = true;
              SetPathCamera(kInvalidUniqueId, mgr);
              SetSpindleCamera(kInvalidUniqueId, mgr);
            }
            break;
          }
        }
      }
    }
  }
  x2b0_inactiveCameraHints.clear();

  bool activeHintAdded = false;
  for (const auto& id : x334_activeCameraHints) {
    if (const TCastToConstPtr<CScriptCameraHint> hint = mgr.GetObjectById(id)) {
      bool activeHintPresent = false;
      for (auto it = xac_cameraHints.begin(); it != xac_cameraHints.end(); ++it) {
        if (it->second == id) {
          activeHintPresent = true;
          break;
        }
      }

      if (!activeHintPresent) {
        activeHintAdded = true;
        xac_cameraHints.emplace_back(hint->GetPriority(), id);
      }
    }
  }
  x334_activeCameraHints.clear();

  if (inactiveHintRemoved || activeHintAdded || invalidHintRemoved) {
    std::sort(xac_cameraHints.begin(), xac_cameraHints.end(),
              [](const auto& a, const auto& b) { return a.first < b.first; });
    zeus::CTransform ballCamXf = x80_ballCamera->GetTransform();
    if ((inactiveHintRemoved || invalidHintRemoved) && xac_cameraHints.empty()) {
      RestoreHintlessCamera(mgr);
      return;
    }
    bool foundHint = false;
    const CScriptCameraHint* bestHint = nullptr;
    for (auto& h : xac_cameraHints) {
      if (const TCastToConstPtr<CScriptCameraHint> hint = mgr.ObjectById(h.second)) {
        bestHint = hint.GetPtr();
        foundHint = true;
        break;
      }
    }
    if (!foundHint) {
      RestoreHintlessCamera(mgr);
    }

    bool changeHint = false;
    if (bestHint && foundHint) {
      if ((bestHint->GetHint().GetOverrideFlags() & 0x80) != 0 && xac_cameraHints.size() > 1) {
        zeus::CVector3f ballPos = mgr.GetPlayer().GetBallPosition();
        if ((bestHint->GetHint().GetOverrideFlags() & 0x100) != 0) {
          zeus::CVector3f camToBall = ballPos - ballCamXf.origin;
          if (camToBall.canBeNormalized()) {
            camToBall.normalize();
          } else {
            camToBall = ballCamXf.basis[1];
          }

          for (auto it = xac_cameraHints.begin() + 1; it != xac_cameraHints.end(); ++it) {
            if (const TCastToConstPtr<CScriptCameraHint> hint = mgr.ObjectById(it->second)) {
              if ((hint->GetHint().GetOverrideFlags() & 0x80) != 0 && hint->GetPriority() == bestHint->GetPriority() &&
                  hint->GetAreaIdAlways() == bestHint->GetAreaIdAlways()) {
                zeus::CVector3f hintToBall = ballPos - bestHint->GetTranslation();
                if (hintToBall.canBeNormalized()) {
                  hintToBall.normalize();
                } else {
                  hintToBall = bestHint->GetTransform().basis[1];
                }

                const float camHintDot = zeus::clamp(-1.f, camToBall.dot(hintToBall), 1.f);

                zeus::CVector3f thisHintToBall = ballPos - hint->GetTranslation();
                if (thisHintToBall.canBeNormalized()) {
                  thisHintToBall.normalize();
                } else {
                  thisHintToBall = hint->GetTransform().basis[1];
                }

                const float camThisHintDot = zeus::clamp(-1.f, camToBall.dot(thisHintToBall), 1.f);

                if (camThisHintDot > camHintDot) {
                  bestHint = hint.GetPtr();
                }
              } else {
                break;
              }
            } else {
              break;
            }
          }
        } else {
          if (const TCastToConstPtr<CActor> act = mgr.GetObjectById(bestHint->GetFirstHelper())) {
            const zeus::CVector3f f26 = act->GetTranslation() - mgr.GetPlayer().GetBallPosition();
            zeus::CVector3f ballToHelper = f26;
            if (ballToHelper.canBeNormalized()) {
              ballToHelper.normalize();
            } else {
              ballToHelper = bestHint->GetTransform().basis[1];
            }

            for (auto it = xac_cameraHints.begin() + 1; it != xac_cameraHints.end(); ++it) {
              if (const TCastToConstPtr<CScriptCameraHint> hint = mgr.ObjectById(it->second)) {
                if ((hint->GetHint().GetOverrideFlags() & 0x80) != 0 &&
                    hint->GetPriority() == bestHint->GetPriority() &&
                    hint->GetAreaIdAlways() == bestHint->GetAreaIdAlways()) {
                  zeus::CVector3f hintToHelper = act->GetTranslation() - bestHint->GetTranslation();
                  if (hintToHelper.canBeNormalized()) {
                    hintToHelper.normalize();
                  } else {
                    hintToHelper = bestHint->GetTransform().basis[1];
                  }

                  const float ballHintDot = zeus::clamp(-1.f, ballToHelper.dot(hintToHelper), 1.f);

                  zeus::CVector3f thisBallToHelper = f26;
                  if (thisBallToHelper.canBeNormalized()) {
                    thisBallToHelper.normalize();
                  } else {
                    thisBallToHelper = hint->GetTransform().basis[1];
                  }

                  zeus::CVector3f thisHintToHelper = act->GetTranslation() - hint->GetTranslation();
                  if (thisHintToHelper.canBeNormalized()) {
                    thisHintToHelper.normalize();
                  } else {
                    thisHintToHelper = hint->GetTransform().basis[1];
                  }

                  const float thisBallHintDot = zeus::clamp(-1.f, thisBallToHelper.dot(thisHintToHelper), 1.f);

                  if (thisBallHintDot > ballHintDot) {
                    bestHint = hint.GetPtr();
                  }
                } else {
                  break;
                }
              } else {
                break;
              }
            }
          }
        }

        if (bestHint->GetUniqueId() != xa6_camHintId) {
          changeHint = true;
        }
      } else if (xa6_camHintId != bestHint->GetUniqueId()) {
        if (bestHint->GetHint().GetBehaviourType() == CBallCamera::EBallCameraBehaviour::HintInitializePosition) {
          if ((bestHint->GetHint().GetOverrideFlags() & 0x20) != 0) {
            x80_ballCamera->TeleportCamera(zeus::lookAt(bestHint->GetTranslation(), x80_ballCamera->GetLookPos()), mgr);
          }
          DeleteCameraHint(bestHint->GetUniqueId(), mgr);
          if ((bestHint->GetHint().GetOverrideFlags() & 0x2000) != 0) {
            SkipBallCameraCinematic(mgr);
          }
          changeHint = false;
        } else {
          changeHint = true;
        }
      }

      if (changeHint) {
        ApplyCameraHint(*bestHint, mgr);
      }
    }
  }
}

void CCameraManager::ThinkCameras(float dt, CStateManager& mgr) {
  CGameCameraList gcList = mgr.GetCameraObjectList();

  for (CEntity* ent : gcList) {
    if (const TCastToPtr<CGameCamera> gc = ent) {
      gc->Think(dt, mgr);
      gc->UpdatePerspective(dt);
    }
  }

  if (IsInCinematicCamera()) {
    return;
  }

  const TUniqueId camId = GetLastCameraId();
  if (const CGameCamera* cam = TCastToConstPtr<CGameCamera>(mgr.GetObjectById(camId))) {
    x3bc_curFov = cam->GetFov();
  }
}

void CCameraManager::UpdateFog(float dt, CStateManager& mgr) {
  if (x98_fogDensitySpeed != 0.f) {
    x94_fogDensityFactor += dt * x98_fogDensitySpeed;
    if ((x98_fogDensitySpeed > 0.f) ? x94_fogDensityFactor > x9c_fogDensityFactorTarget
                                    : x94_fogDensityFactor < x9c_fogDensityFactorTarget) {
      x94_fogDensityFactor = x9c_fogDensityFactorTarget;
      x98_fogDensitySpeed = 0.f;
    }
  }

  if (x74_fluidCounter) {
    if (const TCastToConstPtr<CScriptWater> water = mgr.GetObjectById(x78_fluidId)) {
      const zeus::CVector2f zRange(GetCurrentCamera(mgr)->GetNearClipDistance(),
                                   CalculateFogDensity(mgr, water.GetPtr()));
      x3c_fog.SetFogExplicit(ERglFogMode::PerspExp, water->GetInsideFogColor(), zRange);
      if (mgr.GetPlayerState()->GetActiveVisor(mgr) == CPlayerState::EPlayerVisor::Thermal) {
        mgr.GetCameraFilterPass(4).DisableFilter(0.f);
      } else {
        mgr.GetCameraFilterPass(4).SetFilter(EFilterType::Multiply, EFilterShape::Fullscreen, 0.f,
                                             water->GetInsideFogColor(), {});
      }
    }
    xa0_26_inWater = true;
  } else if (xa0_26_inWater) {
    mgr.GetCameraManager()->x3c_fog.DisableFog();
    mgr.GetCameraFilterPass(4).DisableFilter(0.f);
    xa0_26_inWater = false;
  }

  x3c_fog.Update(dt);
}

void CCameraManager::UpdateRumble(float dt, CStateManager& mgr) {
  x30_shakeOffset = zeus::skZero3f;
  for (auto it = x14_shakers.begin(); it != x14_shakers.end();) {
    CCameraShakeData& shaker = *it;
    shaker.Update(dt, mgr);
    if (shaker.x4_curTime >= shaker.x0_duration) {
      it = x14_shakers.erase(it);
      continue;
    }
    x30_shakeOffset += shaker.GetPoint();
    ++it;
  }

  if (!x14_shakers.empty() && !xa0_25_rumbling && xa0_24_pendingRumble) {
    mgr.GetRumbleManager().Rumble(mgr, ERumbleFxId::CameraShake, 1.f, ERumblePriority::Two);
    xa0_25_rumbling = true;
  }

  if (x90_rumbleCooldown > 0.f) {
    x90_rumbleCooldown -= dt;
  } else if (xa0_25_rumbling) {
    xa0_24_pendingRumble = false;
    xa0_25_rumbling = false;
  }

  if (mgr.GetPlayer().GetCameraState() != CPlayer::EPlayerCameraState::FirstPerson && !IsInCinematicCamera()) {
    x30_shakeOffset = zeus::skZero3f;
  }
}

void CCameraManager::UpdateListener(CStateManager& mgr) {
  const zeus::CTransform xf = GetCurrentCameraTransform(mgr);
  CSfxManager::UpdateListener(xf.origin, zeus::skZero3f, xf.frontVector(), xf.upVector(), 1.f);
}

float CCameraManager::CalculateFogDensity(CStateManager& mgr, const CScriptWater* water) const {
  const float distanceFactor = 1.f - water->GetFluidPlane().GetAlpha();
  float distance = 0;
  if (mgr.GetPlayerState()->HasPowerUp(CPlayerState::EItemType::GravitySuit)) {
    distance =
        g_tweakGame->GetGravityWaterFogDistanceRange() * distanceFactor + g_tweakGame->GetGravityWaterFogDistanceBase();
  } else {
    distance = g_tweakGame->GetWaterFogDistanceRange() * distanceFactor + g_tweakGame->GetWaterFogDistanceBase();
  }

  return distance * x94_fogDensityFactor;
}

void CCameraManager::ResetCameras(CStateManager& mgr) {
  zeus::CTransform xf = mgr.GetPlayer().CreateTransformFromMovementDirection();
  xf.origin = mgr.GetPlayer().GetEyePosition();

  for (CEntity* ent : mgr.GetCameraObjectList()) {
    const TCastToPtr<CGameCamera> camObj(ent);
    camObj->Reset(xf, mgr);
  }
}

void CCameraManager::SetSpecialCameras(CFirstPersonCamera& fp, CBallCamera& ball) {
  x7c_fpCamera = &fp;
  x80_ballCamera = &ball;
}

void CCameraManager::ProcessInput(const CFinalInput& input, CStateManager& stateMgr) {
  for (CEntity* ent : stateMgr.GetCameraObjectList()) {
    if (ent == nullptr) {
      continue;
    }
    auto& cam = static_cast<CGameCamera&>(*ent);
    if (input.ControllerIdx() != cam.x16c_controllerIdx) {
      continue;
    }
    cam.ProcessInput(input, stateMgr);
  }
}

void CCameraManager::RenderCameras(CStateManager& mgr) {
  for (CEntity* cam : mgr.GetCameraObjectList()) {
    static_cast<CGameCamera*>(cam)->Render(mgr);
  }
}

void CCameraManager::SetupBallCamera(CStateManager& mgr) {
  if (const TCastToConstPtr<CScriptCameraHint> hint = mgr.ObjectById(xa6_camHintId)) {
    if (hint->GetHint().GetBehaviourType() == CBallCamera::EBallCameraBehaviour::HintInitializePosition) {
      if ((hint->GetHint().GetOverrideFlags() & 0x20) != 0) {
        x80_ballCamera->TeleportCamera(hint->GetTransform(), mgr);
      }
      AddInactiveCameraHint(xa6_camHintId, mgr);
    } else {
      ApplyCameraHint(*hint, mgr);
    }
  }
}

void CCameraManager::SetPlayerCamera(CStateManager& mgr, TUniqueId newCamId) {
  if (x88_interpCamera->GetActive()) {
    x88_interpCamera->SetActive(false);
    x80_ballCamera->SkipFovInterpolation();
    if (!ShouldBypassInterpolation())
      SetCurrentCameraId(newCamId, mgr);
  }
}

float CCameraManager::GetCameraBobMagnitude() const {
  return 1.f - zeus::clamp(-1.f,
                           std::fabs(zeus::clamp(-1.f, x7c_fpCamera->GetTransform().basis[1].dot(zeus::skUp), 1.f)) /
                               std::cos(2.f * M_PIF / 12.f),
                           1.f);
}

bool CCameraManager::HasBallCameraInitialPositionHint(CStateManager& mgr) const {
  if (HasCameraHint(mgr)) {
    switch (mgr.GetCameraManager()->GetCameraHint(mgr)->GetHint().GetBehaviourType()) {
    case CBallCamera::EBallCameraBehaviour::HintBallToCam:
    case CBallCamera::EBallCameraBehaviour::HintFixedPosition:
    case CBallCamera::EBallCameraBehaviour::HintFixedTransform:
    case CBallCamera::EBallCameraBehaviour::PathCamera:
    case CBallCamera::EBallCameraBehaviour::SpindleCamera:
      return true;
    default:
      return false;
    }
  }
  return false;
}

void CCameraManager::RemoveCinemaCamera(TUniqueId uid, CStateManager& mgr) {
  const auto search = std::find(x4_cineCameras.cbegin(), x4_cineCameras.cend(), uid);

  if (search == x4_cineCameras.cend()) {
    return;
  }

  x4_cineCameras.erase(search);
}

void CCameraManager::DeleteCameraHint(TUniqueId id, CStateManager& mgr) {
  const TCastToPtr<CScriptCameraHint> hint = mgr.ObjectById(id);

  if (!hint) {
    return;
  }

  const auto search = std::find_if(x2b0_inactiveCameraHints.cbegin(), x2b0_inactiveCameraHints.cend(),
                                   [id](TUniqueId tid) { return tid == id; });

  if (search != x2b0_inactiveCameraHints.cend()) {
    return;
  }

  hint->ClearIdList();
  hint->SetInactive(true);
  if (x2b0_inactiveCameraHints.size() != 64) {
    x2b0_inactiveCameraHints.push_back(id);
  }
}

void CCameraManager::AddInactiveCameraHint(TUniqueId id, CStateManager& mgr) {
  if (const TCastToConstPtr<CScriptCameraHint> hint = mgr.ObjectById(id)) {
    const auto search = std::find_if(x2b0_inactiveCameraHints.cbegin(), x2b0_inactiveCameraHints.cend(),
                                     [id](TUniqueId tid) { return tid == id; });
    if (search == x2b0_inactiveCameraHints.cend() && x2b0_inactiveCameraHints.size() != 64) {
      x2b0_inactiveCameraHints.push_back(id);
    }
  }
}

void CCameraManager::AddActiveCameraHint(TUniqueId id, CStateManager& mgr) {
  if (const TCastToConstPtr<CScriptCameraHint> hint = mgr.ObjectById(id)) {
    const auto search = std::find_if(x334_activeCameraHints.cbegin(), x334_activeCameraHints.cend(),
                                     [id](TUniqueId tid) { return tid == id; });
    if (search == x334_activeCameraHints.cend() && xac_cameraHints.size() != 64 &&
        x334_activeCameraHints.size() != 64) {
      x334_activeCameraHints.push_back(id);
    }
  }
}

TUniqueId CCameraManager::GetLastCineCameraId() const {
  if (x4_cineCameras.empty()) {
    return kInvalidUniqueId;
  }
  return x4_cineCameras.back();
}

const CCinematicCamera* CCameraManager::GetLastCineCamera(CStateManager& mgr) const {
  return static_cast<const CCinematicCamera*>(mgr.GetObjectById(GetLastCineCameraId()));
}

const CScriptCameraHint* CCameraManager::GetCameraHint(CStateManager& mgr) const {
  return TCastToConstPtr<CScriptCameraHint>(mgr.GetObjectById(xa6_camHintId)).GetPtr();
}

bool CCameraManager::HasCameraHint(CStateManager& mgr) const {
  if (xac_cameraHints.empty() || xa6_camHintId == kInvalidUniqueId)
    return false;
  return mgr.GetObjectById(xa6_camHintId) != nullptr;
}

bool CCameraManager::IsInterpolationCameraActive() const { return x88_interpCamera->GetActive(); }

void CCameraManager::SetFogDensity(float fogDensityTarget, float fogDensitySpeed) {
  x9c_fogDensityFactorTarget = fogDensityTarget;
  x98_fogDensitySpeed = (x9c_fogDensityFactorTarget >= x94_fogDensityFactor ? fogDensitySpeed : -fogDensitySpeed);
}
} // namespace metaforce