#include "Runtime/MP1/CFrontEndUI.hpp"

#include <algorithm>
#include <array>
#include <ctime>

#include "NESEmulator/CNESEmulator.hpp"

#include "Runtime/CArchitectureMessage.hpp"
#include "Runtime/CArchitectureQueue.hpp"
#include "Runtime/CDependencyGroup.hpp"
#include "Runtime/CDvdFile.hpp"
#include "Runtime/CGameState.hpp"
#include "Runtime/CSimplePool.hpp"
#include "Runtime/GameGlobalObjects.hpp"
#include "Runtime/Audio/CAudioGroupSet.hpp"
#include "Runtime/Audio/CSfxManager.hpp"
#include "Runtime/Graphics/CMoviePlayer.hpp"
#include "Runtime/GuiSys/CGuiFrame.hpp"
#include "Runtime/GuiSys/CGuiModel.hpp"
#include "Runtime/GuiSys/CGuiSliderGroup.hpp"
#include "Runtime/GuiSys/CGuiTableGroup.hpp"
#include "Runtime/GuiSys/CGuiTextPane.hpp"
#include "Runtime/GuiSys/CGuiWidgetDrawParms.hpp"
#include "Runtime/GuiSys/CStringTable.hpp"
#include "Runtime/Input/RumbleFxTable.hpp"
#include "Runtime/MP1/CQuitGameScreen.hpp"
#include "Runtime/MP1/CSaveGameScreen.hpp"
#include "Runtime/MP1/CSlideShow.hpp"
#include "Runtime/MP1/MP1.hpp"

namespace urde::MP1 {
namespace {
#define FE_USE_SECONDS_IN_ELAPSED 1

/* Music volume constants */
constexpr float FE1_VOL = 0.7421875f;
constexpr float FE2_VOL = 0.7421875f;

/* L/R Stereo transition cues */
constexpr std::array<std::array<u16, 2>, 3> FETransitionBackSFX{{
    {SFXfnt_transfore_00L, SFXfnt_transfore_00R},
    {SFXfnt_transfore_01L, SFXfnt_transfore_01R},
    {SFXfnt_transfore_02L, SFXfnt_transfore_02R},
}};

constexpr std::array<std::array<u16, 2>, 3> FETransitionForwardSFX{{
    {SFXfnt_transback_00L, SFXfnt_transback_00R},
    {SFXfnt_transback_01L, SFXfnt_transback_01R},
    {SFXfnt_transback_02L, SFXfnt_transback_02R},
}};

struct FEMovie {
  const char* path;
  bool loop;
};

constexpr std::array<FEMovie, 9> FEMovies{{
    {"Video/00_first_start.thp", false},
    {"Video/01_startloop.thp", true},
    {"Video/02_start_fileselect_A.thp", false},
    {"Video/03_fileselectloop.thp", true},
    {"Video/04_fileselect_playgame_A.thp", false},
    {"Video/06_fileselect_GBA.thp", false},
    {"Video/07_GBAloop.thp", true},
    {"Video/08_GBA_fileselect.thp", false},
    {"Video/08_GBA_fileselect.thp", false},
}};

constexpr SObjectTag g_DefaultWorldTag = {FOURCC('MLVL'), 0x158efe17};

constexpr std::array<float, 3> AudioFadeTimeA{
    0.44f,
    5.41f,
    3.41f,
};

constexpr std::array<float, 3> AudioFadeTimeB{
    4.2f,
    6.1f,
    6.1f,
};

constexpr std::array<CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType, 10> NextLinkUI{
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::ConnectSocket,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::PressStartAndSelect,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::BeginLink,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::Linking,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::Empty,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::TurnOffGBA,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::Complete,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::InsertPak,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::Empty,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::Empty,
};

constexpr std::array<CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType, 10> PrevLinkUI{
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::Cancelled,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::Cancelled,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::Cancelled,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::Cancelled,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::Empty,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::Cancelled,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::Empty,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::Cancelled,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::Empty,
    CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EUIType::Empty,
};
} // Anonymous namespace

void CFrontEndUI::PlayAdvanceSfx() {
  CSfxManager::SfxStart(SFXfnt_advance_L, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
  CSfxManager::SfxStart(SFXfnt_advance_R, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
}

CFrontEndUI::SNewFileSelectFrame::SNewFileSelectFrame(CSaveGameScreen* sui, u32 rnd, CFrontEndUITouchBar& touchBar)
: x0_rnd(rnd), x4_saveUI(sui), m_touchBar(touchBar) {
  x10_frme = g_SimplePool->GetObj("FRME_NewFileSelect");
}

void CFrontEndUI::SNewFileSelectFrame::FinishedLoading() {
  x1c_loadedFrame->Reset();
  x1c_loadedFrame->SetAspectConstraint(1.78f);

  x20_tablegroup_fileselect = static_cast<CGuiTableGroup*>(x1c_loadedFrame->FindWidget("tablegroup_fileselect"));
  x24_model_erase = static_cast<CGuiModel*>(x1c_loadedFrame->FindWidget("model_erase"));
  xf8_model_erase_position = x24_model_erase->GetLocalPosition();
  x28_textpane_erase = FindTextPanePair(x1c_loadedFrame, "textpane_erase");
  x38_textpane_gba = FindTextPanePair(x1c_loadedFrame, "textpane_gba");
  x30_textpane_cheats = FindTextPanePair(x1c_loadedFrame, "textpane_cheats");
  x48_textpane_popupadvance = FindTextPanePair(x1c_loadedFrame, "textpane_popupadvance");
  x50_textpane_popupcancel = FindTextPanePair(x1c_loadedFrame, "textpane_popupcancel");
  x58_textpane_popupextra = FindTextPanePair(x1c_loadedFrame, "textpane_popupextra");
  x40_tablegroup_popup = static_cast<CGuiTableGroup*>(x1c_loadedFrame->FindWidget("tablegroup_popup"));
  x44_model_dash7 = static_cast<CGuiModel*>(x1c_loadedFrame->FindWidget("model_dash7"));
  x60_textpane_cancel = static_cast<CGuiTextPane*>(x1c_loadedFrame->FindWidget("textpane_cancel"));
  FindAndSetPairText(x1c_loadedFrame, "textpane_title", g_MainStringTable->GetString(97));
  CGuiTextPane* proceed = static_cast<CGuiTextPane*>(x1c_loadedFrame->FindWidget("textpane_proceed"));
  if (proceed)
    proceed->TextSupport().SetText(g_MainStringTable->GetString(85));
  x40_tablegroup_popup->SetIsVisible(false);
  x40_tablegroup_popup->SetIsActive(false);
  x40_tablegroup_popup->SetVertical(false);
  CGuiWidget* worker = x40_tablegroup_popup->GetWorkerWidget(2);
  worker->SetIsSelectable(false);
  worker->SetVisibility(false, ETraversalMode::Children);

  x20_tablegroup_fileselect->SetMenuAdvanceCallback([this](CGuiTableGroup* caller) { DoFileMenuAdvance(caller); });
  x20_tablegroup_fileselect->SetMenuSelectionChangeCallback(
      [this](CGuiTableGroup* caller, int oldSel) { DoSelectionChange(caller, oldSel); });
  x20_tablegroup_fileselect->SetMenuCancelCallback([this](CGuiTableGroup* caller) { DoFileMenuCancel(caller); });

  x40_tablegroup_popup->SetMenuAdvanceCallback([this](CGuiTableGroup* caller) { DoPopupAdvance(caller); });
  x40_tablegroup_popup->SetMenuSelectionChangeCallback(
      [this](CGuiTableGroup* caller, int oldSel) { DoSelectionChange(caller, oldSel); });
  x40_tablegroup_popup->SetMenuCancelCallback([this](CGuiTableGroup* caller) { DoPopupCancel(caller); });

  for (size_t i = 0; i < x64_fileSelections.size(); ++i) {
    x64_fileSelections[i] = FindFileSelectOption(x1c_loadedFrame, int(i));
  }

  x104_rowPitch =
      (x64_fileSelections[1].x0_base->GetLocalPosition() - x64_fileSelections[0].x0_base->GetLocalPosition()).z();
}

bool CFrontEndUI::SNewFileSelectFrame::PumpLoad() {
  if (x1c_loadedFrame)
    return true;
  if (x10_frme.IsLoaded()) {
    if (x10_frme->GetIsFinishedLoading()) {
      x1c_loadedFrame = x10_frme.GetObj();
      FinishedLoading();
      return true;
    }
  }
  return false;
}

bool CFrontEndUI::SNewFileSelectFrame::IsTextDoneAnimating() const {
  if (x64_fileSelections[0].x28_curField != 4)
    return false;
  if (x64_fileSelections[1].x28_curField != 4)
    return false;
  if (x64_fileSelections[2].x28_curField != 4)
    return false;
  if (!x28_textpane_erase.x0_panes[0]->GetTextSupport().IsAnimationDone())
    return false;
  return x38_textpane_gba.x0_panes[0]->GetTextSupport().IsAnimationDone();
}

void CFrontEndUI::SNewFileSelectFrame::Update(float dt) {
  bool saveReady = x4_saveUI->GetUIType() == CSaveGameScreen::EUIType::SaveReady;
  if (saveReady != x10c_saveReady) {
    if (saveReady) {
      ClearFrameContents();
    } else if (x8_subMenu != ESubMenu::Root) {
      ResetFrame();
      DeactivateEraseGamePopup();
      DeactivateNewGamePopup();
      x8_subMenu = ESubMenu::Root;
    }
    x10c_saveReady = saveReady;
  }
  if (x10c_saveReady)
    SetupFrameContents();

  x1c_loadedFrame->Update(dt);
}

CFrontEndUI::SNewFileSelectFrame::EAction
CFrontEndUI::SNewFileSelectFrame::ProcessUserInput(const CFinalInput& input, CFrontEndUITouchBar::EAction tbAction) {
  xc_action = EAction::None;

  if (x8_subMenu != ESubMenu::EraseGamePopup)
    x4_saveUI->ProcessUserInput(input);

  if (IsTextDoneAnimating())
    x108_curTime = std::min(0.5f, x108_curTime + input.DeltaTime());

  if (x108_curTime < 0.5f)
    return xc_action;

  if (x10c_saveReady) {
    x1c_loadedFrame->ProcessUserInput(input);
    if (x8_subMenu == ESubMenu::Root || x8_subMenu == ESubMenu::EraseGame) {
      if (m_touchBar.GetPhase() != CFrontEndUITouchBar::EPhase::FileSelect)
        HandleActiveChange(x20_tablegroup_fileselect);

      if (tbAction >= CFrontEndUITouchBar::EAction::FileA && tbAction <= CFrontEndUITouchBar::EAction::ImageGallery) {
        x20_tablegroup_fileselect->SetUserSelection(int(tbAction) - int(CFrontEndUITouchBar::EAction::FileA));
        HandleActiveChange(x20_tablegroup_fileselect);
        DoFileMenuAdvance(x20_tablegroup_fileselect);
      } else if (tbAction == CFrontEndUITouchBar::EAction::Back) {
        DoFileMenuCancel(x20_tablegroup_fileselect);
      }
    } else if (x8_subMenu == ESubMenu::EraseGamePopup) {
      if (m_touchBar.GetPhase() != CFrontEndUITouchBar::EPhase::EraseBack)
        HandleActiveChange(x40_tablegroup_popup);

      if (tbAction != CFrontEndUITouchBar::EAction::None) {
        if (tbAction == CFrontEndUITouchBar::EAction::Confirm)
          x40_tablegroup_popup->SetUserSelection(1);
        else
          x40_tablegroup_popup->SetUserSelection(0);
        HandleActiveChange(x40_tablegroup_popup);
        DoPopupAdvance(x40_tablegroup_popup);
      }
    } else if (x8_subMenu == ESubMenu::NewGamePopup) {
      if (m_touchBar.GetPhase() != CFrontEndUITouchBar::EPhase::StartOptions)
        HandleActiveChange(x40_tablegroup_popup);

      if (tbAction == CFrontEndUITouchBar::EAction::Back) {
        DoPopupCancel(x40_tablegroup_popup);
      } else if (tbAction != CFrontEndUITouchBar::EAction::None) {
        if (tbAction == CFrontEndUITouchBar::EAction::Options)
          x40_tablegroup_popup->SetUserSelection(1);
        else if (tbAction == CFrontEndUITouchBar::EAction::Start || tbAction == CFrontEndUITouchBar::EAction::Hard)
          x40_tablegroup_popup->SetUserSelection(0);
        else if (tbAction == CFrontEndUITouchBar::EAction::Normal)
          x40_tablegroup_popup->SetUserSelection(2);
        HandleActiveChange(x40_tablegroup_popup);
        DoPopupAdvance(x40_tablegroup_popup);
      }
    }
  }

  if (x10d_needsEraseToggle) {
    if (x40_tablegroup_popup->GetIsActive())
      DeactivateEraseGamePopup();
    else
      ActivateEraseGamePopup();
    x10d_needsEraseToggle = false;
  }

  if (x10e_needsNewToggle) {
    if (x40_tablegroup_popup->GetIsActive())
      DeactivateNewGamePopup();
    else
      ActivateNewGamePopup();
    x10e_needsNewToggle = false;
  }

  return xc_action;
}

void CFrontEndUI::SNewFileSelectFrame::Draw() const {
  if (x1c_loadedFrame && x10c_saveReady)
    x1c_loadedFrame->Draw(CGuiWidgetDrawParms::Default);
}

void CFrontEndUI::SNewFileSelectFrame::HandleActiveChange(CGuiTableGroup* active) {
  if (!active)
    return;

  active->SetColors(zeus::skWhite, zeus::CColor{0.627450f, 0.627450f, 0.627450f, 0.784313f});

  if (active == x20_tablegroup_fileselect) {
    x24_model_erase->SetLocalTransform(zeus::CTransform::Translate(
        zeus::CVector3f{0.f, 0.f, active->GetUserSelection() * x104_rowPitch} + xf8_model_erase_position));

    /* Set Touch Bar contents here */
    std::array<CFrontEndUITouchBar::SFileSelectDetail, 3> tbDetails{};
    for (size_t i = 0; i < tbDetails.size(); ++i) {
      if (const CGameState::GameFileStateInfo* data = x4_saveUI->GetGameData(int(i))) {
        tbDetails[i].state =
            data->x20_hardMode ? CFrontEndUITouchBar::EFileState::Hard : CFrontEndUITouchBar::EFileState::Normal;
        tbDetails[i].percent = data->x18_itemPercent;
      }
    }
    m_touchBar.SetFileSelectPhase(tbDetails.data(), x8_subMenu == ESubMenu::EraseGame, CSlideShow::SlideShowGalleryFlags());
  } else if (active == x40_tablegroup_popup) {
    if (x8_subMenu == ESubMenu::EraseGamePopup)
      m_touchBar.SetPhase(CFrontEndUITouchBar::EPhase::EraseBack);
    else if (x8_subMenu == ESubMenu::NewGamePopup)
      m_touchBar.SetStartOptionsPhase(g_GameState->SystemOptions().GetPlayerBeatNormalMode());
    else
      m_touchBar.SetPhase(CFrontEndUITouchBar::EPhase::None);
  }

  if (x8_subMenu == ESubMenu::Root || x8_subMenu == ESubMenu::NewGamePopup)
    x24_model_erase->SetIsVisible(false);
  else
    x24_model_erase->SetIsVisible(true);
}

void CFrontEndUI::SNewFileSelectFrame::DeactivateEraseGamePopup() {
  x40_tablegroup_popup->SetIsActive(false);
  x40_tablegroup_popup->SetIsVisible(false);
  x20_tablegroup_fileselect->SetIsActive(true);

  HandleActiveChange(x20_tablegroup_fileselect);

  x64_fileSelections[x20_tablegroup_fileselect->GetUserSelection()].x0_base->SetColor(zeus::skWhite);
}

void CFrontEndUI::SNewFileSelectFrame::ActivateEraseGamePopup() {
  x40_tablegroup_popup->SetIsActive(true);
  x40_tablegroup_popup->SetIsVisible(true);
  x40_tablegroup_popup->SetLocalTransform(
      zeus::CTransform::Translate(0.f, 0.f, x20_tablegroup_fileselect->GetUserSelection() * x104_rowPitch) *
      x40_tablegroup_popup->GetTransform());
  x20_tablegroup_fileselect->SetIsActive(false);

  x8_subMenu = ESubMenu::EraseGamePopup;
  HandleActiveChange(x40_tablegroup_popup);

  x48_textpane_popupadvance.SetPairText(g_MainStringTable->GetString(95));
  x50_textpane_popupcancel.SetPairText(g_MainStringTable->GetString(38));

  x64_fileSelections[x20_tablegroup_fileselect->GetUserSelection()].x0_base->SetColor(zeus::CColor{1.f, 1.f, 1.f, 0.f});
  x44_model_dash7->SetVisibility(false, ETraversalMode::Children);
}

void CFrontEndUI::SNewFileSelectFrame::DeactivateNewGamePopup() {
  x40_tablegroup_popup->SetIsActive(false);
  x40_tablegroup_popup->SetIsVisible(false);
  x20_tablegroup_fileselect->SetIsActive(true);

  CGuiWidget* worker = x40_tablegroup_popup->GetWorkerWidget(2);
  worker->SetIsSelectable(false);
  worker->SetVisibility(false, ETraversalMode::Children);

  x44_model_dash7->SetVisibility(false, ETraversalMode::Children);

  HandleActiveChange(x20_tablegroup_fileselect);

  x64_fileSelections[x20_tablegroup_fileselect->GetUserSelection()].x0_base->SetColor(zeus::skWhite);
  x60_textpane_cancel->TextSupport().SetText(u"");
}

void CFrontEndUI::SNewFileSelectFrame::ActivateNewGamePopup() {
  x40_tablegroup_popup->SetIsActive(true);
  x40_tablegroup_popup->SetIsVisible(true);
  x40_tablegroup_popup->SetUserSelection(0);
  x40_tablegroup_popup->SetLocalTransform(
      zeus::CTransform::Translate(0.f, 0.f, x20_tablegroup_fileselect->GetUserSelection() * x104_rowPitch) *
      x40_tablegroup_popup->GetTransform());
  x20_tablegroup_fileselect->SetIsActive(false);

  x8_subMenu = ESubMenu::NewGamePopup;
  HandleActiveChange(x40_tablegroup_popup);
  x64_fileSelections[x20_tablegroup_fileselect->GetUserSelection()].x0_base->SetColor(zeus::CColor{1.f, 1.f, 1.f, 0.f});

  PlayAdvanceSfx();

  if (g_GameState->SystemOptions().GetPlayerBeatNormalMode()) {
    x48_textpane_popupadvance.SetPairText(g_MainStringTable->GetString(102));
    x50_textpane_popupcancel.SetPairText(g_MainStringTable->GetString(94));
    x58_textpane_popupextra.SetPairText(g_MainStringTable->GetString(101));
    CGuiWidget* worker = x40_tablegroup_popup->GetWorkerWidget(2);
    worker->SetIsSelectable(true);
    worker->SetVisibility(true, ETraversalMode::Children);
    x44_model_dash7->SetVisibility(true, ETraversalMode::Children);
  } else {
    x48_textpane_popupadvance.SetPairText(g_MainStringTable->GetString(67));
    x50_textpane_popupcancel.SetPairText(g_MainStringTable->GetString(94));
    x44_model_dash7->SetVisibility(false, ETraversalMode::Children);
  }
  x60_textpane_cancel->TextSupport().SetText(g_MainStringTable->GetString(82));
}

void CFrontEndUI::SNewFileSelectFrame::ResetFrame() {
  x8_subMenu = ESubMenu::Root;

  x38_textpane_gba.x0_panes[0]->SetIsSelectable(true);
  x38_textpane_gba.x0_panes[0]->TextSupport().SetFontColor(zeus::skWhite);

  x30_textpane_cheats.x0_panes[0]->SetIsSelectable(true);
  x30_textpane_cheats.x0_panes[0]->TextSupport().SetFontColor(zeus::skWhite);

  ClearFrameContents();

  for (int i = 2; i >= 0; --i)
    x20_tablegroup_fileselect->GetWorkerWidget(i)->SetIsSelectable(true);
  x60_textpane_cancel->TextSupport().SetText(u"");
}

void CFrontEndUI::SNewFileSelectFrame::ActivateErase() {
  x8_subMenu = ESubMenu::EraseGame;
  x28_textpane_erase.x0_panes[0]->SetIsSelectable(false);
  zeus::CColor color = zeus::skGrey;
  color.a() = 0.5f;
  x28_textpane_erase.x0_panes[0]->TextSupport().SetFontColor(color);
  x38_textpane_gba.x0_panes[0]->TextSupport().SetFontColor(color);
  x30_textpane_cheats.x0_panes[0]->TextSupport().SetFontColor(color);
  x38_textpane_gba.x0_panes[0]->SetIsSelectable(false);
  x30_textpane_cheats.x0_panes[0]->SetIsSelectable(false);

  for (int i = 2; i >= 0; --i) {
    SFileMenuOption& fileOpt = x64_fileSelections[i];
    if (x4_saveUI->GetGameData(i)) {
      fileOpt.x0_base->SetIsSelectable(true);
      x20_tablegroup_fileselect->SetUserSelection(i);
    } else {
      fileOpt.x0_base->SetIsSelectable(false);
    }
  }

  x60_textpane_cancel->TextSupport().SetText(g_MainStringTable->GetString(82));
  HandleActiveChange(x20_tablegroup_fileselect);
}

void CFrontEndUI::SNewFileSelectFrame::ClearFrameContents() {
  x108_curTime = 0.f;
  bool hasSave = false;
  for (size_t i = 0; i < x64_fileSelections.size(); ++i) {
    if (x4_saveUI->GetGameData(int(i))) {
      hasSave = true;
    }

    SFileMenuOption& option = x64_fileSelections[i];
    option.x2c_chRate = SFileMenuOption::ComputeRandom();
    option.x28_curField = -1;
    for (auto& panePair : option.x4_textpanes) {
      panePair.SetPairText(u"");
    }
  }

  StartTextAnimating(x28_textpane_erase.x0_panes[0], g_MainStringTable->GetString(38), 60.f);
  StartTextAnimating(x38_textpane_gba.x0_panes[0], g_MainStringTable->GetString(37), 60.f);
  StartTextAnimating(x30_textpane_cheats.x0_panes[0], g_MainStringTable->GetString(96), 60.f);

  StartTextAnimating(x28_textpane_erase.x0_panes[1], g_MainStringTable->GetString(38), 60.f);
  StartTextAnimating(x38_textpane_gba.x0_panes[1], g_MainStringTable->GetString(37), 60.f);
  StartTextAnimating(x30_textpane_cheats.x0_panes[1], g_MainStringTable->GetString(96), 60.f);

  if (hasSave) {
    x28_textpane_erase.x0_panes[0]->SetIsSelectable(true);
    x28_textpane_erase.x0_panes[0]->TextSupport().SetFontColor(zeus::skWhite);
  } else {
    x28_textpane_erase.x0_panes[0]->SetIsSelectable(false);
    zeus::CColor color = zeus::skGrey;
    color.a() = 0.5f;
    x28_textpane_erase.x0_panes[0]->TextSupport().SetFontColor(color);
  }

  x20_tablegroup_fileselect->SetUserSelection(0);
  CGuiTextPane* cheats = static_cast<CGuiTextPane*>(x20_tablegroup_fileselect->GetWorkerWidget(5));
  if (CSlideShow::SlideShowGalleryFlags()) {
    cheats->SetIsSelectable(true);
    x30_textpane_cheats.x0_panes[0]->TextSupport().SetFontColor(zeus::skWhite);
  } else {
    cheats->SetIsSelectable(false);
    zeus::CColor color = zeus::skGrey;
    color.a() = 0.5f;
    x30_textpane_cheats.x0_panes[0]->TextSupport().SetFontColor(color);
  }

  HandleActiveChange(x20_tablegroup_fileselect);
}

void CFrontEndUI::SNewFileSelectFrame::SetupFrameContents() {
  for (size_t i = 0; i < x64_fileSelections.size(); ++i) {
    SFileMenuOption& option = x64_fileSelections[i];
    if (option.x28_curField == 4) {
      continue;
    }

    SGuiTextPair* pair = (option.x28_curField == UINT32_MAX) ? nullptr : &option.x4_textpanes[option.x28_curField];
    if (!pair || pair->x0_panes[0]->GetTextSupport().GetNumCharsPrinted() >=
                     pair->x0_panes[0]->GetTextSupport().GetNumCharsTotal()) {
      if (++option.x28_curField < 4) {
        std::u16string str;
        SGuiTextPair& populatePair = option.x4_textpanes[option.x28_curField];
        const CGameState::GameFileStateInfo* data = x4_saveUI->GetGameData(int(i));

        switch (option.x28_curField) {
        case 0:
          // Completion percent
          if (data) {
            std::u16string fileStr = g_MainStringTable->GetString((data->x20_hardMode ? 106 : 39) + int(i));
            str = fileStr + fmt::format(fmt(u"  {:02d}%"), data->x18_itemPercent);
            break;
          }
          str = g_MainStringTable->GetString(36);
          break;

        case 1:
          // World name
          if (data) {
            if (g_MemoryCardSys->HasSaveWorldMemory(data->x8_mlvlId)) {
              const CSaveWorldMemory& wldMem = g_MemoryCardSys->GetSaveWorldMemory(data->x8_mlvlId);
              str = wldMem.GetFrontEndName();
            }
            break;
          }
          str = g_MainStringTable->GetString(51);
          break;
#if FE_USE_SECONDS_IN_ELAPSED
          case 2:
          // Formatted time
          if (data) {
            auto pt = std::div(data->x0_playTime, 3600);
            str = fmt::format(fmt(u"{:02d}:{:02d}:{:02d}"), pt.quot, pt.rem / 60, pt.rem % 60);
            break;
          }
          str = g_MainStringTable->GetString(52);
          break;
        case 3:
          // "Elapsed"
          str = std::u16string(u"    ") + std::u16string(g_MainStringTable->GetString(data ? 54 : 53));
          break;
#else
        case 2:
          // Formatted time
          if (data) {
            auto pt = std::div(data->x0_playTime, 3600);
            str = hecl::Char16Format(L"%02d:%02d", pt.quot, pt.rem / 60);
            break;
          }
          str = g_MainStringTable->GetString(52);
          break;

        case 3:
          // "Elapsed"
          str = g_MainStringTable->GetString(data ? 54 : 53);
          break;
#endif

        default:
          break;
        }

        StartTextAnimating(populatePair.x0_panes[0], str, option.x2c_chRate);
        StartTextAnimating(populatePair.x0_panes[1], str, option.x2c_chRate);
      }
    }
  }
}

void CFrontEndUI::SNewFileSelectFrame::DoPopupCancel(CGuiTableGroup* caller) {
  if (x8_subMenu == ESubMenu::EraseGamePopup) {
    CSfxManager::SfxStart(SFXfnt_back, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
    x8_subMenu = ESubMenu::EraseGame;
    x10d_needsEraseToggle = true;
  } else {
    CSfxManager::SfxStart(SFXfnt_back, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
    x8_subMenu = ESubMenu::Root;
    x10e_needsNewToggle = true;
  }
}

void CFrontEndUI::SNewFileSelectFrame::DoPopupAdvance(CGuiTableGroup* caller) {
  if (x8_subMenu == ESubMenu::EraseGamePopup) {
    if (x40_tablegroup_popup->GetUserSelection() == 1) {
      x4_saveUI->EraseGame(x20_tablegroup_fileselect->GetUserSelection());
      ResetFrame();
    } else
      x8_subMenu = ESubMenu::EraseGame;
    x10d_needsEraseToggle = true;
  } else {
    if (g_GameState->SystemOptions().GetPlayerBeatNormalMode()) {
      if (x40_tablegroup_popup->GetUserSelection() == 1) {
        PlayAdvanceSfx();
        xc_action = EAction::GameOptions;
        return;
      }
      g_GameState->SetHardMode(!x40_tablegroup_popup->GetUserSelection());
      x4_saveUI->StartGame(x20_tablegroup_fileselect->GetUserSelection());
    } else {
      if (x40_tablegroup_popup->GetUserSelection() == 1) {
        PlayAdvanceSfx();
        xc_action = EAction::GameOptions;
        return;
      }
      x4_saveUI->StartGame(x20_tablegroup_fileselect->GetUserSelection());
    }
  }
}

void CFrontEndUI::SNewFileSelectFrame::DoFileMenuCancel(CGuiTableGroup* caller) {
  if (x8_subMenu == ESubMenu::EraseGame) {
    CSfxManager::SfxStart(SFXfnt_back, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
    ResetFrame();
  }
}

void CFrontEndUI::SNewFileSelectFrame::DoSelectionChange(CGuiTableGroup* caller, int oldSel) {
  HandleActiveChange(caller);
  CSfxManager::SfxStart(SFXfnt_selection_change, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
}

void CFrontEndUI::SNewFileSelectFrame::DoFileMenuAdvance(CGuiTableGroup* caller) {
  int userSel = x20_tablegroup_fileselect->GetUserSelection();
  if (userSel < 3) {
    if (x8_subMenu == ESubMenu::EraseGame) {
      if (x4_saveUI->GetGameData(userSel)) {
        PlayAdvanceSfx();
        x10d_needsEraseToggle = true;
      }
    } else {
      if (x4_saveUI->GetGameData(userSel)) {
        m_touchBar.SetPhase(CFrontEndUITouchBar::EPhase::None);
        x4_saveUI->StartGame(userSel);
      } else
        x10e_needsNewToggle = true;
    }
  } else if (userSel == 3) {
    PlayAdvanceSfx();
    ActivateErase();
  } else if (userSel == 4) {
    xc_action = EAction::FusionBonus;
  } else if (userSel == 5) {
    xc_action = EAction::SlideShow;
  }
}

CFrontEndUI::SFileMenuOption CFrontEndUI::SNewFileSelectFrame::FindFileSelectOption(CGuiFrame* frame, int idx) {
  SFileMenuOption ret;
  ret.x0_base = frame->FindWidget(fmt::format(fmt("basewidget_file{}"), idx));
  ret.x4_textpanes[0] = FindTextPanePair(frame, fmt::format(fmt("textpane_filename{}"), idx));
  ret.x4_textpanes[1] = FindTextPanePair(frame, fmt::format(fmt("textpane_world{}"), idx));
  ret.x4_textpanes[2] = FindTextPanePair(frame, fmt::format(fmt("textpane_playtime{}"), idx));
  ret.x4_textpanes[3] = FindTextPanePair(frame, fmt::format(fmt("textpane_date{}"), idx));
  return ret;
}

void CFrontEndUI::SNewFileSelectFrame::StartTextAnimating(CGuiTextPane* text, std::u16string_view str, float chRate) {
  text->TextSupport().SetText(str);
  text->TextSupport().SetTypeWriteEffectOptions(true, 0.1f, chRate);
}

CFrontEndUI::SFusionBonusFrame::SFusionBonusFrame(CFrontEndUITouchBar& touchBar) : m_touchBar(touchBar) {
  x4_gbaSupport = std::make_unique<CGBASupport>();
  xc_gbaScreen = g_SimplePool->GetObj("FRME_GBAScreen");
  x18_gbaLink = g_SimplePool->GetObj("FRME_GBALink");
}

void CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::SetUIText(EUIType tp) {
  int instructions = -1;
  int yes = -1;
  int no = -1;

  bool cableVisible = false;
  bool circleGcVisible = false;
  bool circleGbaVisible = false;
  bool circleStartVisible = false;
  bool pakoutVisible = false;
  bool gbaScreenVisible = false;
  bool connectVisible = false;

  switch (tp) {
  case EUIType::InsertPak:
    instructions = 73; // Insert Game Pak
    no = 82;
    yes = 83;
    pakoutVisible = true;
    circleGbaVisible = true;
    break;
  case EUIType::ConnectSocket:
    instructions = 68; // Connect socket
    no = 82;
    yes = 83;
    cableVisible = true;
    circleGcVisible = true;
    circleGbaVisible = true;
    break;
  case EUIType::PressStartAndSelect:
    instructions = 74; // Hold start and select
    no = 82;
    yes = 83;
    cableVisible = true;
    circleStartVisible = true;
    gbaScreenVisible = true;
    break;
  case EUIType::BeginLink:
    instructions = 75; // Begin link?
    no = 82;
    yes = 83;
    cableVisible = true;
    gbaScreenVisible = true;
    break;
  case EUIType::TurnOffGBA:
    instructions = 76; // Turn off GBA
    no = 82;
    yes = 83;
    cableVisible = true;
    gbaScreenVisible = true;
    circleStartVisible = true;
    break;
  case EUIType::Linking:
    x4_gbaSupport->StartLink();
    instructions = 72; // Linking
    cableVisible = true;
    gbaScreenVisible = true;
    connectVisible = true;
    break;
  case EUIType::LinkFailed:
    instructions = 69; // Link failed
    no = 82;
    yes = 83;
    cableVisible = true;
    circleGcVisible = true;
    circleGbaVisible = true;
    circleStartVisible = true;
    gbaScreenVisible = true;
    break;
  case EUIType::LinkCompleteOrLinking:
    yes = 83;
    instructions = x40_linkInProgress + 71; // Complete or linking
    cableVisible = true;
    gbaScreenVisible = true;
    break;
  case EUIType::Complete:
  case EUIType::Cancelled:
  default:
    break;
  }

  std::u16string instructionsStr;
  if (instructions != -1)
    instructionsStr = g_MainStringTable->GetString(instructions);
  xc_textpane_instructions.SetPairText(instructionsStr);

  std::u16string yesStr;
  if (yes != -1)
    yesStr = g_MainStringTable->GetString(yes);
  x14_textpane_yes->TextSupport().SetText(yesStr);

  std::u16string noStr;
  if (no != -1)
    noStr = g_MainStringTable->GetString(no);
  x18_textpane_no->TextSupport().SetText(noStr);

  x1c_model_gc->SetVisibility(true, ETraversalMode::Children);
  x20_model_gba->SetVisibility(true, ETraversalMode::Children);
  x24_model_cable->SetVisibility(cableVisible, ETraversalMode::Children);
  x28_model_circlegcport->SetVisibility(circleGcVisible, ETraversalMode::Children);
  x2c_model_circlegbaport->SetVisibility(circleGbaVisible, ETraversalMode::Children);
  x30_model_circlestartselect->SetVisibility(circleStartVisible, ETraversalMode::Children);
  x34_model_pakout->SetVisibility(pakoutVisible, ETraversalMode::Children);
  x38_model_gbascreen->SetVisibility(gbaScreenVisible, ETraversalMode::Children);
  x3c_model_connect->SetVisibility(connectVisible, ETraversalMode::Children);

  x0_uiType = tp;
}

CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::EAction CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::ProcessUserInput(
    const CFinalInput& input, bool linkInProgress, CFrontEndUITouchBar::EAction tbAction) {
  if (linkInProgress != x40_linkInProgress) {
    x40_linkInProgress = linkInProgress;
    SetUIText(x0_uiType);
  }

  switch (x0_uiType) {
  case EUIType::InsertPak:
  case EUIType::ConnectSocket:
  case EUIType::PressStartAndSelect:
  case EUIType::BeginLink:
  case EUIType::LinkFailed:
  case EUIType::LinkCompleteOrLinking:
  case EUIType::TurnOffGBA:
    if (input.PA() ||
        input.PSpecialKey(boo::ESpecialKey::Enter) ||
        input.PMouseButton(boo::EMouseButton::Primary) ||
        tbAction == CFrontEndUITouchBar::EAction::Confirm) {
      PlayAdvanceSfx();
      SetUIText(NextLinkUI[size_t(x0_uiType)]);
    } else if (input.PB() || input.PSpecialKey(boo::ESpecialKey::Esc) ||
               tbAction == CFrontEndUITouchBar::EAction::Back) {
      const EUIType prevUi = PrevLinkUI[size_t(x0_uiType)];
      if (prevUi == EUIType::Empty) {
        break;
      }
      CSfxManager::SfxStart(SFXfnt_back, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
      SetUIText(prevUi);
    }
    break;
  case EUIType::Linking:
    if (x4_gbaSupport->GetPhase() == CGBASupport::EPhase::Complete) {
      if (x4_gbaSupport->IsFusionLinked())
        g_GameState->SystemOptions().SetPlayerLinkedFusion(true);
      if (x4_gbaSupport->IsFusionBeat())
        g_GameState->SystemOptions().SetPlayerBeatFusion(true);
      if (x4_gbaSupport->IsFusionLinked()) {
        PlayAdvanceSfx();
        SetUIText(EUIType::LinkCompleteOrLinking);
      } else {
        CSfxManager::SfxStart(SFXfnt_back, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
        SetUIText(EUIType::LinkFailed);
      }
    } else if (x4_gbaSupport->GetPhase() == CGBASupport::EPhase::Failed) {
      CSfxManager::SfxStart(SFXfnt_back, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
      SetUIText(EUIType::LinkFailed);
    }
    break;
  case EUIType::Complete:
    return EAction::Complete;
  case EUIType::Cancelled:
    return EAction::Cancelled;
  default:
    break;
  }

  return EAction::None;
}

void CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::Update(float dt) {
  x4_gbaSupport->Update(dt);
  x8_frme->Update(dt);
}

void CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::FinishedLoading() {
  x8_frme->SetAspectConstraint(1.78f);

  xc_textpane_instructions = FindTextPanePair(x8_frme, "textpane_instructions");
  x14_textpane_yes = static_cast<CGuiTextPane*>(x8_frme->FindWidget("textpane_yes"));
  x18_textpane_no = static_cast<CGuiTextPane*>(x8_frme->FindWidget("textpane_no"));
  x1c_model_gc = static_cast<CGuiModel*>(x8_frme->FindWidget("model_gc"));
  x20_model_gba = static_cast<CGuiModel*>(x8_frme->FindWidget("model_gba"));
  x24_model_cable = static_cast<CGuiModel*>(x8_frme->FindWidget("model_cable"));
  x28_model_circlegcport = static_cast<CGuiModel*>(x8_frme->FindWidget("model_circlegcport"));
  x2c_model_circlegbaport = static_cast<CGuiModel*>(x8_frme->FindWidget("model_circlegbaport"));
  x30_model_circlestartselect = static_cast<CGuiModel*>(x8_frme->FindWidget("model_circlestartselect"));
  x34_model_pakout = static_cast<CGuiModel*>(x8_frme->FindWidget("model_pakout"));
  x38_model_gbascreen = static_cast<CGuiModel*>(x8_frme->FindWidget("model_gbascreen"));
  x3c_model_connect = static_cast<CGuiModel*>(x8_frme->FindWidget("model_connect"));
  SetUIText(EUIType::InsertPak);
}

void CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::Draw() { x8_frme->Draw(CGuiWidgetDrawParms::Default); }

CFrontEndUI::SFusionBonusFrame::SGBALinkFrame::SGBALinkFrame(CGuiFrame* linkFrame, CGBASupport* support,
                                                             bool linkInProgress)
: x4_gbaSupport(support), x8_frme(linkFrame), x40_linkInProgress(linkInProgress) {
  support->InitializeSupport();
  FinishedLoading();
}

void CFrontEndUI::SFusionBonusFrame::FinishedLoading() {
  x24_loadedFrame->SetAspectConstraint(1.78f);

  x28_tablegroup_options = static_cast<CGuiTableGroup*>(x24_loadedFrame->FindWidget("tablegroup_options"));
  x2c_tablegroup_fusionsuit = static_cast<CGuiTableGroup*>(x24_loadedFrame->FindWidget("tablegroup_fusionsuit"));
  x30_textpane_instructions = FindTextPanePair(x24_loadedFrame, "textpane_instructions");

  FindAndSetPairText(x24_loadedFrame, "textpane_nes", g_MainStringTable->GetString(66));
  FindAndSetPairText(x24_loadedFrame, "textpane_fusionsuit", g_MainStringTable->GetString(63));
  FindAndSetPairText(x24_loadedFrame, "textpane_fusionsuitno", g_MainStringTable->GetString(65));
  FindAndSetPairText(x24_loadedFrame, "textpane_fusionsuityes", g_MainStringTable->GetString(64));
  FindAndSetPairText(x24_loadedFrame, "textpane_title", g_MainStringTable->GetString(100));

  static_cast<CGuiTextPane*>(x24_loadedFrame->FindWidget("textpane_proceed"))
      ->TextSupport()
      .SetText(g_MainStringTable->GetString(85));
  static_cast<CGuiTextPane*>(x24_loadedFrame->FindWidget("textpane_cancel"))
      ->TextSupport()
      .SetText(g_MainStringTable->GetString(82));

  x2c_tablegroup_fusionsuit->SetIsActive(false);
  x2c_tablegroup_fusionsuit->SetIsVisible(false);
  x2c_tablegroup_fusionsuit->SetVertical(false);
  x2c_tablegroup_fusionsuit->SetUserSelection(g_GameState->SystemOptions().GetPlayerFusionSuitActive());

  SetTableColors(x28_tablegroup_options);
  SetTableColors(x2c_tablegroup_fusionsuit);

  x28_tablegroup_options->SetMenuAdvanceCallback([this](CGuiTableGroup* caller) { DoAdvance(caller); });
  x28_tablegroup_options->SetMenuSelectionChangeCallback(
      [this](CGuiTableGroup* caller, int oldSel) { DoSelectionChange(caller, oldSel); });
  x28_tablegroup_options->SetMenuCancelCallback([this](CGuiTableGroup* caller) { DoCancel(caller); });
  x2c_tablegroup_fusionsuit->SetMenuSelectionChangeCallback(
      [this](CGuiTableGroup* caller, int oldSel) { DoSelectionChange(caller, oldSel); });
}

bool CFrontEndUI::SFusionBonusFrame::PumpLoad() {
  if (x24_loadedFrame)
    return true;
  if (!xc_gbaScreen.IsLoaded())
    return false;
  if (!x18_gbaLink.IsLoaded())
    return false;
  if (!x4_gbaSupport->IsReady())
    return false;
  if (!xc_gbaScreen->GetIsFinishedLoading())
    return false;
  x24_loadedFrame = xc_gbaScreen.GetObj();
  FinishedLoading();
  return true;
}

void CFrontEndUI::SFusionBonusFrame::SetTableColors(CGuiTableGroup* tbgp) const {
  tbgp->SetColors(zeus::skWhite, zeus::CColor{0.627450f, 0.627450f, 0.627450f, 0.784313f});
}

void CFrontEndUI::SFusionBonusFrame::Update(float dt, CSaveGameScreen* saveUI) {
  bool doDraw = !saveUI || saveUI->GetUIType() == CSaveGameScreen::EUIType::SaveReady;

  if (doDraw != x38_lastDoDraw) {
    x38_lastDoDraw = doDraw;
    ResetCompletionFlags();
  }

  if (x0_gbaLinkFrame)
    x0_gbaLinkFrame->Update(dt);
  else if (x24_loadedFrame)
    x24_loadedFrame->Update(dt);

  bool showFusionSuit = (g_GameState->SystemOptions().GetPlayerLinkedFusion() &&
                         g_GameState->SystemOptions().GetPlayerBeatNormalMode()) ||
                        m_gbaOverride;
  bool showFusionSuitProceed = showFusionSuit && x28_tablegroup_options->GetUserSelection() == 1;
  x2c_tablegroup_fusionsuit->SetIsActive(showFusionSuitProceed);
  x2c_tablegroup_fusionsuit->SetIsVisible(showFusionSuitProceed);
  x24_loadedFrame->FindWidget("textpane_proceed")->SetIsVisible(showFusionSuitProceed);

  std::u16string instructionStr;
  x30_textpane_instructions.x0_panes[0]->TextSupport().SetFontColor(zeus::skWhite);
  if (x28_tablegroup_options->GetUserSelection() == 1) {
    /* Fusion Suit */
    if (x3a_mpNotComplete)
      instructionStr = g_MainStringTable->GetString(80); // MP not complete
    else if (!showFusionSuit)
      instructionStr = g_MainStringTable->GetString(78); // To enable fusion suit
  } else {
    /* NES Metroid */
    if (x39_fusionNotComplete)
      instructionStr = g_MainStringTable->GetString(79); // You have not completed fusion
    else if (!g_GameState->SystemOptions().GetPlayerBeatFusion())
      instructionStr = g_MainStringTable->GetString(77); // To play NES Metroid
  }

  x30_textpane_instructions.SetPairText(instructionStr);
}

CFrontEndUI::SFusionBonusFrame::EAction CFrontEndUI::SFusionBonusFrame::ProcessUserInput(
    const CFinalInput& input, CSaveGameScreen* sui, CFrontEndUITouchBar::EAction tbAction) {
  x8_action = EAction::None;

  if (sui)
    sui->ProcessUserInput(input);

  if (x38_lastDoDraw) {
    if (x0_gbaLinkFrame) {
      if (m_touchBar.GetPhase() != CFrontEndUITouchBar::EPhase::ProceedBack)
        m_touchBar.SetPhase(CFrontEndUITouchBar::EPhase::ProceedBack);

      SGBALinkFrame::EAction action = x0_gbaLinkFrame->ProcessUserInput(input, sui, tbAction);
      if (action != SGBALinkFrame::EAction::None) {
        x0_gbaLinkFrame.reset();
        if (action == SGBALinkFrame::EAction::Complete) {
          if (x28_tablegroup_options->GetUserSelection() == 0 && !g_GameState->SystemOptions().GetPlayerBeatFusion())
            x39_fusionNotComplete = true;
          else if (sui)
            sui->SaveNESState();
        }
      }
    } else if (x24_loadedFrame) {
      CFinalInput useInput = input;
      if (input.PZ() || input.PKey('\t')) {
        useInput.x2d_b28_PA = true;
        m_gbaOverride = true;
      }

      bool showFusionSuit = (g_GameState->SystemOptions().GetPlayerLinkedFusion() &&
                             g_GameState->SystemOptions().GetPlayerBeatNormalMode()) ||
                            m_gbaOverride;
      if (m_touchBar.GetPhase() != CFrontEndUITouchBar::EPhase::FusionBonus) {
        m_touchBar.SetFusionBonusPhase(showFusionSuit && g_GameState->SystemOptions().GetPlayerFusionSuitActive());
      }

      x24_loadedFrame->ProcessUserInput(useInput);

      switch (tbAction) {
      case CFrontEndUITouchBar::EAction::NESMetroid:
        x28_tablegroup_options->SetUserSelection(0);
        ResetCompletionFlags();
        SetTableColors(x28_tablegroup_options);
        DoAdvance(x28_tablegroup_options);
        break;
      case CFrontEndUITouchBar::EAction::FusionSuit:
        x28_tablegroup_options->SetUserSelection(1);
        ResetCompletionFlags();
        SetTableColors(x28_tablegroup_options);
        if (showFusionSuit) {
          if (x2c_tablegroup_fusionsuit->GetUserSelection() == 1) {
            x2c_tablegroup_fusionsuit->SetUserSelection(0);
            DoSelectionChange(x2c_tablegroup_fusionsuit, 0);
          } else {
            x2c_tablegroup_fusionsuit->SetUserSelection(1);
            DoSelectionChange(x2c_tablegroup_fusionsuit, 1);
          }
        } else {
          DoAdvance(x28_tablegroup_options);
        }
        break;
      case CFrontEndUITouchBar::EAction::Back:
        DoCancel(x28_tablegroup_options);
        break;
      default:
        break;
      }
    }
  }

  return x8_action;
}

void CFrontEndUI::SFusionBonusFrame::Draw() const {
  if (!x38_lastDoDraw)
    return;
  if (x0_gbaLinkFrame)
    x0_gbaLinkFrame->Draw();
  else if (x24_loadedFrame)
    x24_loadedFrame->Draw(CGuiWidgetDrawParms::Default);
}

void CFrontEndUI::SFusionBonusFrame::DoCancel(CGuiTableGroup* caller) {
  x8_action = EAction::GoBack;
  x28_tablegroup_options->SetUserSelection(0);
  x2c_tablegroup_fusionsuit->SetIsActive(false);
  x30_textpane_instructions.SetPairText(u"");
  SetTableColors(x28_tablegroup_options);
}

void CFrontEndUI::SFusionBonusFrame::DoSelectionChange(CGuiTableGroup* caller, int oldSel) {
  if (caller == x28_tablegroup_options) {
    CSfxManager::SfxStart(SFXfnt_selection_change, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
    ResetCompletionFlags();
  } else {
    CSfxManager::SfxStart(SFXfnt_enum_change, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
    bool fusionActive = x2c_tablegroup_fusionsuit->GetUserSelection() == 1;
    g_GameState->SystemOptions().SetPlayerFusionSuitActive(fusionActive);
    g_GameState->GetPlayerState()->SetIsFusionEnabled(fusionActive);
    m_touchBar.SetFusionBonusPhase(g_GameState->SystemOptions().GetPlayerFusionSuitActive());
  }
  SetTableColors(caller);
}

void CFrontEndUI::SFusionBonusFrame::DoAdvance(CGuiTableGroup* caller) {
  switch (x28_tablegroup_options->GetUserSelection()) {
  case 1:
    /* Fusion Suit */
    if (x3a_mpNotComplete || m_gbaOverride) {
      x3a_mpNotComplete = false;
      PlayAdvanceSfx();
    } else if (g_GameState->SystemOptions().GetPlayerBeatNormalMode()) {
      if (g_GameState->SystemOptions().GetPlayerLinkedFusion())
        return;
      x0_gbaLinkFrame = std::make_unique<SGBALinkFrame>(x18_gbaLink.GetObj(), x4_gbaSupport.get(), false);
      PlayAdvanceSfx();
    } else {
      x3a_mpNotComplete = true;
      CSfxManager::SfxStart(SFXfnt_back, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
    }
    break;
  case 0:
    /* NES Metroid */
    if (x39_fusionNotComplete && !m_gbaOverride) {
      x39_fusionNotComplete = false;
      PlayAdvanceSfx();
    } else if (g_GameState->SystemOptions().GetPlayerBeatFusion() || m_gbaOverride) {
      // x8_action = EAction::None;
      CSfxManager::SfxStart(SFXfnt_back, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
      x8_action = EAction::PlayNESMetroid;
    } else {
      x0_gbaLinkFrame = std::make_unique<SGBALinkFrame>(x18_gbaLink.GetObj(), x4_gbaSupport.get(), false);
      PlayAdvanceSfx();
    }
    break;
  default:
    break;
  }
}

void CFrontEndUI::SGuiTextPair::SetPairText(std::u16string_view str) {
  x0_panes[0]->TextSupport().SetText(str);
  x0_panes[1]->TextSupport().SetText(str);
}

CFrontEndUI::SGuiTextPair CFrontEndUI::FindTextPanePair(CGuiFrame* frame, std::string_view name) {
  SGuiTextPair ret;
  ret.x0_panes[0] = static_cast<CGuiTextPane*>(frame->FindWidget(name));
  ret.x0_panes[1] = static_cast<CGuiTextPane*>(frame->FindWidget(fmt::format(fmt("{}b"), name)));
  return ret;
}

void CFrontEndUI::FindAndSetPairText(CGuiFrame* frame, std::string_view name, std::u16string_view str) {
  CGuiTextPane* w1 = static_cast<CGuiTextPane*>(frame->FindWidget(name));
  w1->TextSupport().SetText(str);
  CGuiTextPane* w2 = static_cast<CGuiTextPane*>(frame->FindWidget(fmt::format(fmt("{}b"), name)));
  w2->TextSupport().SetText(str);
}

void CFrontEndUI::SFrontEndFrame::FinishedLoading() {
  x14_loadedFrme->SetAspectConstraint(1.78f);

  x18_tablegroup_mainmenu = static_cast<CGuiTableGroup*>(x14_loadedFrme->FindWidget("tablegroup_mainmenu"));
  x1c_gbaPair = FindTextPanePair(x14_loadedFrme, "textpane_gba");
  x1c_gbaPair.SetPairText(g_MainStringTable->GetString(37));
  x24_cheatPair = FindTextPanePair(x14_loadedFrme, "textpane_cheats");
  x24_cheatPair.SetPairText(g_MainStringTable->GetString(96));

  FindAndSetPairText(x14_loadedFrme, "textpane_start", g_MainStringTable->GetString(67));
  FindAndSetPairText(x14_loadedFrme, "textpane_options", g_MainStringTable->GetString(94));
  FindAndSetPairText(x14_loadedFrme, "textpane_title", g_MainStringTable->GetString(98));

  CGuiTextPane* proceed = static_cast<CGuiTextPane*>(x14_loadedFrme->FindWidget("textpane_proceed"));
  if (proceed)
    proceed->TextSupport().SetText(g_MainStringTable->GetString(85));

  x18_tablegroup_mainmenu->SetMenuAdvanceCallback([this](CGuiTableGroup* caller) { DoAdvance(caller); });
  x18_tablegroup_mainmenu->SetMenuSelectionChangeCallback(
      [this](CGuiTableGroup* caller, int oldSel) { DoSelectionChange(caller, oldSel); });
  x18_tablegroup_mainmenu->SetMenuCancelCallback([this](CGuiTableGroup* caller) { DoCancel(caller); });

  HandleActiveChange(x18_tablegroup_mainmenu);
}

bool CFrontEndUI::SFrontEndFrame::PumpLoad() {
  if (x14_loadedFrme)
    return true;
  if (x8_frme.IsLoaded()) {
    if (CGuiFrame* frme = x8_frme.GetObj()) {
      if (frme->GetIsFinishedLoading()) {
        x14_loadedFrme = frme;
        FinishedLoading();
        return true;
      }
    }
  }
  return false;
}

void CFrontEndUI::SFrontEndFrame::Update(float dt) {
  CGuiTextPane* imageGallery = static_cast<CGuiTextPane*>(x18_tablegroup_mainmenu->GetWorkerWidget(3));

  if (CSlideShow::SlideShowGalleryFlags()) {
    imageGallery->SetIsSelectable(true);
    x24_cheatPair.x0_panes[0]->TextSupport().SetFontColor(zeus::skWhite);
  } else {
    imageGallery->SetIsSelectable(false);
    zeus::CColor color = zeus::skGrey;
    color.a() = 0.5f;
    x24_cheatPair.x0_panes[0]->TextSupport().SetFontColor(color);
  }

  x14_loadedFrme->Update(dt);
}

CFrontEndUI::SFrontEndFrame::EAction
CFrontEndUI::SFrontEndFrame::ProcessUserInput(const CFinalInput& input, CFrontEndUITouchBar::EAction tbAction) {
  if (m_touchBar.GetPhase() != CFrontEndUITouchBar::EPhase::NoCardSelect)
    m_touchBar.SetNoCardSelectPhase(CSlideShow::SlideShowGalleryFlags());

  x4_action = EAction::None;
  x14_loadedFrme->ProcessUserInput(input);

  switch (tbAction) {
  case CFrontEndUITouchBar::EAction::Start:
    x18_tablegroup_mainmenu->SetUserSelection(0);
    HandleActiveChange(x18_tablegroup_mainmenu);
    DoAdvance(x18_tablegroup_mainmenu);
    break;
  case CFrontEndUITouchBar::EAction::FusionBonus:
    x18_tablegroup_mainmenu->SetUserSelection(1);
    HandleActiveChange(x18_tablegroup_mainmenu);
    DoAdvance(x18_tablegroup_mainmenu);
    break;
  case CFrontEndUITouchBar::EAction::Options:
    x18_tablegroup_mainmenu->SetUserSelection(2);
    HandleActiveChange(x18_tablegroup_mainmenu);
    DoAdvance(x18_tablegroup_mainmenu);
    break;
  case CFrontEndUITouchBar::EAction::ImageGallery:
    x18_tablegroup_mainmenu->SetUserSelection(3);
    HandleActiveChange(x18_tablegroup_mainmenu);
    DoAdvance(x18_tablegroup_mainmenu);
    break;
  default:
    break;
  }

  return x4_action;
}

void CFrontEndUI::SFrontEndFrame::Draw() const { x14_loadedFrme->Draw(CGuiWidgetDrawParms::Default); }

void CFrontEndUI::SFrontEndFrame::HandleActiveChange(CGuiTableGroup* active) {
  active->SetColors(zeus::skWhite, zeus::CColor{0.627450f, 0.627450f, 0.627450f, 0.784313f});
}

void CFrontEndUI::SFrontEndFrame::DoCancel(CGuiTableGroup* caller) { /* Intentionally empty */ }

void CFrontEndUI::SFrontEndFrame::DoSelectionChange(CGuiTableGroup* caller, int oldSel) {
  CSfxManager::SfxStart(SFXfnt_selection_change, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
  HandleActiveChange(caller);
}

void CFrontEndUI::SFrontEndFrame::DoAdvance(CGuiTableGroup* caller) {
  switch (x18_tablegroup_mainmenu->GetUserSelection()) {
  case 0:
    CSfxManager::SfxStart(FETransitionForwardSFX[x0_rnd][0], 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
    CSfxManager::SfxStart(FETransitionForwardSFX[x0_rnd][1], 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
    x4_action = EAction::StartGame;
    break;
  case 1:
    x4_action = EAction::FusionBonus;
    break;
  case 2:
    PlayAdvanceSfx();
    x4_action = EAction::GameOptions;
    break;
  case 3:
    PlayAdvanceSfx();
    x4_action = EAction::SlideShow;
    break;
  default:
    break;
  }
}

CFrontEndUI::SFrontEndFrame::SFrontEndFrame(u32 rnd, CFrontEndUITouchBar& touchBar)
: x0_rnd(rnd), m_touchBar(touchBar) {
  x8_frme = g_SimplePool->GetObj("FRME_FrontEndPL");
}

CFrontEndUI::SNesEmulatorFrame::SNesEmulatorFrame() {
  x4_nesEmu = std::make_unique<CNESEmulator>();

  const SObjectTag* deface = g_ResFactory->GetResourceIdByName("FONT_Deface14B");
  CGuiTextProperties props(false, true, EJustification::Left, EVerticalJustification::Center);
  xc_textSupport = std::make_unique<CGuiTextSupport>(deface->id, props, zeus::skWhite, zeus::skBlack,
                                                     zeus::skWhite, 0, 0, g_SimplePool,
                                                     CGuiWidget::EGuiModelDrawFlags::Alpha);
  xc_textSupport->SetText(g_MainStringTable->GetString(103));
  xc_textSupport->AutoSetExtent();
  xc_textSupport->ClearRenderBuffer();
}

void CFrontEndUI::SNesEmulatorFrame::SetMode(EMode mode) {
  switch (mode) {
  case EMode::Emulator:
    x8_quitScreen.reset();
    break;
  case EMode::SaveProgress:
    x8_quitScreen = std::make_unique<CQuitGameScreen>(EQuitType::SaveProgress);
    break;
  case EMode::ContinuePlaying:
    x8_quitScreen = std::make_unique<CQuitGameScreen>(EQuitType::ContinuePlaying);
    break;
  case EMode::QuitNESMetroid:
    x8_quitScreen = std::make_unique<CQuitGameScreen>(EQuitType::QuitNESMetroid);
    break;
  default:
    break;
  }
  x0_mode = mode;
}

void CFrontEndUI::SNesEmulatorFrame::ProcessUserInput(const CFinalInput& input, CSaveGameScreen* sui) {
  bool processInput = true;
  if (sui && sui->GetUIType() != CSaveGameScreen::EUIType::SaveReady)
    processInput = false;
  if (sui)
    sui->ProcessUserInput(input);
  if (!processInput)
    return;

  switch (x0_mode) {
  case EMode::Emulator:
    x4_nesEmu->ProcessUserInput(input, 4);
    if ((input.ControllerIdx() == 0 && input.PL()) || input.PSpecialKey(boo::ESpecialKey::Esc))
      SetMode(EMode::QuitNESMetroid);
    break;
  case EMode::SaveProgress:
  case EMode::ContinuePlaying:
  case EMode::QuitNESMetroid:
    x8_quitScreen->ProcessUserInput(input);
    break;
  default:
    break;
  }
}

bool CFrontEndUI::SNesEmulatorFrame::Update(float dt, CSaveGameScreen* saveUi) {
  bool doUpdate = (saveUi && saveUi->GetUIType() != CSaveGameScreen::EUIType::SaveReady) ? false : true;
  x10_remTime = std::max(x10_remTime - dt, 0.f);

  zeus::CColor geomCol(zeus::skWhite);
  geomCol.a() = std::min(x10_remTime, 1.f);
  xc_textSupport->SetGeometryColor(geomCol);
  if (xc_textSupport->GetIsTextSupportFinishedLoading()) {
    xc_textSupport->AutoSetExtent();
    xc_textSupport->ClearRenderBuffer();
  }

  if (!doUpdate)
    return false;

  switch (x0_mode) {
  case EMode::Emulator: {
    x4_nesEmu->Update();
    if (!x4_nesEmu->IsGameOver())
      x14_emulationSuspended = false;
    if (x4_nesEmu->IsGameOver() && !x14_emulationSuspended) {
      x14_emulationSuspended = true;
      if (saveUi && !saveUi->IsSavingDisabled()) {
        SetMode(EMode::SaveProgress);
        break;
      }
      SetMode(EMode::ContinuePlaying);
      break;
    }
    if (x4_nesEmu->GetPasswordEntryState() == CNESEmulator::EPasswordEntryState::NotEntered && saveUi)
      x4_nesEmu->LoadPassword(g_GameState->SystemOptions().GetNESState());
    break;
  }

  case EMode::SaveProgress: {
    if (saveUi) {
      EQuitAction action = x8_quitScreen->Update(dt);
      if (action == EQuitAction::Yes) {
        memmove(g_GameState->SystemOptions().GetNESState(), x4_nesEmu->GetPassword(), 18);
        saveUi->SaveNESState();
        SetMode(EMode::ContinuePlaying);
      } else if (action == EQuitAction::No)
        SetMode(EMode::ContinuePlaying);
    } else
      SetMode(EMode::ContinuePlaying);
    break;
  }

  case EMode::ContinuePlaying: {
    EQuitAction action = x8_quitScreen->Update(dt);
    if (action == EQuitAction::Yes)
      SetMode(EMode::Emulator);
    else if (action == EQuitAction::No)
      return true;
    break;
  }

  case EMode::QuitNESMetroid: {
    EQuitAction action = x8_quitScreen->Update(dt);
    if (action == EQuitAction::Yes)
      return true;
    else if (action == EQuitAction::No)
      SetMode(EMode::Emulator);
    break;
  }

  default:
    break;
  }

  return false;
}

void CFrontEndUI::SNesEmulatorFrame::Draw(CSaveGameScreen* saveUi) const {
  zeus::CColor mulColor = zeus::skWhite;
  bool blackout = saveUi && saveUi->GetUIType() != CSaveGameScreen::EUIType::SaveReady;

  if (blackout)
    mulColor = zeus::skBlack;
  else if (x8_quitScreen)
    mulColor = zeus::CColor{0.376470f, 0.376470f, 0.376470f, 1.f};

  x4_nesEmu->Draw(mulColor, x15_enableFiltering);
  if (!blackout && x8_quitScreen)
    x8_quitScreen->Draw();

  if (x10_remTime >= 7.5f)
    return;
  if (x10_remTime <= 0.f)
    return;
  if (xc_textSupport->GetIsTextSupportFinishedLoading()) {
    float aspect = g_Viewport.aspect / 1.33f;
    CGraphics::SetOrtho(-320.f * aspect, 320.f * aspect, 240.f, -240.f, -4096.f, 4096.f);
    CGraphics::SetViewPointMatrix(zeus::CTransform());
    CGraphics::SetModelMatrix(zeus::CTransform::Translate(-220.f, 0.f, -200.f));
    xc_textSupport->Render();
  }
}

CFrontEndUI::SOptionsFrontEndFrame::SOptionsFrontEndFrame() {
  x4_frme = g_SimplePool->GetObj("FRME_OptionsFrontEnd");
  x10_pauseScreen = g_SimplePool->GetObj("STRG_PauseScreen");
  x134_24_visible = true;
}

void CFrontEndUI::SOptionsFrontEndFrame::DoSliderChange(CGuiSliderGroup* caller, float value) {
  if (x28_tablegroup_rightmenu->GetIsActive()) {
    int leftSel = x24_tablegroup_leftmenu->GetUserSelection();
    int rightSel = x28_tablegroup_rightmenu->GetUserSelection();
    const auto& optionCategory = GameOptionsRegistry[leftSel];
    const SGameOption& option = optionCategory.second[rightSel];
    CGameOptions::SetOption(option.option, caller->GetGurVal());
    m_touchBarValueDirty = true;
  }
}

void CFrontEndUI::SOptionsFrontEndFrame::DoMenuCancel(CGuiTableGroup* caller) {
  if (x28_tablegroup_rightmenu == caller) {
    DeactivateRightMenu();
    x24_tablegroup_leftmenu->SetIsActive(true);
    x28_tablegroup_rightmenu->SetIsActive(false);
    x28_tablegroup_rightmenu->SetUserSelection(0);
    SetTableColors(x28_tablegroup_rightmenu);
    CSfxManager::SfxStart(SFXfnt_back, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
  }
}

void CFrontEndUI::SOptionsFrontEndFrame::DoMenuSelectionChange(CGuiTableGroup* caller, int oldSel) {
  SetTableColors(caller);
  if (x24_tablegroup_leftmenu == caller) {
    SetRightUIText();
    CSfxManager::SfxStart(SFXfnt_selection_change, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
  } else if (x28_tablegroup_rightmenu == caller) {
    HandleRightSelectionChange();
    CSfxManager::SfxStart(SFXfnt_selection_change, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
  } else if (x2c_tablegroup_double == caller || x30_tablegroup_triple == caller) {
    if (x28_tablegroup_rightmenu->GetIsActive()) {
      int leftSel = x24_tablegroup_leftmenu->GetUserSelection();
      int rightSel = x28_tablegroup_rightmenu->GetUserSelection();
      const auto& optionCategory = GameOptionsRegistry[leftSel];
      const SGameOption& option = optionCategory.second[rightSel];
      CGameOptions::SetOption(option.option, caller->GetUserSelection());
      m_touchBarValueDirty = true;
      CSfxManager::SfxStart(SFXfnt_enum_change, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);

      if (option.option == EGameOption::Rumble && caller->GetUserSelection() > 0) {
        x40_rumbleGen.HardStopAll();
        x40_rumbleGen.Rumble(RumbleFxTable[size_t(ERumbleFxId::PlayerBump)], 1.f, ERumblePriority::One, EIOPort::Zero);
      }
    }
  }
}

void CFrontEndUI::SOptionsFrontEndFrame::DoLeftMenuAdvance(CGuiTableGroup* caller) {
  if (caller == x24_tablegroup_leftmenu) {
    HandleRightSelectionChange();
    x28_tablegroup_rightmenu->SetUserSelection(0);
    x24_tablegroup_leftmenu->SetIsActive(false);
    x28_tablegroup_rightmenu->SetIsActive(true);
    CSfxManager::SfxStart(SFXfnt_advance_L, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
    CSfxManager::SfxStart(SFXfnt_advance_R, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
  }
}

void CFrontEndUI::SOptionsFrontEndFrame::DeactivateRightMenu() {
  x2c_tablegroup_double->SetIsActive(false);
  x30_tablegroup_triple->SetIsActive(false);
  x34_slidergroup_slider->SetIsActive(false);
  x2c_tablegroup_double->SetVisibility(false, ETraversalMode::Children);
  x30_tablegroup_triple->SetVisibility(false, ETraversalMode::Children);
  x34_slidergroup_slider->SetVisibility(false, ETraversalMode::Children);
}

void CFrontEndUI::SOptionsFrontEndFrame::HandleRightSelectionChange() {
  DeactivateRightMenu();
  int leftSel = x24_tablegroup_leftmenu->GetUserSelection();
  int rightSel = x28_tablegroup_rightmenu->GetUserSelection();
  const auto& optionCategory = GameOptionsRegistry[leftSel];
  const SGameOption& option = optionCategory.second[rightSel];

  switch (option.type) {
  case EOptionType::Float:
    x34_slidergroup_slider->SetIsActive(true);
    x34_slidergroup_slider->SetVisibility(true, ETraversalMode::Children);
    x34_slidergroup_slider->SetMinVal(option.minVal);
    x34_slidergroup_slider->SetMaxVal(option.maxVal);
    x34_slidergroup_slider->SetIncrement(option.increment);
    x34_slidergroup_slider->SetCurVal(CGameOptions::GetOption(option.option));
    x34_slidergroup_slider->SetLocalTransform(zeus::CTransform::Translate(0.f, 0.f, rightSel * x38_rowPitch) *
                                              x34_slidergroup_slider->GetTransform());
    break;

  case EOptionType::DoubleEnum:
    x2c_tablegroup_double->SetUserSelection(CGameOptions::GetOption(option.option));
    x2c_tablegroup_double->SetIsVisible(true);
    x2c_tablegroup_double->SetIsActive(true);
    x2c_tablegroup_double->SetLocalTransform(zeus::CTransform::Translate(0.f, 0.f, rightSel * x38_rowPitch) *
                                             x2c_tablegroup_double->GetTransform());
    SetTableColors(x2c_tablegroup_double);
    break;

  case EOptionType::TripleEnum:
    x30_tablegroup_triple->SetUserSelection(CGameOptions::GetOption(option.option));
    x30_tablegroup_triple->SetIsVisible(true);
    x30_tablegroup_triple->SetIsActive(true);
    x30_tablegroup_triple->SetLocalTransform(zeus::CTransform::Translate(0.f, 0.f, rightSel * x38_rowPitch) *
                                             x30_tablegroup_triple->GetTransform());
    SetTableColors(x30_tablegroup_triple);
    break;

  default:
    break;
  }
}

void CFrontEndUI::SOptionsFrontEndFrame::SetRightUIText() {
  const int userSel = x24_tablegroup_leftmenu->GetUserSelection();
  const auto& options = GameOptionsRegistry[userSel];

  for (int i = 0; i < 5; ++i) {
    std::string name = fmt::format(fmt("textpane_right{}"), i);
    if (i < static_cast<int>(options.first)) {
      FindTextPanePair(x1c_loadedFrame, name).SetPairText(x20_loadedPauseStrg->GetString(options.second[i].stringId));
      x28_tablegroup_rightmenu->GetWorkerWidget(i)->SetIsSelectable(true);
    } else {
      FindTextPanePair(x1c_loadedFrame, name).SetPairText(u"");
      x28_tablegroup_rightmenu->GetWorkerWidget(i)->SetIsSelectable(false);
    }
  }
}

void CFrontEndUI::SOptionsFrontEndFrame::SetTableColors(CGuiTableGroup* tbgp) const {
  tbgp->SetColors(zeus::skWhite, zeus::CColor{0.627450f, 0.627450f, 0.627450f, 0.784313f});
}

void CFrontEndUI::SOptionsFrontEndFrame::FinishedLoading() {
  x1c_loadedFrame->SetAspectConstraint(1.78f);

  x24_tablegroup_leftmenu = static_cast<CGuiTableGroup*>(x1c_loadedFrame->FindWidget("tablegroup_leftmenu"));
  x28_tablegroup_rightmenu = static_cast<CGuiTableGroup*>(x1c_loadedFrame->FindWidget("tablegroup_rightmenu"));
  x2c_tablegroup_double = static_cast<CGuiTableGroup*>(x1c_loadedFrame->FindWidget("tablegroup_double"));
  x30_tablegroup_triple = static_cast<CGuiTableGroup*>(x1c_loadedFrame->FindWidget("tablegroup_triple"));
  x34_slidergroup_slider = static_cast<CGuiSliderGroup*>(x1c_loadedFrame->FindWidget("slidergroup_slider"));

  x24_tablegroup_leftmenu->SetMenuAdvanceCallback([this](CGuiTableGroup* caller) { DoLeftMenuAdvance(caller); });
  x24_tablegroup_leftmenu->SetMenuSelectionChangeCallback(
      [this](CGuiTableGroup* caller, int oldSel) { DoMenuSelectionChange(caller, oldSel); });

  x38_rowPitch = x24_tablegroup_leftmenu->GetWorkerWidget(1)->GetIdlePosition().z() -
                 x24_tablegroup_leftmenu->GetWorkerWidget(0)->GetIdlePosition().z();

  x28_tablegroup_rightmenu->SetMenuSelectionChangeCallback(
      [this](CGuiTableGroup* caller, int oldSel) { DoMenuSelectionChange(caller, oldSel); });
  x28_tablegroup_rightmenu->SetMenuCancelCallback([this](CGuiTableGroup* caller) { DoMenuCancel(caller); });

  x2c_tablegroup_double->SetMenuSelectionChangeCallback(
      [this](CGuiTableGroup* caller, int oldSel) { DoMenuSelectionChange(caller, oldSel); });
  x2c_tablegroup_double->SetMenuCancelCallback([this](CGuiTableGroup* caller) { DoMenuCancel(caller); });

  x30_tablegroup_triple->SetMenuSelectionChangeCallback(
      [this](CGuiTableGroup* caller, int oldSel) { DoMenuSelectionChange(caller, oldSel); });
  x30_tablegroup_triple->SetMenuCancelCallback([this](CGuiTableGroup* caller) { DoMenuCancel(caller); });

  x34_slidergroup_slider->SetSelectionChangedCallback(
      [this](CGuiSliderGroup* caller, float value) { DoSliderChange(caller, value); });

  FindTextPanePair(x1c_loadedFrame, "textpane_double0").SetPairText(x20_loadedPauseStrg->GetString(95)); // Off
  FindTextPanePair(x1c_loadedFrame, "textpane_double1").SetPairText(x20_loadedPauseStrg->GetString(94)); // On
  FindTextPanePair(x1c_loadedFrame, "textpane_triple0").SetPairText(x20_loadedPauseStrg->GetString(96)); // Mono
  FindTextPanePair(x1c_loadedFrame, "textpane_triple1").SetPairText(x20_loadedPauseStrg->GetString(97)); // Stereo
  FindTextPanePair(x1c_loadedFrame, "textpane_triple2").SetPairText(x20_loadedPauseStrg->GetString(98)); // Dolby

  FindTextPanePair(x1c_loadedFrame, "textpane_title").SetPairText(g_MainStringTable->GetString(99)); // OPTIONS

  if (CGuiTextPane* proceed = static_cast<CGuiTextPane*>(x1c_loadedFrame->FindWidget("textpane_proceed")))
    proceed->TextSupport().SetText(g_MainStringTable->GetString(85));

  if (CGuiTextPane* cancel = static_cast<CGuiTextPane*>(x1c_loadedFrame->FindWidget("textpane_cancel")))
    cancel->TextSupport().SetText(g_MainStringTable->GetString(82));

  // Visor, Display, Sound, Controller
  for (int i = 0; i < 4; ++i) {
    const std::string name = fmt::format(fmt("textpane_filename{}"), i);
    FindTextPanePair(x1c_loadedFrame, name).SetPairText(x20_loadedPauseStrg->GetString(16 + i));
  }

  x2c_tablegroup_double->SetVertical(false);
  x30_tablegroup_triple->SetVertical(false);

  x24_tablegroup_leftmenu->SetIsActive(true);
  x28_tablegroup_rightmenu->SetIsActive(false);

  SetTableColors(x24_tablegroup_leftmenu);
  SetTableColors(x28_tablegroup_rightmenu);
  SetTableColors(x2c_tablegroup_double);
  SetTableColors(x30_tablegroup_triple);

  SetRightUIText();
  DeactivateRightMenu();
}

bool CFrontEndUI::SOptionsFrontEndFrame::PumpLoad() {
  if (x1c_loadedFrame)
    return true;
  if (!x4_frme.IsLoaded())
    return false;
  if (!x10_pauseScreen.IsLoaded())
    return false;
  if (!x4_frme->GetIsFinishedLoading())
    return false;
  x1c_loadedFrame = x4_frme.GetObj();
  x20_loadedPauseStrg = x10_pauseScreen.GetObj();
  m_touchBar = NewGameOptionsTouchBar();
  FinishedLoading();
  return true;
}

bool CFrontEndUI::SOptionsFrontEndFrame::ProcessUserInput(const CFinalInput& input, CSaveGameScreen* sui) {
  x134_25_exitOptions = false;
  if (sui)
    sui->ProcessUserInput(input);
  if (x1c_loadedFrame && x134_24_visible) {
    if ((input.PB() || input.PSpecialKey(boo::ESpecialKey::Esc)) && x24_tablegroup_leftmenu->GetIsActive()) {
      x134_25_exitOptions = true;
      CSfxManager::SfxStart(SFXfnt_back, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
    } else {
      x1c_loadedFrame->ProcessUserInput(input);
      int leftSel = x24_tablegroup_leftmenu->GetUserSelection();
      int rightSel = x28_tablegroup_rightmenu->GetUserSelection();
      CGameOptions::TryRestoreDefaults(input, leftSel, rightSel, true, false);

      CGameOptionsTouchBar::EAction tbAction = m_touchBar->PopAction();
      if (x28_tablegroup_rightmenu->GetIsActive()) {
        if (tbAction == CGameOptionsTouchBar::EAction::Advance && !m_touchBarInValue) {
          int value;
          m_touchBar->GetSelection(leftSel, rightSel, value);
          x28_tablegroup_rightmenu->SetUserSelection(rightSel);
          SetTableColors(x28_tablegroup_rightmenu);
          HandleRightSelectionChange();
          const auto& optionCategory = GameOptionsRegistry[leftSel];
          const SGameOption& option = optionCategory.second[rightSel];
          if (option.type != EOptionType::RestoreDefaults) {
            m_touchBarInValue = true;
            m_touchBarValueDirty = true;
          } else {
            CGameOptions::TryRestoreDefaults(input, leftSel, rightSel, true, true);
          }
        } else if (tbAction == CGameOptionsTouchBar::EAction::Back) {
          if (m_touchBarInValue)
            m_touchBarInValue = false;
          else
            DoMenuCancel(x28_tablegroup_rightmenu);
        } else if (tbAction == CGameOptionsTouchBar::EAction::ValueChange) {
          int value;
          m_touchBar->GetSelection(leftSel, rightSel, value);
          const auto& optionCategory = GameOptionsRegistry[leftSel];
          const SGameOption& option = optionCategory.second[rightSel];
          CGameOptions::SetOption(option.option, value);
          if (option.type != EOptionType::Float)
            m_touchBarValueDirty = true;
          HandleRightSelectionChange();
        } else {
          if (m_touchBarInValue) {
            if (m_touchBarValueDirty) {
              const auto& optionCategory = GameOptionsRegistry[leftSel];
              const SGameOption& option = optionCategory.second[rightSel];
              int value = CGameOptions::GetOption(option.option);
              m_touchBar->SetSelection(leftSel, rightSel, value);
              m_touchBarValueDirty = false;
            }
          } else {
            m_touchBar->SetSelection(leftSel, -1, -1);
          }
        }
      } else {
        if (tbAction == CGameOptionsTouchBar::EAction::Advance) {
          int value;
          m_touchBar->GetSelection(leftSel, rightSel, value);
          x24_tablegroup_leftmenu->SetUserSelection(leftSel);
          SetTableColors(x24_tablegroup_leftmenu);
          SetRightUIText();
          DoLeftMenuAdvance(x24_tablegroup_leftmenu);
        }
        if (tbAction == CGameOptionsTouchBar::EAction::Back) {
          x134_25_exitOptions = true;
          CSfxManager::SfxStart(SFXfnt_back, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
        } else {
          m_touchBar->SetSelection(-1, -1, -1);
        }
      }
    }
  }
  return !x134_25_exitOptions;
}

void CFrontEndUI::SOptionsFrontEndFrame::Update(float dt, CSaveGameScreen* sui) {
  x40_rumbleGen.Update(dt);
  x134_24_visible = !sui || sui->GetUIType() == CSaveGameScreen::EUIType::SaveReady;

  if (!PumpLoad())
    return;

  x0_uiAlpha = std::min(1.f, x0_uiAlpha + dt);
  x1c_loadedFrame->Update(dt);

  bool isSliding = x34_slidergroup_slider->GetState() != CGuiSliderGroup::EState::None;
  if (x3c_sliderSfx.operator bool() != isSliding) {
    if (isSliding) {
      x3c_sliderSfx =
          CSfxManager::SfxStart(SFXui_frontend_options_slider_change_lp, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
    } else {
      CSfxManager::SfxStop(x3c_sliderSfx);
      x3c_sliderSfx.reset();
    }
  }
}

void CFrontEndUI::SOptionsFrontEndFrame::Draw() const {
  if (x1c_loadedFrame && x134_24_visible) {
    CGuiWidgetDrawParms params(x0_uiAlpha, zeus::skZero3f);
    x1c_loadedFrame->Draw(params);
  }
}

CFrontEndUI::CFrontEndUI() : CIOWin("FrontEndUI") {
  CMain* m = static_cast<CMain*>(g_Main);

  CRandom16 r(time(nullptr));
  x18_rndA = r.Range(0, 2);
  x1c_rndB = r.Range(0, 2);

  x20_depsGroup = g_SimplePool->GetObj("FrontEnd_DGRP");
  x38_pressStart = g_SimplePool->GetObj("TXTR_PressStart");
  x44_frontendAudioGrp = g_SimplePool->GetObj("FrontEnd_AGSC");

  xdc_saveUI = std::make_unique<CSaveGameScreen>(ESaveContext::FrontEnd, g_GameState->GetCardSerial());

  m->ResetGameState();
  g_GameState->SetCurrentWorldId(g_DefaultWorldTag.id);
  g_GameState->GameOptions().ResetToDefaults();
  g_GameState->WriteBackupBuf();

  for (int i = 0; CDvdFile::FileExists(GetAttractMovieFileName(i)); ++i)
    ++xc0_attractCount;

  m_touchBar = NewFrontEndUITouchBar();
  m_touchBar->SetPhase(CFrontEndUITouchBar::EPhase::None);
}

void CFrontEndUI::StartSlideShow(CArchitectureQueue& queue) {
  xf4_curAudio->StopMixing();
  queue.Push(MakeMsg::CreateCreateIOWin(EArchMsgTarget::IOWinManager, 12, 11, std::make_shared<CSlideShow>()));
}

std::string CFrontEndUI::GetAttractMovieFileName(int idx) { return fmt::format(fmt("Video/attract{}.thp"), idx); }

std::string CFrontEndUI::GetNextAttractMovieFileName() {
  std::string ret = GetAttractMovieFileName(xbc_nextAttract);
  xbc_nextAttract = (xbc_nextAttract + 1) % xc0_attractCount;
  return ret;
}

void CFrontEndUI::SetCurrentMovie(EMenuMovie movie) {
  if (xb8_curMovie == movie) {
    return;
  }

  StopAttractMovie();

  if (xb8_curMovie != EMenuMovie::Stopped) {
    xcc_curMoviePtr->SetPlayMode(CMoviePlayer::EPlayMode::Stopped);
    xcc_curMoviePtr->Rewind();
  }

  xb8_curMovie = movie;

  if (xb8_curMovie != EMenuMovie::Stopped) {
    xcc_curMoviePtr = x70_menuMovies[size_t(xb8_curMovie)].get();
    xcc_curMoviePtr->SetPlayMode(CMoviePlayer::EPlayMode::Playing);
  } else {
    xcc_curMoviePtr = nullptr;
  }
}

void CFrontEndUI::StopAttractMovie() {
  if (!xc4_attractMovie)
    return;
  xc4_attractMovie.reset();
  xcc_curMoviePtr = nullptr;
}

void CFrontEndUI::StartAttractMovie() {
  if (xc4_attractMovie)
    return;
  SetCurrentMovie(EMenuMovie::Stopped);
  xc4_attractMovie = std::make_unique<CMoviePlayer>(GetNextAttractMovieFileName().c_str(), 0.f, false, true);
  xcc_curMoviePtr = xc4_attractMovie.get();
}

void CFrontEndUI::StartStateTransition(EScreen screen) {
  switch (x50_curScreen) {
  case EScreen::Title:
    if (screen != EScreen::FileSelect)
      break;
    SetCurrentMovie(EMenuMovie::StartFileSelectA);
    SetFadeBlackTimer(xcc_curMoviePtr->GetTotalSeconds());
    break;
  case EScreen::FileSelect:
    if (screen == EScreen::ToPlayGame) {
      SetCurrentMovie(EMenuMovie::FileSelectPlayGameA);
      SetFadeBlackTimer(xcc_curMoviePtr->GetTotalSeconds());
    } else if (screen == EScreen::FusionBonus) {
      SetCurrentMovie(EMenuMovie::FileSelectGBA);
      SetFadeBlackTimer(xcc_curMoviePtr->GetTotalSeconds());
      CSfxManager::SfxStart(SFXfnt_tofusion_L, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
      CSfxManager::SfxStart(SFXfnt_tofusion_R, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
    }
    break;
  case EScreen::FusionBonus:
    if (screen == EScreen::ToPlayGame) {
      SetCurrentMovie(EMenuMovie::GBAFileSelectB);
      SetFadeBlackTimer(xcc_curMoviePtr->GetTotalSeconds());
    } else if (screen == EScreen::FileSelect) {
      SetCurrentMovie(EMenuMovie::GBAFileSelectA);
      SetFadeBlackTimer(xcc_curMoviePtr->GetTotalSeconds());
      CSfxManager::SfxStart(SFXfnt_fromfusion_L, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
      CSfxManager::SfxStart(SFXfnt_fromfusion_R, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
    }
    break;
  default:
    break;
  }

  switch (screen) {
  case EScreen::OpenCredits:
  case EScreen::Title:
    SetCurrentMovie(EMenuMovie::FirstStart);
    SetFadeBlackTimer(xcc_curMoviePtr->GetTotalSeconds());
    break;
  case EScreen::AttractMovie:
    StartAttractMovie();
    SetFadeBlackWithMovie();
    break;
  default:
    break;
  }

  x54_nextScreen = screen;
}

void CFrontEndUI::CompleteStateTransition() {
  EScreen oldScreen = x50_curScreen;
  x50_curScreen = x54_nextScreen;

  switch (x50_curScreen) {
  case EScreen::AttractMovie:
    x54_nextScreen = EScreen::OpenCredits;
    x50_curScreen = EScreen::OpenCredits;
    xd0_playerSkipToTitle = false;
    StartStateTransition(EScreen::Title);
    break;

  case EScreen::Title:
    SetCurrentMovie(EMenuMovie::StartLoop);
    SetFadeBlackTimer(30.f);
    break;

  case EScreen::FileSelect:
    SetCurrentMovie(EMenuMovie::FileSelectLoop);
    if (oldScreen == EScreen::Title) {
      xf4_curAudio->StopMixing();
      xf4_curAudio = xd8_audio2.get();
      xf4_curAudio->StartMixing();
    }
    if (xdc_saveUI)
      xdc_saveUI->ResetCardDriver();
    break;

  case EScreen::FusionBonus:
    SetCurrentMovie(EMenuMovie::GBALoop);
    break;

  case EScreen::ToPlayGame:
    x14_phase = EPhase::ExitFrontEnd;
    break;

  default:
    break;
  }
}

void CFrontEndUI::HandleDebugMenuReturnValue(CGameDebug::EReturnValue val, CArchitectureQueue& queue) {}

void CFrontEndUI::Draw() const {
  if (x14_phase < EPhase::DisplayFrontEnd)
    return;
  SCOPED_GRAPHICS_DEBUG_GROUP("CFrontEndUI::Draw", zeus::skGreen);

  if (xec_emuFrme) {
    xec_emuFrme->Draw(xdc_saveUI.get());
  } else {
    // g_Renderer->SetDepthReadWrite(false, false);
    g_Renderer->SetViewportOrtho(false, -4096.f, 4096.f);

    /* Correct movie aspect ratio */
    float hPad, vPad;
    if (g_Viewport.aspect >= 1.78f) {
      hPad = 1.78f / g_Viewport.aspect;
      vPad = 1.78f / 1.33f;
    } else {
      hPad = 1.f;
      vPad = g_Viewport.aspect / 1.33f;
    }

    if (xcc_curMoviePtr && xcc_curMoviePtr->GetIsFullyCached()) {
      /* Render movie */
      xcc_curMoviePtr->SetFrame({-hPad, vPad, 0.f}, {-hPad, -vPad, 0.f}, {hPad, -vPad, 0.f}, {hPad, vPad, 0.f});
      xcc_curMoviePtr->DrawFrame();
    }

    if (x50_curScreen == EScreen::FileSelect && x54_nextScreen == EScreen::FileSelect) {
      /* Render active FileSelect UI */
      if (xf0_optionsFrme)
        xf0_optionsFrme->Draw();
      else if (xe0_frontendCardFrme)
        xe0_frontendCardFrme->Draw();
      else
        xe8_frontendNoCardFrme->Draw();
    } else if (x50_curScreen == EScreen::FusionBonus && x54_nextScreen == EScreen::FusionBonus) {
      /* Render Fusion bonus UI */
      xe4_fusionBonusFrme->Draw();
    }

    if (x64_pressStartAlpha > 0.f && x38_pressStart.IsLoaded() && m_pressStartQuad) {
      /* Render "Press Start" */
      zeus::CRectangle rect(0.5f - x38_pressStart->GetWidth() / 2.f / 640.f * hPad,
                            0.5f + (x38_pressStart->GetHeight() / 2.f - 240.f + 72.f) / 480.f * vPad,
                            x38_pressStart->GetWidth() / 640.f * hPad, x38_pressStart->GetHeight() / 480.f * vPad);
      zeus::CColor color = zeus::skWhite;
      color.a() = x64_pressStartAlpha;
      const_cast<CTexturedQuadFilterAlpha&>(*m_pressStartQuad).draw(color, 1.f, rect);
    }

    if (xc0_attractCount > 0) {
      /* Render fade-to-black into attract movie */
      if (((x50_curScreen == EScreen::Title && x54_nextScreen == EScreen::Title) ||
           x54_nextScreen == EScreen::AttractMovie) &&
          x58_fadeBlackTimer < 1.f) {
        /* To black */
        zeus::CColor color = zeus::skBlack;
        color.a() = 1.f - x58_fadeBlackTimer;
        const_cast<CColoredQuadFilter&>(m_fadeToBlack).draw(color);
      }
    }

    if (xd0_playerSkipToTitle) {
      /* Render fade-through-black into title if player skips */
      if (x50_curScreen == EScreen::OpenCredits && x54_nextScreen == EScreen::Title) {
        /* To black */
        zeus::CColor color = zeus::skBlack;
        color.a() = zeus::clamp(0.f, 1.f - x58_fadeBlackTimer, 1.f);
        const_cast<CColoredQuadFilter&>(m_fadeToBlack).draw(color);
      } else if (x50_curScreen == EScreen::Title && x54_nextScreen == EScreen::Title) {
        /* From black with 30-sec skip to title */
        zeus::CColor color = zeus::skBlack;
        color.a() = 1.f - zeus::clamp(0.f, 30.f - x58_fadeBlackTimer, 1.f);
        const_cast<CColoredQuadFilter&>(m_fadeToBlack).draw(color);
      }
    }

    if (xdc_saveUI) {
      /* Render memory card feedback strings */
      if ((CanShowSaveUI() && !xdc_saveUI->IsHiddenFromFrontEnd()) ||
          ((x50_curScreen == EScreen::FileSelect && x54_nextScreen == EScreen::FileSelect) ||
           (x50_curScreen == EScreen::FusionBonus && x54_nextScreen == EScreen::FusionBonus)))
        xdc_saveUI->Draw();
    }
  }
}

void CFrontEndUI::UpdateMovies(float dt) {
  if (xcc_curMoviePtr && x5c_fadeBlackWithMovie) {
    /* Set fade-to-black timer to match attract movie */
    x5c_fadeBlackWithMovie = false;
    x58_fadeBlackTimer = xcc_curMoviePtr->GetTotalSeconds();
  }

  /* Advance playing menu movies */
  for (auto& movie : x70_menuMovies)
    if (movie)
      movie->Update(dt);

  /* Advance attract movie */
  if (xc4_attractMovie)
    xc4_attractMovie->Update(dt);
}

void CFrontEndUI::FinishedLoadingDepsGroup() {
  /* Transfer DGRP tokens into FrontEnd and lock */
  const CDependencyGroup* dgrp = x20_depsGroup.GetObj();
  x2c_deps.reserve(dgrp->GetObjectTagVector().size());
  for (const SObjectTag& tag : dgrp->GetObjectTagVector()) {
    x2c_deps.push_back(g_SimplePool->GetObj(tag));
    x2c_deps.back().Lock();
  }
  x44_frontendAudioGrp.Lock();
}

bool CFrontEndUI::PumpLoad() {
  /* Poll all tokens for load completion */
  for (CToken& tok : x2c_deps)
    if (!tok.IsLoaded())
      return false;
  if (!x44_frontendAudioGrp.IsLoaded())
    return false;

  /* Ready to construct texture quads */
  m_pressStartQuad.emplace(EFilterType::Add, x38_pressStart);

  return true;
}

bool CFrontEndUI::PumpMovieLoad() {
  /* Prepare all FrontEnd movies and pause each */
  if (xd1_moviesLoaded) {
    return true;
  }

  for (size_t i = 0; i < x70_menuMovies.size(); ++i) {
    if (x70_menuMovies[i] != nullptr) {
      continue;
    }

    const FEMovie& movie = FEMovies[i];
    std::string path = movie.path;
    if (i == size_t(EMenuMovie::StartFileSelectA)) {
      const auto pos = path.find("A.thp");
      if (pos != std::string::npos) {
        path[pos] = 'A' + x18_rndA;
      }
    } else if (i == size_t(EMenuMovie::FileSelectPlayGameA)) {
      const auto pos = path.find("A.thp");
      if (pos != std::string::npos) {
        path[pos] = 'A' + x1c_rndB;
      }
    }

    x70_menuMovies[i] = std::make_unique<CMoviePlayer>(path.c_str(), 0.05f, movie.loop, false);
    x70_menuMovies[i]->SetPlayMode(CMoviePlayer::EPlayMode::Stopped);
    return false;
  }

  xd1_moviesLoaded = true;
  return true;
}

void CFrontEndUI::ProcessUserInput(const CFinalInput& input, CArchitectureQueue& queue) {
  if (static_cast<CMain*>(g_Main)->GetCardBusy())
    return;
  if (input.ControllerIdx() > 1)
    return;

  if (xec_emuFrme) {
    /* NES emulator pre-empts user input if active */
    xec_emuFrme->ProcessUserInput(input, xdc_saveUI.get());
    return;
  }

  /* Controllers other than first shall not pass */
  if (x14_phase != EPhase::DisplayFrontEnd || input.ControllerIdx() != 0)
    return;

  /* Pop most recent action from Touch Bar */
  CFrontEndUITouchBar::EAction touchBarAction = m_touchBar->PopAction();

  if (x50_curScreen != x54_nextScreen) {
    if (x54_nextScreen == EScreen::AttractMovie &&
        (input.PStart() || input.PA() ||
         input.PSpecialKey(boo::ESpecialKey::Esc) ||
         input.PSpecialKey(boo::ESpecialKey::Enter) ||
         input.PMouseButton(boo::EMouseButton::Primary) ||
         touchBarAction == CFrontEndUITouchBar::EAction::Start)) {
      /* Player wants to return to opening credits from attract movie */
      SetFadeBlackTimer(std::min(1.f, x58_fadeBlackTimer));
      PlayAdvanceSfx();
      return;
    }

    if (input.PA() || input.PStart() ||
        input.PSpecialKey(boo::ESpecialKey::Esc) ||
        input.PSpecialKey(boo::ESpecialKey::Enter) ||
        input.PMouseButton(boo::EMouseButton::Primary) ||
        touchBarAction == CFrontEndUITouchBar::EAction::Start) {
      if (x50_curScreen == EScreen::OpenCredits && x54_nextScreen == EScreen::Title && x58_fadeBlackTimer > 1.f) {
        /* Player is too impatient to view opening credits */
        xd0_playerSkipToTitle = true;
        SetFadeBlackTimer(1.f);
        return;
      }
    }
  } else {
    if (x50_curScreen == EScreen::Title) {
      if (input.PStart() || input.PA() ||
          input.PSpecialKey(boo::ESpecialKey::Esc) ||
          input.PSpecialKey(boo::ESpecialKey::Enter) ||
          input.PMouseButton(boo::EMouseButton::Primary) ||
          touchBarAction == CFrontEndUITouchBar::EAction::Start) {
        if (x58_fadeBlackTimer < 30.f - g_tweakGame->GetPressStartDelay()) {
          /* Proceed to file select UI */
          CSfxManager::SfxStart(FETransitionBackSFX[x18_rndA][0], 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
          CSfxManager::SfxStart(FETransitionBackSFX[x18_rndA][1], 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
          m_touchBar->SetPhase(CFrontEndUITouchBar::EPhase::None);
          StartStateTransition(EScreen::FileSelect);
          return;
        }
      }
    } else if (x50_curScreen == EScreen::FileSelect && x54_nextScreen == EScreen::FileSelect) {
      if (xf0_optionsFrme) {
        /* Control options UI */
        if (xf0_optionsFrme->ProcessUserInput(input, xdc_saveUI.get()))
          return;
        /* Exit options UI */
        m_touchBar->SetPhase(CFrontEndUITouchBar::EPhase::None);
        xf0_optionsFrme.reset();
        return;
      } else if (xe0_frontendCardFrme) {
        /* Control FrontEnd with memory card */
        switch (xe0_frontendCardFrme->ProcessUserInput(input, touchBarAction)) {
        case SNewFileSelectFrame::EAction::FusionBonus:
          m_touchBar->SetPhase(CFrontEndUITouchBar::EPhase::None);
          StartStateTransition(EScreen::FusionBonus);
          return;
        case SNewFileSelectFrame::EAction::GameOptions:
          m_touchBar->SetPhase(CFrontEndUITouchBar::EPhase::None);
          xf0_optionsFrme = std::make_unique<SOptionsFrontEndFrame>();
          return;
        case SNewFileSelectFrame::EAction::SlideShow:
          m_touchBar->SetPhase(CFrontEndUITouchBar::EPhase::None);
          xd2_deferSlideShow = true;
          StartSlideShow(queue);
          return;
        default:
          return;
        }
      } else {
        /* Control FrontEnd without memory card */
        switch (xe8_frontendNoCardFrme->ProcessUserInput(input, touchBarAction)) {
        case SFrontEndFrame::EAction::FusionBonus:
          m_touchBar->SetPhase(CFrontEndUITouchBar::EPhase::None);
          StartStateTransition(EScreen::FusionBonus);
          return;
        case SFrontEndFrame::EAction::GameOptions:
          m_touchBar->SetPhase(CFrontEndUITouchBar::EPhase::None);
          xf0_optionsFrme = std::make_unique<SOptionsFrontEndFrame>();
          return;
        case SFrontEndFrame::EAction::StartGame:
          TransitionToGame();
          return;
        case SFrontEndFrame::EAction::SlideShow:
          m_touchBar->SetPhase(CFrontEndUITouchBar::EPhase::None);
          xd2_deferSlideShow = true;
          StartSlideShow(queue);
          return;
        default:
          return;
        }
      }
    } else if (x50_curScreen == EScreen::FusionBonus && x54_nextScreen == EScreen::FusionBonus) {
      /* Control Fusion bonus UI */
      switch (xe4_fusionBonusFrme->ProcessUserInput(input, xdc_saveUI.get(), touchBarAction)) {
      case SFusionBonusFrame::EAction::GoBack:
        m_touchBar->SetPhase(CFrontEndUITouchBar::EPhase::None);
        StartStateTransition(EScreen::FileSelect);
        return;
      case SFusionBonusFrame::EAction::PlayNESMetroid:
        m_touchBar->SetPhase(CFrontEndUITouchBar::EPhase::None);
        xf4_curAudio->StopMixing();
        xec_emuFrme = std::make_unique<SNesEmulatorFrame>();
        if (xdc_saveUI)
          xdc_saveUI->SetInGame(true);
        return;
      default:
        return;
      }
    }
  }
}

void CFrontEndUI::TransitionToGame() {
  if (x14_phase >= EPhase::ToPlayGame) {
    return;
  }

  const auto sfx = FETransitionForwardSFX[x1c_rndB];
  CSfxManager::SfxStart(sfx[0], 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);
  CSfxManager::SfxStart(sfx[1], 1.f, 0.f, false, 0x7f, false, kInvalidAreaId);

  x14_phase = EPhase::ToPlayGame;
  m_touchBar->SetPhase(CFrontEndUITouchBar::EPhase::None);
  StartStateTransition(EScreen::ToPlayGame);
}

void CFrontEndUI::UpdateMusicVolume() {
  float volMul = (xf4_curAudio == xd4_audio1.get()) ? FE1_VOL : FE2_VOL;
  if (xf4_curAudio) {
    float vol = volMul * x68_musicVol * (g_GameState->GameOptions().GetMusicVolume() / 127.f);
    xf4_curAudio->SetVolume(vol);
  }
}

CIOWin::EMessageReturn CFrontEndUI::Update(float dt, CArchitectureQueue& queue) {
  if (xdc_saveUI && x50_curScreen >= EScreen::FileSelect) {
    switch (xdc_saveUI->Update(dt)) {
    case EMessageReturn::Exit:
      /* Memory card operation complete, transition to game */
      TransitionToGame();
      break;
    case EMessageReturn::RemoveIOWinAndExit:
    case EMessageReturn::RemoveIOWin:
      /* No memory card available, fallback to non-save UI */
      xe0_frontendCardFrme.reset();
      xdc_saveUI.reset();
      break;
    default:
      break;
    }
  }

  /* Set music fade volume */
  UpdateMusicVolume();

  switch (x14_phase) {
  case EPhase::LoadDepsGroup:
    /* Poll DGRP load */
    if (!x20_depsGroup.IsLoaded())
      return EMessageReturn::Exit;
    FinishedLoadingDepsGroup();
    x20_depsGroup.Unlock();
    x14_phase = EPhase::LoadDeps;
    [[fallthrough]];

  case EPhase::LoadDeps:
    /* Poll loading DGRP resources */
    if (PumpLoad()) {
      xe0_frontendCardFrme = std::make_unique<SNewFileSelectFrame>(xdc_saveUI.get(), x1c_rndB, *m_touchBar);
      xe4_fusionBonusFrme = std::make_unique<SFusionBonusFrame>(*m_touchBar);
      xe8_frontendNoCardFrme = std::make_unique<SFrontEndFrame>(x1c_rndB, *m_touchBar);
      x38_pressStart.GetObj();
      CAudioSys::AddAudioGroup(x44_frontendAudioGrp->GetAudioGroupData());
      xd4_audio1 = std::make_unique<CStaticAudioPlayer>("Audio/frontend_1.rsf", 416480, 1973664);
      xd8_audio2 = std::make_unique<CStaticAudioPlayer>("Audio/frontend_2.rsf", 273556, 1636980);
      x14_phase = EPhase::LoadFrames;
    }
    if (x14_phase == EPhase::LoadDeps)
      return EMessageReturn::Exit;
    [[fallthrough]];

  case EPhase::LoadFrames:
    /* Poll loading music and FRME resources */
    if (!xd4_audio1->IsReady() || !xd8_audio2->IsReady() || !xe0_frontendCardFrme->PumpLoad() ||
        !xe4_fusionBonusFrme->PumpLoad() || !xe8_frontendNoCardFrme->PumpLoad() || !xdc_saveUI->PumpLoad())
      return EMessageReturn::Exit;
    xf4_curAudio = xd4_audio1.get();
    xf4_curAudio->StartMixing();
    x14_phase = EPhase::LoadMovies;
    [[fallthrough]];

  case EPhase::LoadMovies: {
    /* Poll loading movies */
    bool moviesReady = true;
    if (PumpMovieLoad()) {
      /* Prime first frame of movies */
      UpdateMovies(dt);

      moviesReady = std::all_of(x70_menuMovies.cbegin(), x70_menuMovies.cend(),
                                [](const auto& movie) { return movie->GetIsFullyCached(); });
    } else {
      moviesReady = false;
    }

    if (moviesReady) {
      /* Ready to display FrontEnd */
      x14_phase = EPhase::DisplayFrontEnd;
      m_touchBar->SetPhase(CFrontEndUITouchBar::EPhase::PressStart);
      StartStateTransition(EScreen::Title);
    } else {
      return EMessageReturn::Exit;
    }
    [[fallthrough]];
  }

  case EPhase::DisplayFrontEnd:
  case EPhase::ToPlayGame:
    /* Displaying frontend to user */
    if (xec_emuFrme) {
      /* Update just the emulator if active */
      if (xec_emuFrme->Update(dt, xdc_saveUI.get())) {
        /* Exit emulator */
        xec_emuFrme.reset();
        if (xdc_saveUI)
          xdc_saveUI->SetInGame(false);
        xf4_curAudio->StartMixing();
      }
      break;
    }

    if (xd2_deferSlideShow) {
      /* Start mixing slideshow music */
      xd2_deferSlideShow = false;
      xf4_curAudio->StartMixing();
      if (xdc_saveUI)
        xdc_saveUI->ResetCardDriver();
    }

    if (x50_curScreen == EScreen::FileSelect && x54_nextScreen == EScreen::FileSelect) {
      /* Main FrontEnd UI tree active */
      if (xf0_optionsFrme) {
        bool optionsActive = true;
        if (xdc_saveUI && xdc_saveUI->GetUIType() != CSaveGameScreen::EUIType::SaveReady)
          optionsActive = false;

        if (optionsActive) {
          /* Update options UI */
          xf0_optionsFrme->Update(dt, xdc_saveUI.get());
        } else {
          /* Save triggered; exit options UI here */
          xf0_optionsFrme.reset();
        }
      } else if (xe0_frontendCardFrme) {
        /* Update FrontEnd with memory card UI */
        xe0_frontendCardFrme->Update(dt);
      } else {
        /* Update FrontEnd without memory card UI */
        xe8_frontendNoCardFrme->Update(dt);
      }
    } else if (x50_curScreen == EScreen::FusionBonus && x54_nextScreen == EScreen::FusionBonus) {
      /* Update Fusion bonus UI */
      xe4_fusionBonusFrme->Update(dt, xdc_saveUI.get());
    }

    if (x50_curScreen != x54_nextScreen && xcc_curMoviePtr &&
        (xcc_curMoviePtr->GetIsMovieFinishedPlaying() || xcc_curMoviePtr->IsLooping())) {
      /* Movie-based transition complete */
      CompleteStateTransition();
    }

    if (x58_fadeBlackTimer > 0.f && !x5c_fadeBlackWithMovie) {
      SetFadeBlackTimer(std::max(0.f, x58_fadeBlackTimer - dt));
      if (x58_fadeBlackTimer == 0.f) {
        if (x50_curScreen == EScreen::Title && x54_nextScreen == EScreen::Title) {
          if (xc0_attractCount > 0) {
            /* Screen black, start attract movie */
            StartStateTransition(EScreen::AttractMovie);
          }
        } else if (x54_nextScreen == EScreen::AttractMovie) {
          /* Attract movie done, play open credits again */
          CompleteStateTransition();
        } else if (x50_curScreen != x54_nextScreen) {
          /* Fade-based transition complete */
          CompleteStateTransition();
        }
      }
    }

    /* Advance active movies */
    UpdateMovies(dt);

    if (x50_curScreen == EScreen::Title && x54_nextScreen == EScreen::Title) {
      /* Update press-start pulsing */
      if (x58_fadeBlackTimer < 30.f - g_tweakGame->GetPressStartDelay()) {
        x60_pressStartTime = std::fmod(x60_pressStartTime + dt, 1.f);
        if (x60_pressStartTime < 0.5f)
          x64_pressStartAlpha = x60_pressStartTime / 0.5f;
        else
          x64_pressStartAlpha = (1.f - x60_pressStartTime) / 0.5f;
      }
    } else {
      /* Clear press-start pulsing */
      x60_pressStartTime = 0.f;
      x64_pressStartAlpha = 0.f;
    }

    if (x50_curScreen == EScreen::Title && x54_nextScreen == EScreen::FileSelect) {
      /* Fade out title music */
      x68_musicVol =
          1.f - zeus::clamp(0.f, (xcc_curMoviePtr->GetPlayedSeconds() - AudioFadeTimeA[x18_rndA]) / 2.5f, 1.f);
    } else if (x54_nextScreen == EScreen::ToPlayGame) {
      /* Fade out menu music */
      float delay = AudioFadeTimeB[x1c_rndB];
      x68_musicVol =
          1.f -
          zeus::clamp(0.f, (xcc_curMoviePtr->GetPlayedSeconds() - delay) / (xcc_curMoviePtr->GetTotalSeconds() - delay),
                      1.f);
    } else {
      /* Full music volume */
      x68_musicVol = 1.f;
    }

    return EMessageReturn::Exit;

  case EPhase::ExitFrontEnd:
    /* Remove FrontEnd IOWin and begin updating next IOWin */
    return EMessageReturn::RemoveIOWin;

  default:
    break;
  }

  return EMessageReturn::Exit;
}

CIOWin::EMessageReturn CFrontEndUI::OnMessage(const CArchitectureMessage& msg, CArchitectureQueue& queue) {
  switch (msg.GetType()) {
  case EArchMsgType::UserInput: {
    /* Forward user events */
    const CArchMsgParmUserInput& input = MakeMsg::GetParmUserInput(msg);
    ProcessUserInput(input.x4_parm, queue);
    break;
  }
  case EArchMsgType::TimerTick: {
    /* Forward frame events */
    float dt = MakeMsg::GetParmTimerTick(msg).x4_parm;
    return Update(dt, queue);
  }
  case EArchMsgType::QuitGameplay: {
    /* Immediately exit FrontEnd */
    x14_phase = EPhase::ExitFrontEnd;
    break;
  }
  default:
    break;
  }
  return EMessageReturn::Normal;
}

} // namespace urde::MP1