//------------------------------------------------------------------------- /* Copyright (C) 2010-2019 EDuke32 developers and contributors Copyright (C) 2019 Nuke.YKT This file is part of NBlood. NBlood is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License version 2 as published by the Free Software Foundation. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ //------------------------------------------------------------------------- #include "ns.h" // Must come before everything else! #include "build.h" #include "compat.h" #include "mouse.h" #include "common_game.h" #include "blood.h" #include "config.h" #include "gamemenu.h" #include "globals.h" #include "inifile.h" #include "levels.h" #include "menu.h" #include "qav.h" #include "resource.h" #include "view.h" #include "c_bind.h" BEGIN_BLD_NS CMenuTextMgr gMenuTextMgr; CGameMenuMgr gGameMenuMgr; extern CGameMenuItemPicCycle itemSorryPicCycle; extern CGameMenuItemQAV itemBloodQAV; CMenuTextMgr::CMenuTextMgr() { at0 = -1; } static char buffer[21][45]; void CMenuTextMgr::DrawText(const char *pString, int nFont, int x, int y, int nShade, int nPalette, bool shadow ) { viewDrawText(nFont, pString, x, y, nShade, nPalette, 0, shadow); } void CMenuTextMgr::GetFontInfo(int nFont, const char *pString, int *pXSize, int *pYSize) { if (nFont < 0 || nFont >= 5) return; viewGetFontInfo(nFont, pString, pXSize, pYSize); } bool CGameMenuMgr::m_bInitialized = false; bool CGameMenuMgr::m_bActive = false; bool CGameMenuMgr::m_bScanning = false; CGameMenuMgr::CGameMenuMgr() { dassert(!m_bInitialized); m_bInitialized = true; Clear(); } CGameMenuMgr::~CGameMenuMgr() { m_bInitialized = false; Clear(); } void CGameMenuMgr::InitializeMenu(void) { if (pActiveMenu) { CGameMenuEvent event; event.at0 = kMenuEventInit; event.at2 = 0; pActiveMenu->Event(event); } } void CGameMenuMgr::DeInitializeMenu(void) { if (pActiveMenu) { CGameMenuEvent event; event.at0 = kMenuEventDeInit; event.at2 = 0; pActiveMenu->Event(event); } } bool CGameMenuMgr::Push(CGameMenu *pMenu, int nItem) { if (nMenuPointer == 0) { m_mouselastactivity = -M_MOUSETIMEOUT; m_mousewake_watchpoint = 0; mouseLockToWindow(0); mouseMoveToCenter(); mouseReadAbs(&m_prevmousepos, &g_mouseAbs); } dassert(pMenu != NULL); if (nMenuPointer == 8) return false; pActiveMenu = pMenuStack[nMenuPointer] = pMenu; nMenuPointer++; if (nItem >= 0) pMenu->SetFocusItem(nItem); m_bActive = true; gInputMode = kInputMenu; InitializeMenu(); m_menuchange_watchpoint = 1; m_mousecaught = 1; return true; } void CGameMenuMgr::Pop(void) { if (nMenuPointer > 0) { DeInitializeMenu(); nMenuPointer--; if (nMenuPointer == 0) Deactivate(); else pActiveMenu = pMenuStack[nMenuPointer-1]; m_menuchange_watchpoint = 1; } m_mousecaught = 1; } void CGameMenuMgr::PostPop(void) { m_postPop = true; } void CGameMenuMgr::Draw(void) { if (pActiveMenu) { pActiveMenu->Draw(); viewUpdatePages(); } if (m_postPop) { Pop(); m_postPop = false; } int32_t mousestatus = mouseReadAbs(&m_mousepos, &g_mouseAbs); if (mousestatus && inputState.mouseClickState() == MOUSE_PRESSED) m_mousedownpos = m_mousepos; int16_t mousetile = 1043; // red arrow if (tilesiz[mousetile].x > 0 && mousestatus) { if (!MOUSEACTIVECONDITION) m_mousewake_watchpoint = 1; if (MOUSEACTIVECONDITIONAL(inputState.mouseAdvanceClickState()) || m_mousepos.x != m_prevmousepos.x || m_mousepos.y != m_prevmousepos.y) { m_prevmousepos = m_mousepos; m_mouselastactivity = (int)totalclock; } else m_mousewake_watchpoint = 0; m_mousecaught = 0; } else { m_mouselastactivity = -M_MOUSETIMEOUT; m_mousewake_watchpoint = 0; } // Display the mouse cursor, except on touch devices. if (MOUSEACTIVECONDITION) { vec2_t cursorpos = { m_mousepos.x + (7 << 16), m_mousepos.y + (6 << 16) }; if ((unsigned) mousetile < MAXTILES) { int32_t scale = 65536; int16_t rotate = 768; uint32_t stat = 2|4|8; int8_t alpha = MOUSEALPHA; //CURSORALPHA; rotatesprite_fs_alpha(cursorpos.x, cursorpos.y, scale, rotate, mousetile, 0, 0, stat, alpha); } } else inputState.clearMouseClickState(); } void CGameMenuMgr::Clear(void) { pActiveMenu = NULL; memset(pMenuStack, 0, sizeof(pMenuStack)); nMenuPointer = 0; m_postPop = false; } void CGameMenuMgr::Process(void) { if (!pActiveMenu) return; if (m_menuchange_watchpoint > 0) m_menuchange_watchpoint++; CGameMenuEvent event; event.at0 = 0; event.at2 = 0; char key; if (!pActiveMenu->MouseEvent(event) && (key = inputState.keyGetScan()) != 0 ) { inputState.keyFlushScans(); inputState.keyFlushChars(); event.at2 = key; switch (key) { case sc_Escape: event.at0 = kMenuEventEscape; break; case sc_Tab: if (inputState.ShiftPressed()) event.at0 = kMenuEventUp; else event.at0 = kMenuEventDown; break; case sc_UpArrow: case sc_kpad_8: event.at0 = kMenuEventUp; gGameMenuMgr.m_mouselastactivity = -M_MOUSETIMEOUT; break; case sc_DownArrow: case sc_kpad_2: event.at0 = kMenuEventDown; gGameMenuMgr.m_mouselastactivity = -M_MOUSETIMEOUT; break; case sc_Enter: case sc_kpad_Enter: event.at0 = kMenuEventEnter; break; case sc_Space: event.at0 = kMenuEventSpace; break; case sc_LeftArrow: case sc_kpad_4: event.at0 = kMenuEventLeft; break; case sc_RightArrow: case sc_kpad_6: event.at0 = kMenuEventRight; break; case sc_Delete: case sc_kpad_Period: event.at0 = kMenuEventDelete; break; case sc_BackSpace: event.at0 = kMenuEventBackSpace; break; default: event.at0 = kMenuEventKey; break; } } if (pActiveMenu->Event(event)) Pop(); if (m_menuchange_watchpoint >= 3) m_menuchange_watchpoint = 0; } void CGameMenuMgr::Deactivate(void) { Clear(); inputState.keyFlushScans(); inputState.keyFlushChars(); m_bActive = false; mouseLockToWindow(1); gInputMode = kInputGame; } bool CGameMenuMgr::MouseOutsideBounds(vec2_t const * const pos, const int32_t x, const int32_t y, const int32_t width, const int32_t height) { return pos->x < x || pos->x >= x + width || pos->y < y || pos->y >= y + height; } CGameMenu::CGameMenu() { m_nItems = 0; m_nFocus = at8 = -1; atc = 0; } CGameMenu::CGameMenu(int unk) { m_nItems = 0; m_nFocus = at8 = -1; atc = unk; } CGameMenu::~CGameMenu() { if (!atc) return; for (int i = 0; i < m_nItems; i++) { if (pItemList[i] != &itemBloodQAV && pItemList[i] != &itemSorryPicCycle) delete pItemList[i]; } } void CGameMenu::InitializeItems(CGameMenuEvent &event) { for (int i = 0; i < m_nItems; i++) { pItemList[i]->Event(event); } } void CGameMenu::Draw(void) { for (int i = 0; i < m_nItems; i++) { if (pItemList[i]->pPreDrawCallback) pItemList[i]->pPreDrawCallback(pItemList[i]); if (i == m_nFocus || (i != m_nFocus && !pItemList[i]->bNoDraw)) pItemList[i]->Draw(); } } bool CGameMenu::Event(CGameMenuEvent &event) { if (m_nItems <= 0) return true; switch (event.at0) { case kMenuEventInit: case kMenuEventDeInit: if (at8 >= 0) m_nFocus = at8; InitializeItems(event); return false; } if (m_nFocus < 0) return true; return pItemList[m_nFocus]->Event(event); } void CGameMenu::Add(CGameMenuItem *pItem, bool active) { dassert(pItem != NULL); dassert(m_nItems < kMaxGameMenuItems); pItemList[m_nItems] = pItem; pItem->pMenu = this; if (active) m_nFocus = at8 = m_nItems; m_nItems++; } void CGameMenu::SetFocusItem(int nItem) { dassert(nItem >= 0 && nItem < m_nItems && nItem < kMaxGameMenuItems); if (CanSelectItem(nItem)) m_nFocus = at8 = nItem; } void CGameMenu::SetFocusItem(CGameMenuItem *pItem) { for (int i = 0; i < m_nItems; i++) if (pItemList[i] == pItem) { SetFocusItem(i); break; } } bool CGameMenu::CanSelectItem(int nItem) { dassert(nItem >= 0 && nItem < m_nItems && nItem < kMaxGameMenuItems); return pItemList[nItem]->bCanSelect && pItemList[nItem]->bEnable; } void CGameMenu::FocusPrevItem(void) { dassert(m_nFocus >= -1 && m_nFocus < m_nItems && m_nFocus < kMaxGameMenuItems); int t = m_nFocus; do { m_nFocus--; if (m_nFocus < 0) m_nFocus += m_nItems; if (CanSelectItem(m_nFocus)) break; } while(t != m_nFocus); } void CGameMenu::FocusNextItem(void) { dassert(m_nFocus >= -1 && m_nFocus < m_nItems && m_nFocus < kMaxGameMenuItems); int t = m_nFocus; do { m_nFocus++; if (m_nFocus >= m_nItems) m_nFocus = 0; if (CanSelectItem(m_nFocus)) break; } while(t != m_nFocus); } bool CGameMenu::IsFocusItem(CGameMenuItem *pItem) { if (m_nFocus < 0) return false; dassert(m_nFocus >= 0 && m_nFocus < m_nItems && m_nFocus < kMaxGameMenuItems); return pItemList[m_nFocus] == pItem; } bool CGameMenu::MouseEvent(CGameMenuEvent &event) { if (m_nItems <= 0 || m_nFocus < 0) return true; return pItemList[m_nFocus]->MouseEvent(event); } CGameMenuItem::CGameMenuItem() { m_pzText = NULL; m_nX = m_nY = m_nWidth = 0; bCanSelect = 1; bEnable = 1; m_nFont = -1; pMenu = NULL; bNoDraw = 0; pPreDrawCallback = NULL; } CGameMenuItem::~CGameMenuItem() { } bool CGameMenuItem::Event(CGameMenuEvent &event) { switch (event.at0) { case kMenuEventEscape: return true; case kMenuEventUp: pMenu->FocusPrevItem(); break; case kMenuEventDown: pMenu->FocusNextItem(); break; } return false; } bool CGameMenuItem::MouseEvent(CGameMenuEvent &event) { event.at0 = kMenuEventNone; if (MOUSEINACTIVECONDITIONAL(inputState.MouseGetButtons()&LEFT_MOUSE)) { event.at0 = kMenuEventEnter; inputState.MouseClearButton(LEFT_MOUSE); } else if (inputState.MouseGetButtons()&RIGHT_MOUSE) { event.at0 = kMenuEventEscape; inputState.MouseClearButton(RIGHT_MOUSE); } else if (inputState.MouseGetButtons()&WHEELUP_MOUSE) { inputState.MouseClearButton(WHEELUP_MOUSE); event.at0 = kMenuEventUp; } else if (inputState.MouseGetButtons()&WHEELDOWN_MOUSE) { inputState.MouseClearButton(WHEELDOWN_MOUSE); event.at0 = kMenuEventDown; } return event.at0 != kMenuEventNone; } CGameMenuItemText::CGameMenuItemText() { m_pzText = 0; bEnable = 0; } CGameMenuItemText::CGameMenuItemText(const char *a1, int a2, int a3, int a4, int a5) { m_nWidth = 0; m_pzText = a1; m_nFont = a2; m_nX = a3; m_nY = a4; at20 = a5; bEnable = 0; } void CGameMenuItemText::Draw(void) { if (m_pzText) { int width; int x = m_nX; switch (at20) { case 1: gMenuTextMgr.GetFontInfo(m_nFont, m_pzText, &width, NULL); x = m_nX-width/2; break; case 2: gMenuTextMgr.GetFontInfo(m_nFont, m_pzText, &width, NULL); x = m_nX-width; break; } gMenuTextMgr.DrawText(m_pzText,m_nFont, x, m_nY, -128, 0, false); } } CGameMenuItemTitle::CGameMenuItemTitle() { m_pzText = 0; bEnable = 0; } CGameMenuItemTitle::CGameMenuItemTitle(const char *a1, int a2, int a3, int a4, int a5) { m_nWidth = 0; m_pzText = a1; m_nFont = a2; m_nX = a3; m_nY = a4; at20 = a5; bEnable = 0; } void CGameMenuItemTitle::Draw(void) { if (m_pzText) { int height; gMenuTextMgr.GetFontInfo(m_nFont, NULL, NULL, &height); if (at20 >= 0) rotatesprite(320<<15, m_nY<<16, 65536, 0, at20, -128, 0, 78, 0, 0, xdim-1, ydim-1); viewDrawText(m_nFont, m_pzText, m_nX, m_nY-height/2, -128, 0, 1, false); } } CGameMenuItemZBool::CGameMenuItemZBool() { at20 = false; m_pzText = 0; at21 = "On"; at25 = "Off"; } CGameMenuItemZBool::CGameMenuItemZBool(const char *a1, int a2, int a3, int a4, int a5, bool a6, void(*a7)(CGameMenuItemZBool *), const char *a8, const char *a9) { m_pzText = a1; m_nFont = a2; m_nX = a3; m_nY = a4; m_nWidth = a5; at20 = a6; at29 = a7; if (!a8) at21 = "On"; else at21 = a8; if (!a9) at25 = "Off"; else at25 = a9; } void CGameMenuItemZBool::Draw(void) { int shade = bEnable ? 32 : 48; int pal = bEnable ? 0 : 5; if (pMenu->IsFocusItem(this)) shade = 32-((int)totalclock&63); if (m_pzText) gMenuTextMgr.DrawText(m_pzText, m_nFont, m_nX, m_nY, shade, pal, false); const char *value = at20 ? at21 : at25; int width, height; gMenuTextMgr.GetFontInfo(m_nFont, value, &width, &height); gMenuTextMgr.DrawText(value, m_nFont, m_nWidth-1+m_nX-width, m_nY, shade, pal, false); int mx = m_nX<<16; int my = m_nY<<16; int mw = m_nWidth<<16; int mh = height<<16; if (bEnable && MOUSEACTIVECONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousepos, mx, my, mw, mh))) { if (MOUSEWATCHPOINTCONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_prevmousepos, mx, my, mw, mh))) { pMenu->SetFocusItem(this); } if (!gGameMenuMgr.m_mousecaught && inputState.mouseClickState() == MOUSE_RELEASED && !gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousedownpos, mx, my, mw, mh)) { pMenu->SetFocusItem(this); CGameMenuEvent event = { kMenuEventEnter, 0 }; gGameMenuMgr.m_mousecaught = 1; if (Event(event)) gGameMenuMgr.PostPop(); } } } bool CGameMenuItemZBool::Event(CGameMenuEvent &event) { switch (event.at0) { case kMenuEventEnter: case kMenuEventSpace: at20 = !at20; if (at29) at29(this); return false; } return CGameMenuItem::Event(event); } CGameMenuItemChain::CGameMenuItemChain() { m_pzText = NULL; at24 = NULL; at28 = -1; at2c = NULL; at30 = 0; } CGameMenuItemChain::CGameMenuItemChain(const char *a1, int a2, int a3, int a4, int a5, int a6, CGameMenu *a7, int a8, void(*a9)(CGameMenuItemChain *), int a10) { m_pzText = a1; m_nFont = a2; m_nX = a3; m_nY = a4; m_nWidth = a5; at20 = a6; at24 = a7; at28 = a8; at2c = a9; at30 = a10; } void CGameMenuItemChain::Draw(void) { if (!m_pzText) return; int shade = bEnable ? 32 : 48; int pal = bEnable ? 0 : 5; if (pMenu->IsFocusItem(this)) shade = 32-((int)totalclock&63); int width, height; int x = m_nX; int y = m_nY; gMenuTextMgr.GetFontInfo(m_nFont, m_pzText, &width, &height); switch (at20) { case 1: x = m_nX+m_nWidth/2-width/2; break; case 2: x = m_nX+m_nWidth-1-width; break; case 0: default: break; } gMenuTextMgr.DrawText(m_pzText, m_nFont, x, m_nY, shade, pal, true); if (bEnable && MOUSEACTIVECONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousepos, x<<16, y<<16, width<<16, height<<16))) { if (MOUSEWATCHPOINTCONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_prevmousepos, x<<16, y<<16, width<<16, height<<16))) { pMenu->SetFocusItem(this); } if (!gGameMenuMgr.m_mousecaught && inputState.mouseClickState() == MOUSE_RELEASED && !gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousedownpos, x<<16, y<<16, width<<16, height<<16)) { pMenu->SetFocusItem(this); CGameMenuEvent event = { kMenuEventEnter, 0 }; gGameMenuMgr.m_mousecaught = 1; if (Event(event)) gGameMenuMgr.PostPop(); } } } bool CGameMenuItemChain::Event(CGameMenuEvent &event) { switch (event.at0) { case kMenuEventEnter: if (at2c) at2c(this); if (at24) gGameMenuMgr.Push(at24, at28); return false; } return CGameMenuItem::Event(event); } CGameMenuItem7EA1C::CGameMenuItem7EA1C() { m_pzText = NULL; at24 = NULL; at28 = -1; at2c = NULL; at30 = 0; at34 = NULL; at38[0] = 0; at48[0] = 0; } CGameMenuItem7EA1C::CGameMenuItem7EA1C(const char *a1, int a2, int a3, int a4, int a5, const char *a6, const char *a7, int a8, int a9, void(*a10)(CGameMenuItem7EA1C *), int a11) { m_pzText = a1; m_nFont = a2; m_nX = a3; m_nY = a4; m_nWidth = a5; at20 = a8; at28 = a9; at2c = a10; at30 = a11; strncpy(at38, a6, 15); strncpy(at48, a7, 15); } void CGameMenuItem7EA1C::Draw(void) { if (!m_pzText) return; int shade = bEnable ? 32 : 48; int pal = bEnable ? 0 : 5; if (pMenu->IsFocusItem(this)) shade = 32-((int)totalclock&63); int width; int x = m_nX; switch (at20) { case 1: gMenuTextMgr.GetFontInfo(m_nFont, m_pzText, &width, NULL); x = m_nX+m_nWidth/2-width/2; break; case 2: gMenuTextMgr.GetFontInfo(m_nFont, m_pzText, &width, NULL); x = m_nX+m_nWidth-1-width; break; case 0: default: break; } gMenuTextMgr.DrawText(m_pzText, m_nFont, x, m_nY, shade, pal, true); } void CGameMenuItem7EA1C::Setup(void) { if (!at34 || !at24) return; if (!at34->SectionExists(at48)) return; const char *title = at34->GetKeyString(at48, "Title", at48); at24->Add(new CGameMenuItemTitle(title, 1, 160, 20, 2038), false); at24->Add(&itemSorryPicCycle, true); int y = 40; for (int i = 0; i < 21; i++) { sprintf(buffer[i], "Line%d", i+1); if (!at34->KeyExists(at48, buffer[i])) break; const char *line = at34->GetKeyString(at48, buffer[i], NULL); if (line) { if (*line == 0) { y += 10; continue; } at24->Add(new CGameMenuItemText(line, 1, 160, y, 1), false); y += 20; } } at24->Add(&itemBloodQAV, false); } bool CGameMenuItem7EA1C::Event(CGameMenuEvent &event) { switch (event.at0) { case kMenuEventEnter: { if (at2c) at2c(this); if (at24) delete at24; at24 = new CGameMenu(1); /* DICTNODE *pRes = gGuiRes.Lookup(at38, "MNU"); if (pRes) { at34 = new IniFile(gGuiRes.Load(pRes)); Setup(); } */ if (at24) gGameMenuMgr.Push(at24, at28); return false; } case kMenuEventDeInit: if (at34) { delete at34; at34 = NULL; } if (at24) { delete at24; at24 = NULL; } return false; } return CGameMenuItem::Event(event); } CGameMenuItem7EE34::CGameMenuItem7EE34() { m_pzText = NULL; at28 = NULL; at20 = -1; at2c = NULL; } CGameMenuItem7EE34::CGameMenuItem7EE34(const char *a1, int a2, int a3, int a4, int a5, int a6) { m_pzText = NULL; at28 = NULL; at20 = -1; at2c = NULL; m_nFont = a2; m_nX = a3; m_pzText = a1; m_nY = a4; m_nWidth = a5; at24 = a6; } void CGameMenuItem7EE34::Draw(void) { if (!m_pzText) return; int shade = bEnable ? 32 : 48; int pal = bEnable ? 0 : 5; if (pMenu->IsFocusItem(this)) shade = 32-((int)totalclock&63); int width; int x = m_nX; switch (at24) { case 1: gMenuTextMgr.GetFontInfo(m_nFont, m_pzText, &width, NULL); x = m_nX+m_nWidth/2-width/2; break; case 2: gMenuTextMgr.GetFontInfo(m_nFont, m_pzText, &width, NULL); x = m_nX+m_nWidth-1-width; break; case 0: default: break; } gMenuTextMgr.DrawText(m_pzText, m_nFont, x, m_nY, shade, pal, true); } extern void SetVideoModeOld(CGameMenuItemChain *pItem); void CGameMenuItem7EE34::Setup(void) { if (!at28) return; at28->Add(new CGameMenuItemTitle("Video Mode", 1, 160, 20, 2038), false); if (!at2c) { at2c = new CGameMenu(1); at2c->Add(new CGameMenuItemTitle(" Mode Change ", 1, 160, 20, 2038), false); at2c->Add(&itemSorryPicCycle, true); CGameMenuItem *pItem1 = new CGameMenuItemText("VIDEO MODE WAS SET", 1, 160, 90, 1); CGameMenuItem *pItem2 = new CGameMenuItemText("NOT ALL MODES Work correctly", 1, 160, 110, 1); CGameMenuItem *pItem3 = new CGameMenuItemText("Press ESC to exit", 3, 160, 140, 1); at2c->Add(pItem1, false); pItem1->bEnable = 0; at2c->Add(pItem2, false); pItem2->bEnable = 0; at2c->Add(pItem3, true); pItem3->bEnable = 1; at2c->Add(&itemBloodQAV, false); } sprintf(buffer[0], "640 x 480 (default)"); int y = 40; at28->Add(new CGameMenuItemChain(buffer[0], 3, 0, y, 320, 1, at2c, -1, SetVideoModeOld, validmodecnt), true); y += 20; for (int i = 0; i < validmodecnt && i < 20; i++) { sprintf(buffer[i+1], "%d x %d", validmode[i].xdim, validmode[i].ydim); at28->Add(new CGameMenuItemChain(buffer[i+1], 3, 0, y, 320, 1, at2c, -1, SetVideoModeOld, i), false); if (validmodecnt > 10) y += 7; else y += 15; } at28->Add(&itemBloodQAV, false); } bool CGameMenuItem7EE34::Event(CGameMenuEvent &event) { switch (event.at0) { case kMenuEventEnter: if (at28) delete at28; at28 = new CGameMenu(1); Setup(); if (at28) gGameMenuMgr.Push(at28, at20); return false; case kMenuEventDeInit: if (at28) { delete at28; at28 = 0; } return false; } return CGameMenuItem::Event(event); } CGameMenuItemChain7F2F0::CGameMenuItemChain7F2F0() { at34 = -1; } CGameMenuItemChain7F2F0::CGameMenuItemChain7F2F0(char *a1, int a2, int a3, int a4, int a5, int a6, CGameMenu *a7, int a8, void(*a9)(CGameMenuItemChain *), int a10) : CGameMenuItemChain(a1,a2,a3,a4,a5,a6,a7,a8,a9,a10) { at34 = a10; } bool CGameMenuItemChain7F2F0::Event(CGameMenuEvent &event) { switch (event.at0) { case kMenuEventEnter: if (at34 > -1) gGameOptions.nEpisode = at34; return CGameMenuItemChain::Event(event); } return CGameMenuItem::Event(event); } CGameMenuItemBitmap::CGameMenuItemBitmap() { m_pzText = NULL; } CGameMenuItemBitmap::CGameMenuItemBitmap(const char *a1, int a2, int a3, int a4, int a5) { m_pzText = a1; m_nFont = a2; m_nX = a3; m_nY = a4; at20 = a5; } void CGameMenuItemBitmap::Draw(void) { int shade = bEnable ? 32 : 48; int pal = bEnable ? 0 : 5; if (bEnable && pMenu->IsFocusItem(this)) shade = 32-((int)totalclock&63); int x = m_nX; int y = m_nY; if (m_pzText) { int height; gMenuTextMgr.DrawText(m_pzText, m_nFont, x, y, shade, pal, false); gMenuTextMgr.GetFontInfo(m_nFont, NULL, NULL, &height); y += height + 2; } rotatesprite(x<<15,y<<15, 65536, 0, at20, 0, 0, 82, 0, 0, xdim-1,ydim-1); } bool CGameMenuItemBitmap::Event(CGameMenuEvent &event) { if (bEnable && pMenu->IsFocusItem(this)) pMenu->FocusNextItem(); return CGameMenuItem::Event(event); } CGameMenuItemBitmapLS::CGameMenuItemBitmapLS() { m_pzText = NULL; } CGameMenuItemBitmapLS::CGameMenuItemBitmapLS(const char *a1, int a2, int a3, int a4, int a5) { at24 = -1; m_pzText = a1; m_nFont = a2; m_nX = a3; m_nY = a4; at28 = a5; } void CGameMenuItemBitmapLS::Draw(void) { int shade = bEnable ? 32 : 48; int pal = bEnable ? 0 : 5; if (bEnable && pMenu->IsFocusItem(this)) shade = 32-((int)totalclock&63); int x = m_nX; int y = m_nY; if (m_pzText) { int height; gMenuTextMgr.DrawText(m_pzText, m_nFont, x, y, shade, pal, false); gMenuTextMgr.GetFontInfo(m_nFont, NULL, NULL, &height); y += height + 2; } char stat; int16_t ang; int picnum; if (at24 == -1) { stat = 66; ang = 0; picnum = at28; } else { ang = 512; stat = 70; picnum = at24; } rotatesprite(200<<15,215<<15,32768, ang, picnum, 0, 0, stat, 0, 0, xdim-1, ydim-1); } bool CGameMenuItemBitmapLS::Event(CGameMenuEvent &event) { if (bEnable && pMenu->IsFocusItem(this)) pMenu->FocusNextItem(); return CGameMenuItem::Event(event); } CGameMenuItemKeyList::CGameMenuItemKeyList() { m_pzText = NULL; m_nFont = 3; m_nX = 0; m_nY = 0; nRows = 0; nTopDelta = 0; nFocus = 0; nGameFuncs = 0; bScan = false; } CGameMenuItemKeyList::CGameMenuItemKeyList(const char *a1, int a2, int a3, int a4, int a5, int a6, int a7, void(*a8)(CGameMenuItemKeyList *)) { nTopDelta = 0; nFocus = 0; bScan = false; m_pzText = a1; m_nFont = a2; m_nX = a3; m_nY = a4; m_nWidth = a5; nRows = a6; pCallback = a8; nGameFuncs = a7; } void CGameMenuItemKeyList::Scan(void) { inputState.keyFlushChars(); inputState.keyFlushScans(); inputState.ClearKeysDown(); bScan = true; } void CGameMenuItemKeyList::Draw(void) { char buffer[40]; int width, height; int shade; gMenuTextMgr.GetFontInfo(m_nFont, NULL, NULL, &height); int y = m_nY; int k = nFocus - nTopDelta; int nNewFocus = nFocus; bool bClick = false; for (int i = 0; i < nRows; i++, y += height, k++) { auto keys = Bindings.GetKeysForCommand(buttonMap.GetButtonName(k)); FString text = C_NameKeys(keys.Data(), std::min(keys.Size(), 2u)); sprintf(buffer, "%s", buttonMap.GetButtonName(k)); if (k == nFocus) { shade = 32; if (pMenu->IsFocusItem(this)) shade = 32-((int)totalclock&63); viewDrawText(3, buffer, m_nX, y, shade, 0, 0, false); const char *sVal; if (bScan && ((int)totalclock & 32)) sVal = "____"; else sVal = text; gMenuTextMgr.GetFontInfo(m_nFont, sVal, &width, 0); viewDrawText(m_nFont, sVal, m_nX+m_nWidth-1-width, y, shade, 0, 0, false); } else { viewDrawText(3, buffer, m_nX, y, 24, 0, 0, false); gMenuTextMgr.GetFontInfo(m_nFont, text, &width, 0); viewDrawText(m_nFont, text, m_nX+m_nWidth-1-width, y, 24, 0, 0, false); } int mx = m_nX<<16; int my = y<<16; int mw = m_nWidth<<16; int mh = height<<16; if (!bScan && bEnable && MOUSEACTIVECONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousepos, mx, my, mw, mh))) { if (MOUSEWATCHPOINTCONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_prevmousepos, mx, my, mw, mh))) { nNewFocus = k; } if (!gGameMenuMgr.m_mousecaught && inputState.mouseClickState() == MOUSE_RELEASED && !gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousedownpos, mx, my, mw, mh)) { nNewFocus = k; bClick = true; } } } nTopDelta += nNewFocus-nFocus; nFocus = nNewFocus; if (bClick) { CGameMenuEvent event = { kMenuEventEnter, 0 }; gGameMenuMgr.m_mousecaught = 1; if (Event(event)) gGameMenuMgr.PostPop(); } } bool CGameMenuItemKeyList::Event(CGameMenuEvent &event) { if (bScan) { if (inputState.GetLastScanCode() && inputState.GetLastScanCode() != sc_Pause) { if (inputState.keyBufferWaiting()) inputState.keyGetChar(); Bindings.SetBind(inputState.GetLastScanCode(), buttonMap.GetButtonName(nFocus)); inputState.keyFlushChars(); inputState.keyFlushScans(); inputState.ClearKeysDown(); inputState.keyFlushScans(); inputState.keyFlushChars(); bScan = 0; } return false; } switch (event.at0) { case kMenuEventUp: if (event.at2 == sc_Tab || nFocus == 0) { pMenu->FocusPrevItem(); return false; } nFocus--; if (nTopDelta > 0) nTopDelta--; return false; case kMenuEventDown: if (event.at2 == sc_Tab || nFocus == nGameFuncs-1) { pMenu->FocusNextItem(); return false; } nFocus++; if (nTopDelta+1 < nRows) nTopDelta++; return false; case kMenuEventEnter: if (pCallback) pCallback(this); Scan(); return false; case kMenuEventBackSpace: case kMenuEventDelete: Bindings.UnbindACommand(buttonMap.GetButtonName(nFocus)); return false; case kMenuEventScrollUp: if (nFocus-nTopDelta > 0) { nTopDelta++; if (nTopDelta>0) { nFocus--; nTopDelta--; } } return false; case kMenuEventScrollDown: if (nFocus-nTopDelta+nRows < nGameFuncs) { nTopDelta--; if (nTopDelta+1 < nRows) { nFocus++; nTopDelta++; } } return false; } return CGameMenuItem::Event(event); } bool CGameMenuItemKeyList::MouseEvent(CGameMenuEvent &event) { event.at0 = kMenuEventNone; if (MOUSEACTIVECONDITIONAL(inputState.MouseGetButtons()&WHEELUP_MOUSE)) { gGameMenuMgr.m_mouselastactivity = (int)totalclock; inputState.MouseClearButton(WHEELUP_MOUSE); event.at0 = kMenuEventScrollUp; } else if (MOUSEACTIVECONDITIONAL(inputState.MouseGetButtons()&WHEELDOWN_MOUSE)) { gGameMenuMgr.m_mouselastactivity = (int)totalclock; inputState.MouseClearButton(WHEELDOWN_MOUSE); event.at0 = kMenuEventScrollDown; } else return CGameMenuItem::MouseEvent(event); return event.at0 != kMenuEventNone; } CGameMenuItemSlider::CGameMenuItemSlider() { m_pzText = NULL; m_nFont = -1; m_nX = 0; m_nY = 0; nValue = 0; nRangeLow = 0; nStep = 0; pCallback = NULL; pValue = NULL; nSliderTile = 2204; nCursorTile = 2028; nShowValue = kMenuSliderNone; } CGameMenuItemSlider::CGameMenuItemSlider(const char *_pzText, int _nFont, int _nX, int _nY, int _nWidth, int _nValue, int _nRangeLow, int _nRangeHigh, int _nStep, void(*_pCallback)(CGameMenuItemSlider *), int _nSliderTile, int _nCursorTile, int _nShowValue) { m_pzText = _pzText; m_nFont = _nFont; m_nX = _nX; m_nY = _nY; m_nWidth = _nWidth; nRangeLow = _nRangeLow; nRangeHigh = _nRangeHigh; nStep = _nStep; nValue = ClipRange(_nValue, nRangeLow, nRangeHigh); pCallback = _pCallback; nSliderTile = 2204; nCursorTile = 2028; if (_nSliderTile >= 0) nSliderTile = _nSliderTile; if (_nCursorTile >= 0) nCursorTile = _nCursorTile; nShowValue = _nShowValue; } CGameMenuItemSlider::CGameMenuItemSlider(const char *_pzText, int _nFont, int _nX, int _nY, int _nWidth, int *pnValue, int _nRangeLow, int _nRangeHigh, int _nStep, void(*_pCallback)(CGameMenuItemSlider *), int _nSliderTile, int _nCursorTile, int _nShowValue) { m_pzText = _pzText; m_nFont = _nFont; m_nX = _nX; m_nY = _nY; m_nWidth = _nWidth; nRangeLow = _nRangeLow; nRangeHigh = _nRangeHigh; nStep = _nStep; dassert(pnValue != NULL); pValue = pnValue; nValue = ClipRange(*pnValue, nRangeLow, nRangeHigh); pCallback = _pCallback; nSliderTile = 2204; nCursorTile = 2028; if (_nSliderTile >= 0) nSliderTile = _nSliderTile; if (_nCursorTile >= 0) nCursorTile = _nCursorTile; nShowValue = _nShowValue; } void CGameMenuItemSlider::Draw(void) { char buffer[16]; int height; nValue = pValue ? *pValue : nValue; gMenuTextMgr.GetFontInfo(m_nFont, NULL, NULL, &height); int shade = bEnable ? 32 : 48; int shade2 = bEnable ? 0 : 16; int pal = bEnable ? 0 : 5; if (pMenu->IsFocusItem(this)) shade = 32-((int)totalclock&63); if (m_pzText) gMenuTextMgr.DrawText(m_pzText, m_nFont, m_nX, m_nY, shade, pal, false); int sliderX = m_nX+m_nWidth-1-tilesiz[nSliderTile].x/2; rotatesprite(sliderX<<16, (m_nY+height/2)<<16, 65536, 0, nSliderTile, shade2, pal, 10, 0, 0, xdim-1, ydim-1); int nRange = nRangeHigh - nRangeLow; dassert(nRange > 0); int value = nValue - nRangeLow; int width = tilesiz[nSliderTile].x-8; int cursorX = sliderX + ksgn(nStep)*(value * width / nRange - width / 2); rotatesprite(cursorX<<16, (m_nY+height/2)<<16, 65536, 0, nCursorTile, shade2, pal, 10, 0, 0, xdim-1, ydim-1); buffer[0] = 0; switch (nShowValue) { case kMenuSliderNone: break; case kMenuSliderValue: sprintf(buffer, "%i ", nValue); break; case kMenuSliderPercent: sprintf(buffer, "%i%% ", roundscale(value, 100, nRange)); break; case kMenuSliderQ16: snprintf(buffer, 16, "%.3f ", nValue/65536.f); break; } int valueWidth; gMenuTextMgr.GetFontInfo(m_nFont, buffer, &valueWidth, NULL); int valueX = m_nX+m_nWidth-1-tilesiz[nSliderTile].x-valueWidth; gMenuTextMgr.DrawText(buffer, m_nFont, valueX, m_nY, 32, 0, false); int mx = m_nX; int my = m_nY; int mw = m_nWidth; int mh = height; if (height < tilesiz[nSliderTile].y) { my -= (tilesiz[nSliderTile].y-height)/2; height = tilesiz[nSliderTile].y; } mx <<= 16; my <<= 16; mw <<= 16; mh <<= 16; if (bEnable && MOUSEACTIVECONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousepos, mx, my, mw, mh))) { if (MOUSEWATCHPOINTCONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_prevmousepos, mx, my, mw, mh))) { pMenu->SetFocusItem(this); } if (!gGameMenuMgr.m_mousecaught && (inputState.mouseClickState() == MOUSE_PRESSED || inputState.mouseClickState() == MOUSE_HELD)) { pMenu->SetFocusItem(this); int sliderx = m_nX+m_nWidth-1-tilesiz[nSliderTile].x; int sliderwidth = tilesiz[nSliderTile].x; int regionwidth = sliderwidth-8; int regionx = sliderx+(sliderwidth-regionwidth)/2; sliderx <<= 16; sliderwidth <<= 16; regionwidth <<= 16; regionx <<= 16; // region between the x-midline of the slidepoint at the extremes slides proportionally if (!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousepos, regionx, my, regionwidth, mh)) { int dx = (gGameMenuMgr.m_mousepos.x - (regionx+regionwidth/2))*ksgn(nStep); nValue = nRangeLow + roundscale(dx+regionwidth/2, nRange, regionwidth); nValue = ClipRange(nValue, nRangeLow, nRangeHigh); if (pCallback) pCallback(this); gGameMenuMgr.m_mousecaught = 1; } // region outside the x-midlines clamps to the extremes else if (!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousepos, sliderx, my, sliderwidth, mh)) { if ((gGameMenuMgr.m_mousepos.x-(regionx+regionwidth/2))*ksgn(nStep) > 0) nValue = nRangeHigh; else nValue = nRangeLow; if (pCallback) pCallback(this); gGameMenuMgr.m_mousecaught = 1; } } } } bool CGameMenuItemSlider::Event(CGameMenuEvent &event) { nValue = pValue ? *pValue : nValue; switch (event.at0) { case kMenuEventUp: pMenu->FocusPrevItem(); return false; case kMenuEventDown: pMenu->FocusNextItem(); return false; case kMenuEventLeft: if (nStep > 0) nValue = DecBy(nValue, nStep); else nValue = IncBy(nValue, -nStep); nValue = ClipRange(nValue, nRangeLow, nRangeHigh); if (pCallback) pCallback(this); return false; case kMenuEventRight: if (nStep >= 0) nValue = IncBy(nValue, nStep); else nValue = DecBy(nValue, -nStep); nValue = ClipRange(nValue, nRangeLow, nRangeHigh); if (pCallback) pCallback(this); return false; case kMenuEventEnter: if (pCallback) pCallback(this); return false; } return CGameMenuItem::Event(event); } bool CGameMenuItemSlider::MouseEvent(CGameMenuEvent &event) { event.at0 = kMenuEventNone; if (MOUSEINACTIVECONDITIONAL((inputState.MouseGetButtons()&LEFT_MOUSE) && (inputState.MouseGetButtons()&WHEELUP_MOUSE))) { inputState.MouseClearButton(WHEELUP_MOUSE); event.at0 = kMenuEventLeft; } else if (MOUSEINACTIVECONDITIONAL((inputState.MouseGetButtons()&LEFT_MOUSE) && (inputState.MouseGetButtons()&WHEELDOWN_MOUSE))) { inputState.MouseClearButton(WHEELDOWN_MOUSE); event.at0 = kMenuEventRight; } else if (inputState.MouseGetButtons()&RIGHT_MOUSE) { inputState.MouseClearButton(RIGHT_MOUSE); event.at0 = kMenuEventEscape; } else if (inputState.MouseGetButtons()&WHEELUP_MOUSE) { inputState.MouseClearButton(WHEELUP_MOUSE); inputState.MouseClearButton(LEFT_MOUSE); event.at0 = kMenuEventUp; } else if (inputState.MouseGetButtons()&WHEELDOWN_MOUSE) { inputState.MouseClearButton(WHEELDOWN_MOUSE); inputState.MouseClearButton(LEFT_MOUSE); event.at0 = kMenuEventDown; } return event.at0 != kMenuEventNone; } CGameMenuItemSliderFloat::CGameMenuItemSliderFloat() { m_pzText = NULL; m_nFont = -1; m_nX = 0; m_nY = 0; fValue = 0; fRangeLow = 0; fStep = 0; pCallback = NULL; pValue = NULL; nSliderTile = 2204; nCursorTile = 2028; nShowValue = kMenuSliderNone; } CGameMenuItemSliderFloat::CGameMenuItemSliderFloat(const char *_pzText, int _nFont, int _nX, int _nY, int _nWidth, float _fValue, float _fRangeLow, float _fRangeHigh, float _fStep, void(*_pCallback)(CGameMenuItemSliderFloat *), int _nSliderTile, int _nCursorTile, int _nShowValue) { m_pzText = _pzText; m_nFont = _nFont; m_nX = _nX; m_nY = _nY; m_nWidth = _nWidth; fRangeLow = _fRangeLow; fRangeHigh = _fRangeHigh; fStep = _fStep; fValue = ClipRangeF(_fValue, fRangeLow, fRangeHigh); pCallback = _pCallback; nSliderTile = 2204; nCursorTile = 2028; if (_nSliderTile >= 0) nSliderTile = _nSliderTile; if (_nCursorTile >= 0) nCursorTile = _nCursorTile; nShowValue = _nShowValue; } CGameMenuItemSliderFloat::CGameMenuItemSliderFloat(const char *_pzText, int _nFont, int _nX, int _nY, int _nWidth, float *pnValue, float _fRangeLow, float _fRangeHigh, float _fStep, void(*_pCallback)(CGameMenuItemSliderFloat *), int _nSliderTile, int _nCursorTile, int _nShowValue) { m_pzText = _pzText; m_nFont = _nFont; m_nX = _nX; m_nY = _nY; m_nWidth = _nWidth; fRangeLow = _fRangeLow; fRangeHigh = _fRangeHigh; fStep = _fStep; dassert(pnValue != NULL); pValue = pnValue; fValue = ClipRangeF(*pnValue, fRangeLow, fRangeHigh); pCallback = _pCallback; nSliderTile = 2204; nCursorTile = 2028; if (_nSliderTile >= 0) nSliderTile = _nSliderTile; if (_nCursorTile >= 0) nCursorTile = _nCursorTile; nShowValue = _nShowValue; } void CGameMenuItemSliderFloat::Draw(void) { char buffer[16]; int height; fValue = pValue ? *pValue : fValue; gMenuTextMgr.GetFontInfo(m_nFont, NULL, NULL, &height); int shade = bEnable ? 32 : 48; int shade2 = bEnable ? 0 : 16; int pal = bEnable ? 0 : 5; if (pMenu->IsFocusItem(this)) shade = 32-((int)totalclock&63); if (m_pzText) gMenuTextMgr.DrawText(m_pzText, m_nFont, m_nX, m_nY, shade, pal, false); int sliderX = m_nX+m_nWidth-1-tilesiz[nSliderTile].x/2; rotatesprite(sliderX<<16, (m_nY+height/2)<<16, 65536, 0, nSliderTile, shade2, pal, 10, 0, 0, xdim-1, ydim-1); float fRange = fRangeHigh - fRangeLow; dassert(fRange > 0); float value = fValue - fRangeLow; int width = tilesiz[nSliderTile].x-8; int cursorX = sliderX + (int)(ksgnf(fStep)*(value * width / fRange - width / 2)); rotatesprite(cursorX<<16, (m_nY+height/2)<<16, 65536, 0, nCursorTile, shade2, pal, 10, 0, 0, xdim-1, ydim-1); buffer[0] = 0; switch (nShowValue) { case kMenuSliderNone: break; case kMenuSliderValue: snprintf(buffer, 16, "%.3f ", fValue); break; case kMenuSliderPercent: snprintf(buffer, 16, "%.3f%% ", value*100.f/fRange); break; } int valueWidth; gMenuTextMgr.GetFontInfo(m_nFont, buffer, &valueWidth, NULL); int valueX = m_nX+m_nWidth-1-tilesiz[nSliderTile].x-valueWidth; gMenuTextMgr.DrawText(buffer, m_nFont, valueX, m_nY, 32, 0, false); int mx = m_nX; int my = m_nY; int mw = m_nWidth; int mh = height; if (height < tilesiz[nSliderTile].y) { my -= (tilesiz[nSliderTile].y-height)/2; height = tilesiz[nSliderTile].y; } mx <<= 16; my <<= 16; mw <<= 16; mh <<= 16; if (bEnable && MOUSEACTIVECONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousepos, mx, my, mw, mh))) { if (MOUSEWATCHPOINTCONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_prevmousepos, mx, my, mw, mh))) { pMenu->SetFocusItem(this); } if (!gGameMenuMgr.m_mousecaught && (inputState.mouseClickState() == MOUSE_PRESSED || inputState.mouseClickState() == MOUSE_HELD)) { pMenu->SetFocusItem(this); int sliderx = m_nX+m_nWidth-1-tilesiz[nSliderTile].x; int sliderwidth = tilesiz[nSliderTile].x; int regionwidth = sliderwidth-8; int regionx = sliderx+(sliderwidth-regionwidth)/2; sliderx <<= 16; sliderwidth <<= 16; regionwidth <<= 16; regionx <<= 16; // region between the x-midline of the slidepoint at the extremes slides proportionally if (!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousepos, regionx, my, regionwidth, mh)) { int dx = (gGameMenuMgr.m_mousepos.x - (regionx+regionwidth/2))*ksgnf(fStep); fValue = fRangeLow + (dx+regionwidth/2) * fRange / regionwidth; fValue = ClipRangeF(fValue, fRangeLow, fRangeHigh); if (pCallback) pCallback(this); gGameMenuMgr.m_mousecaught = 1; } // region outside the x-midlines clamps to the extremes else if (!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousepos, sliderx, my, sliderwidth, mh)) { if ((gGameMenuMgr.m_mousepos.x-(regionx+regionwidth/2))*ksgnf(fStep) > 0) fValue = fRangeHigh; else fValue = fRangeLow; if (pCallback) pCallback(this); gGameMenuMgr.m_mousecaught = 1; } } } } bool CGameMenuItemSliderFloat::Event(CGameMenuEvent &event) { fValue = pValue ? *pValue : fValue; switch (event.at0) { case kMenuEventUp: pMenu->FocusPrevItem(); return false; case kMenuEventDown: pMenu->FocusNextItem(); return false; case kMenuEventLeft: if (fStep > 0) fValue -= fStep; else fValue += fStep; fValue = ClipRangeF(fValue, fRangeLow, fRangeHigh); if (pCallback) pCallback(this); return false; case kMenuEventRight: if (fStep >= 0) fValue += fStep; else fValue -= fStep; fValue = ClipRangeF(fValue, fRangeLow, fRangeHigh); if (pCallback) pCallback(this); return false; case kMenuEventEnter: if (pCallback) pCallback(this); return false; } return CGameMenuItem::Event(event); } CGameMenuItemZEdit::CGameMenuItemZEdit() { m_pzText = NULL; m_nFont = -1; m_nX = 0; m_nY = 0; at20 = NULL; at24 = 0; at32 = 0; at2c = 0; at30 = 0; at28 = 0; at31 = 1; } CGameMenuItemZEdit::CGameMenuItemZEdit(const char *a1, int a2, int a3, int a4, int a5, char *a6, int a7, char a8, void(*a9)(CGameMenuItemZEdit *, CGameMenuEvent *), int a10) { at30 = 0; at31 = 1; m_pzText = a1; m_nFont = a2; m_nX = a3; m_nY = a4; m_nWidth = a5; at20 = a6; at24 = a7; at32 = a8; at2c = a9; at28 = a10; } void CGameMenuItemZEdit::AddChar(char ch) { int i = strlen(at20); if (i + 1 < at24) { at20[i] = ch; at20[i + 1] = 0; } } void CGameMenuItemZEdit::BackChar(void) { int i = strlen(at20); if (i > 0) at20[i - 1] = 0; } void CGameMenuItemZEdit::Draw(void) { int height, width, textWidth = 0; gMenuTextMgr.GetFontInfo(m_nFont, NULL, &width, &height); if (at20) gMenuTextMgr.GetFontInfo(m_nFont, at20, &textWidth, NULL); int shade = bEnable ? 32 : 48; int pal = bEnable ? 0 : 5; if (pMenu->IsFocusItem(this)) shade = 32-((int)totalclock&63); if (at30) shade = -128; if (m_pzText) gMenuTextMgr.DrawText(m_pzText, m_nFont, m_nX, m_nY, shade, pal, false); int x = m_nX+m_nWidth-1-textWidth;//(at24+1)*width; if (at20 && *at20) { int width; gMenuTextMgr.GetFontInfo(m_nFont, at20, &width, NULL); int shade2; if (at32) { if (at30) shade2 = -128; else shade2 = shade; } else { if (at30) shade2 = shade; else shade2 = 32; } gMenuTextMgr.DrawText(at20, m_nFont, x, m_nY, shade2, pal, false); x += width; } if (at30 && ((int)totalclock & 32)) gMenuTextMgr.DrawText("_", m_nFont, x, m_nY, shade, 0, false); int mx = m_nX<<16; int my = m_nY<<16; int mw = m_nWidth<<16; int mh = height<<16; if (bEnable && MOUSEACTIVECONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousepos, mx, my, mw, mh))) { if (MOUSEWATCHPOINTCONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_prevmousepos, mx, my, mw, mh))) { pMenu->SetFocusItem(this); } if (!gGameMenuMgr.m_mousecaught && inputState.mouseClickState() == MOUSE_RELEASED && !gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousedownpos, mx, my, mw, mh)) { pMenu->SetFocusItem(this); CGameMenuEvent event = { kMenuEventEnter, 0 }; gGameMenuMgr.m_mousecaught = 1; if (Event(event)) gGameMenuMgr.PostPop(); } } } bool CGameMenuItemZEdit::Event(CGameMenuEvent &event) { static char buffer[256]; // Hack if (event.at2 == sc_kpad_2 || event.at2 == sc_kpad_4 || event.at2 == sc_kpad_6 || event.at2 == sc_kpad_8) event.at0 = kMenuEventKey; switch (event.at0) { case kMenuEventEscape: if (at30) { strncpy(at20, buffer, at24); at20[at24-1] = 0; at30 = 0; return false; } return true; case kMenuEventEnter: if (!at31) { if (at2c) at2c(this, &event); return false; } if (at30) { if (at2c) at2c(this, &event); at30 = 0; return false; } strncpy(buffer, at20, at24); buffer[at24-1] = 0; at30 = 1; return false; case kMenuEventBackSpace: if (at30) BackChar(); return false; case kMenuEventKey: case kMenuEventSpace: { char key; if (event.at2 < 128) { if (inputState.ShiftPressed()) key = g_keyAsciiTableShift[event.at2]; else key = g_keyAsciiTable[event.at2]; if (at30 && (isalnum(key) || ispunct(key) || isspace(key))) { AddChar(key); return false; } } return CGameMenuItem::Event(event); } case kMenuEventUp: if (at30) return false; return CGameMenuItem::Event(event); case kMenuEventDown: if (at30) return false; return CGameMenuItem::Event(event); } return CGameMenuItem::Event(event); } CGameMenuItemZEditBitmap::CGameMenuItemZEditBitmap() { m_pzText = NULL; m_nFont = -1; m_nX = 0; m_nY = 0; at20 = NULL; at24 = 0; at36 = 0; at30 = NULL; at2c = NULL; bScan = 0; at28 = 0; at37 = 0; at35 = 1; } CGameMenuItemZEditBitmap::CGameMenuItemZEditBitmap(char *a1, int a2, int a3, int a4, int a5, char *a6, int a7, char a8, void(*a9)(CGameMenuItemZEditBitmap *, CGameMenuEvent *), int a10) { at2c = NULL; bScan = 0; at35 = 1; at37 = 0; m_pzText = a1; m_nFont = a2; m_nX = a3; m_nY = a4; m_nWidth = a5; at20 = a6; at24 = a7; at36 = a8; at30 = a9; at28 = a10; } void CGameMenuItemZEditBitmap::AddChar(char ch) { int i = strlen(at20); if (i + 1 < at24) { at20[i] = ch; at20[i + 1] = 0; } } void CGameMenuItemZEditBitmap::BackChar(void) { int i = strlen(at20); if (i > 0) at20[i - 1] = 0; } void CGameMenuItemZEditBitmap::Draw(void) { int height, width; gMenuTextMgr.GetFontInfo(m_nFont, NULL, &width, &height); int shade = bEnable ? 32 : 48; int pal = bEnable ? 0 : 5; if (pMenu->IsFocusItem(this)) shade = 32-((int)totalclock&63); at2c->at24 = -1; if (bScan) shade = -128; if (m_pzText) gMenuTextMgr.DrawText(m_pzText, m_nFont, m_nX, m_nY, shade, pal, false); int x = m_nX+m_nWidth-1-(at24+1)*width; if (at20 && *at20) { int width; gMenuTextMgr.GetFontInfo(m_nFont, at20, &width, NULL); int shade2; if (at36) { if (bScan) shade2 = -128; else shade2 = shade; } else { if (bScan) shade2 = shade; else shade2 = 32; } gMenuTextMgr.DrawText(at20, m_nFont, x, m_nY, shade2, 0, false); x += width; } if (bScan && ((int)totalclock & 32)) gMenuTextMgr.DrawText("_", m_nFont, x, m_nY, shade, pal, false); int mx = m_nX<<16; int my = m_nY<<16; int mw = m_nWidth<<16; int mh = height<<16; if (!gGameMenuMgr.m_bScanning && bEnable && MOUSEACTIVECONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousepos, mx, my, mw, mh))) { if (MOUSEWATCHPOINTCONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_prevmousepos, mx, my, mw, mh))) { pMenu->SetFocusItem(this); } if (!gGameMenuMgr.m_mousecaught && inputState.mouseClickState() == MOUSE_RELEASED && !gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousedownpos, mx, my, mw, mh)) { pMenu->SetFocusItem(this); CGameMenuEvent event = { kMenuEventEnter, 0 }; gGameMenuMgr.m_mousecaught = 1; if (Event(event)) gGameMenuMgr.PostPop(); } } } bool CGameMenuItemZEditBitmap::Event(CGameMenuEvent &event) { static char buffer[256]; // Hack if (event.at2 == sc_kpad_2 || event.at2 == sc_kpad_4 || event.at2 == sc_kpad_6 || event.at2 == sc_kpad_8) event.at0 = kMenuEventKey; switch (event.at0) { case kMenuEventEscape: if (bScan) { strncpy(at20, buffer, at24); at20[at24-1] = 0; bScan = 0; gGameMenuMgr.m_bScanning = false; gSaveGameActive = false; return false; } gSaveGameActive = true; return true; case kMenuEventEnter: if (!at35 || bScan) { if (at30) at30(this, &event); if (bScan) { bScan = 0; gGameMenuMgr.m_bScanning = false; } gSaveGameActive = false; inputState.ClearKeyStatus(sc_Enter); inputState.ClearKeyStatus(sc_kpad_Enter); return false; } strncpy(buffer, at20, at24); if (at37) at20[0] = 0; buffer[at24-1] = 0; bScan = 1; gGameMenuMgr.m_bScanning = true; return false; case kMenuEventBackSpace: if (bScan) BackChar(); return false; case kMenuEventKey: case kMenuEventSpace: { char key; if (bScan && event.at2 < 128) { if (inputState.ShiftPressed()) key = g_keyAsciiTableShift[event.at2]; else key = g_keyAsciiTable[event.at2]; if (at30 && (isalnum(key) || ispunct(key) || isspace(key))) { AddChar(key); return false; } } return CGameMenuItem::Event(event); } case kMenuEventUp: if (bScan) return false; return CGameMenuItem::Event(event); case kMenuEventDown: if (bScan) return false; return CGameMenuItem::Event(event); } return CGameMenuItem::Event(event); } CGameMenuItemQAV::CGameMenuItemQAV() { at20 = NULL; at24 = NULL; at28 = 0; bEnable = 0; } CGameMenuItemQAV::CGameMenuItemQAV(const char *a1, int a2, int a3, int a4, const char *a5, bool widescreen, bool clearbackground) { m_nWidth = 0; m_pzText = a1; m_nFont = a2; m_nY = a4; at20 = a5; m_nX = a3; bEnable = 0; bWideScreen = widescreen; bClearBackground = clearbackground; } void CGameMenuItemQAV::Draw(void) { if (bClearBackground) videoClearScreen(0); if (at24) { ClockTicks backFC = gFrameClock; gFrameClock = totalclock; int nTicks = (int)totalclock - at30; at30 = (int)totalclock; at2c -= nTicks; if (at2c <= 0 || at2c > at28->at10) { at2c = at28->at10; } at28->Play(at28->at10 - at2c - nTicks, at28->at10 - at2c, -1, NULL); int wx1, wy1, wx2, wy2; wx1 = windowxy1.x; wy1 = windowxy1.y; wx2 = windowxy2.x; wy2 = windowxy2.y; windowxy1.x = 0; windowxy1.y = 0; windowxy2.x = xdim-1; windowxy2.y = ydim-1; if (bWideScreen) { int xdim43 = scale(ydim, 4, 3); int nCount = (xdim+xdim43-1)/xdim43; int backX = at28->x; for (int i = 0; i < nCount; i++) { at28->Draw(at28->at10 - at2c, 10+kQavOrientationLeft, 0, 0); at28->x += 320; } at28->x = backX; } else at28->Draw(at28->at10 - at2c, 10, 0, 0); windowxy1.x = wx1; windowxy1.y = wy1; windowxy2.x = wx2; windowxy2.y = wy2; gFrameClock = backFC; } if (bEnable && !gGameMenuMgr.m_mousecaught && inputState.mouseClickState() == MOUSE_RELEASED) { pMenu->SetFocusItem(this); CGameMenuEvent event = { kMenuEventEnter, 0 }; gGameMenuMgr.m_mousecaught = 1; if (Event(event)) gGameMenuMgr.PostPop(); } } bool CGameMenuItemQAV::Event(CGameMenuEvent &event) { switch (event.at0) { case kMenuEventLeft: case kMenuEventBackSpace: pMenu->FocusPrevItem(); return false; case kMenuEventRight: case kMenuEventEnter: case kMenuEventSpace: pMenu->FocusNextItem(); return false; case kMenuEventInit: if (at20) { if (!at28) { at24 = gSysRes.Lookup(at20, "QAV"); if (!at24) ThrowError("Could not load QAV %s\n", at20); at28 = (QAV*)gSysRes.Lock(at24); at28->nSprite = -1; at28->x = m_nX; at28->y = m_nY; at28->Preload(); at2c = at28->at10; at30 = (int)totalclock; return false; } gSysRes.Lock(at24); } return false; case kMenuEventDeInit: if (at20 && at28) { gSysRes.Unlock(at24); if (at24->LockCount() == 0) at28 = NULL; } return false; } return CGameMenuItem::Event(event); } void CGameMenuItemQAV::Reset(void) { at2c = at28->at10; at30 = (int)totalclock; } CGameMenuItemZCycleSelect::CGameMenuItemZCycleSelect() { m_pzText = NULL; m_nFont = 3; m_nX = 0; m_nY = 0; m_nRows = 0; m_nTopDelta = 0; m_nFocus = 0; m_nItems = 0; m_pzStrings = NULL; m_pReturn = NULL; } CGameMenuItemZCycleSelect::CGameMenuItemZCycleSelect(const char *pzText, int nFont, int nX, int nY, int nWidth, int nRows, int nItems, const char **pzStrings, int *pReturn, void(*pCallback)(CGameMenuItemZCycleSelect *)) { m_nTopDelta = 0; m_nFocus = 0; m_pzText = pzText; m_nFont = nFont; m_nX = nX; m_nY = nY; m_nWidth = nWidth; m_nRows = nRows; m_pCallback = pCallback; m_nItems = nItems; m_pzStrings = pzStrings; m_pReturn = pReturn; } void CGameMenuItemZCycleSelect::Draw(void) { int height; int shade; gMenuTextMgr.GetFontInfo(m_nFont, NULL, NULL, &height); int y = m_nY; int k = m_nFocus - m_nTopDelta; int nNewFocus = m_nFocus; bool bClick = false; for (int i = 0; i < m_nRows; i++, y += height, k++) { if (k == m_nFocus) { shade = 32; if (pMenu->IsFocusItem(this)) shade = 32-((int)totalclock&63); viewDrawText(3, m_pzStrings[k], m_nX, y, shade, 0, 0, false); } else { viewDrawText(3, m_pzStrings[k], m_nX, y, 24, 0, 0, false); } int mx = m_nX<<16; int my = y<<16; int mw = m_nWidth<<16; int mh = height<<16; if (bEnable && MOUSEACTIVECONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousepos, mx, my, mw, mh))) { if (MOUSEWATCHPOINTCONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_prevmousepos, mx, my, mw, mh))) { nNewFocus = k; } if (!gGameMenuMgr.m_mousecaught && inputState.mouseClickState() == MOUSE_RELEASED && !gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousedownpos, mx, my, mw, mh)) { nNewFocus = k; bClick = true; } } } m_nTopDelta += nNewFocus-m_nFocus; m_nFocus = nNewFocus; if (bClick) { CGameMenuEvent event = { kMenuEventEnter, 0 }; gGameMenuMgr.m_mousecaught = 1; if (Event(event)) gGameMenuMgr.PostPop(); } } bool CGameMenuItemZCycleSelect::Event(CGameMenuEvent &event) { switch (event.at0) { case kMenuEventUp: if (event.at2 == sc_Tab || m_nFocus == 0) { pMenu->FocusPrevItem(); return false; } m_nFocus--; if (m_nTopDelta > 0) m_nTopDelta--; return false; case kMenuEventDown: if (event.at2 == sc_Tab || m_nFocus == m_nItems-1) { pMenu->FocusNextItem(); return false; } m_nFocus++; if (m_nTopDelta+1 < m_nRows) m_nTopDelta++; return false; case kMenuEventEnter: if (m_pCallback) m_pCallback(this); *m_pReturn = m_nFocus; return true; case kMenuEventScrollUp: if (m_nFocus-m_nTopDelta > 0) { m_nTopDelta++; if (m_nTopDelta>0) { m_nFocus--; m_nTopDelta--; } } return false; case kMenuEventScrollDown: if (m_nFocus-m_nTopDelta+m_nRows < m_nItems) { m_nTopDelta--; if (m_nTopDelta+1 < m_nRows) { m_nFocus++; m_nTopDelta++; } } return false; } return CGameMenuItem::Event(event); } bool CGameMenuItemZCycleSelect::MouseEvent(CGameMenuEvent &event) { event.at0 = kMenuEventNone; if (MOUSEACTIVECONDITIONAL(inputState.MouseGetButtons()&WHEELUP_MOUSE)) { gGameMenuMgr.m_mouselastactivity = (int)totalclock; inputState.MouseClearButton(WHEELUP_MOUSE); event.at0 = kMenuEventScrollUp; } else if (MOUSEACTIVECONDITIONAL(inputState.MouseGetButtons()&WHEELDOWN_MOUSE)) { gGameMenuMgr.m_mouselastactivity = (int)totalclock; inputState.MouseClearButton(WHEELDOWN_MOUSE); event.at0 = kMenuEventScrollDown; } else return CGameMenuItem::MouseEvent(event); return event.at0 != kMenuEventNone; } CGameMenuItemZCycle::CGameMenuItemZCycle() { m_pzText = NULL; m_nFocus = 0; m_nItems = 0; m_pCallback = NULL; m_pCallbackSelect = NULL; m_pMenuSelect = NULL; m_pItemSelectTitle = NULL; m_pItemSelect = NULL; m_nMenuSelectReturn = -1; } CGameMenuItemZCycle::CGameMenuItemZCycle(const char *a1, int a2, int a3, int a4, int a5, int a6, void(*a7)(CGameMenuItemZCycle *), const char **a8, int a9, int a10, bool bMenu, void(*pCallbackSelect)(CGameMenuItemZCycleSelect*)) { m_pzText = a1; m_nFont = a2; m_nX = a3; m_nY = a4; m_nFocus = 0; m_nWidth = a5; m_nAlign = a6; m_pCallback = a7; m_pCallbackSelect = pCallbackSelect; m_nItems = 0; m_bMenu = bMenu; m_pMenuSelect = NULL; m_pItemSelectTitle = NULL; m_pItemSelect = NULL; m_nMenuSelectReturn = -1; SetTextArray(a8, a9, a10); } CGameMenuItemZCycle::~CGameMenuItemZCycle() { m_pzText = NULL; m_nFocus = 0; m_nItems = 0; m_pCallback = NULL; m_pCallbackSelect = NULL; m_pMenuSelect = NULL; m_pItemSelectTitle = NULL; m_pItemSelect = NULL; m_nMenuSelectReturn = -1; memset(m_pzStrings, 0, sizeof(m_pzStrings)); } void CGameMenuItemZCycle::Draw(void) { int width = 0, height = 0; int shade = bEnable ? 32 : 48; int pal = bEnable ? 0 : 5; if (pMenu->IsFocusItem(this)) shade = 32-((int)totalclock&63); int x = m_nX; int y = m_nY; if (m_nMenuSelectReturn != -1) { m_nFocus = m_nMenuSelectReturn; if (m_pCallback) m_pCallback(this); m_nMenuSelectReturn = -1; } if (m_pzText) { gMenuTextMgr.GetFontInfo(m_nFont, m_pzText, &width, &height); switch (m_nAlign) { case 1: x = m_nX+m_nWidth/2-width/2; break; case 2: x = m_nX+m_nWidth-1-width; break; case 0: default: break; } gMenuTextMgr.DrawText(m_pzText, m_nFont, x, y, shade, pal, false); } const char *pzText; if (!m_nItems) pzText = "????"; else pzText = m_pzStrings[m_nFocus]; dassert(pzText != NULL); gMenuTextMgr.GetFontInfo(m_nFont, pzText, &width, NULL); gMenuTextMgr.DrawText(pzText, m_nFont, m_nX + m_nWidth - 1 - width, y, shade, pal, false); if (bEnable && MOUSEACTIVECONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousepos, x<<16, y<<16, m_nWidth<<16, height<<16))) { if (MOUSEWATCHPOINTCONDITIONAL(!gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_prevmousepos, x<<16, y<<16, m_nWidth<<16, height<<16))) { pMenu->SetFocusItem(this); } if (!gGameMenuMgr.m_mousecaught && inputState.mouseClickState() == MOUSE_RELEASED && !gGameMenuMgr.MouseOutsideBounds(&gGameMenuMgr.m_mousedownpos, x<<16, y<<16, m_nWidth<<16, height<<16)) { pMenu->SetFocusItem(this); CGameMenuEvent event = { kMenuEventEnter, 0 }; gGameMenuMgr.m_mousecaught = 1; if (Event(event)) gGameMenuMgr.PostPop(); } } } bool CGameMenuItemZCycle::Event(CGameMenuEvent &event) { switch (event.at0) { case kMenuEventEnter: if (m_bMenu) { if (m_pMenuSelect) { delete m_pMenuSelect; m_pMenuSelect = NULL; } if (m_pItemSelectTitle) { delete m_pItemSelectTitle; m_pItemSelectTitle = NULL; } if (m_pItemSelect) { delete m_pItemSelect; m_pItemSelect = NULL; } m_pMenuSelect = new CGameMenu(); dassert(m_pMenuSelect != NULL); strncpy(m_zTitle, m_pzText, kMaxTitleLength); int l = strlen(m_zTitle); if (l > 0 && m_zTitle[l-1] == ':') l--; m_zTitle[l] = 0; m_pItemSelectTitle = new CGameMenuItemTitle(m_zTitle, 1, 160, 20, 2038); dassert(m_pItemSelectTitle != NULL); m_pItemSelect = new CGameMenuItemZCycleSelect("", 3, 100, 40, 100, 16, m_nItems, m_pzStrings, &m_nMenuSelectReturn, m_pCallbackSelect); dassert(m_pItemSelect != NULL); m_pMenuSelect->Add(m_pItemSelectTitle, false); m_pMenuSelect->Add(m_pItemSelect, true); m_pMenuSelect->Add(&itemBloodQAV, false); gGameMenuMgr.Push(m_pMenuSelect, -1); return false; } fallthrough__; case kMenuEventRight: case kMenuEventSpace: Next(); if (m_pCallback) m_pCallback(this); return false; case kMenuEventLeft: Prev(); if (m_pCallback) m_pCallback(this); return false; case kMenuEventDeInit: if (m_pMenuSelect) { delete m_pMenuSelect; m_pMenuSelect = NULL; } if (m_pItemSelectTitle) { delete m_pItemSelectTitle; m_pItemSelectTitle = NULL; } if (m_pItemSelect) { delete m_pItemSelect; m_pItemSelect = NULL; } return false; } return CGameMenuItem::Event(event); } void CGameMenuItemZCycle::Add(const char *pItem, bool active) { dassert(pItem != NULL); dassert(m_nItems < kMaxGameCycleItems); m_pzStrings[m_nItems] = pItem; if (active) m_nFocus = m_nItems; m_nItems++; } void CGameMenuItemZCycle::Next(void) { if (m_nItems > 0) { m_nFocus++; if (m_nFocus >= m_nItems) m_nFocus = 0; } } void CGameMenuItemZCycle::Prev(void) { if (m_nItems > 0) { m_nFocus--; if (m_nFocus < 0) m_nFocus += m_nItems; } } void CGameMenuItemZCycle::Clear(void) { m_nItems = m_nFocus = 0; memset(m_pzStrings, 0, sizeof(m_pzStrings)); } void CGameMenuItemZCycle::SetTextArray(const char **pTextArray, int nTextPtrCount, int nIndex) { Clear(); dassert(nTextPtrCount <= kMaxGameCycleItems); for (int i = 0; i < nTextPtrCount; i++) Add(pTextArray[i], false); SetTextIndex(nIndex); } void CGameMenuItemZCycle::SetTextIndex(int nIndex) { m_nFocus = ClipRange(nIndex, 0, m_nItems); } CGameMenuItemYesNoQuit::CGameMenuItemYesNoQuit() { m_pzText = NULL; m_nRestart = 0; } CGameMenuItemYesNoQuit::CGameMenuItemYesNoQuit(const char *a1, int a2, int a3, int a4, int a5, int a6, int a7) { m_pzText = a1; m_nFont = a2; m_nX = a3; m_nY = a4; m_nWidth = a5; at20 = a6; m_nRestart = a7; } void CGameMenuItemYesNoQuit::Draw(void) { if (!m_pzText) return; int shade = 32; if (pMenu->IsFocusItem(this)) shade = 32-((int)totalclock&63); int width; int x = m_nX; switch (at20) { case 1: gMenuTextMgr.GetFontInfo(m_nFont, m_pzText, &width, NULL); x = m_nX+m_nWidth/2-width/2; break; case 2: gMenuTextMgr.GetFontInfo(m_nFont, m_pzText, &width, NULL); x = m_nX+m_nWidth-1-width; break; case 0: default: break; } gMenuTextMgr.DrawText(m_pzText, m_nFont, x, m_nY, shade, 0, true); if (bEnable && !gGameMenuMgr.m_mousecaught && inputState.mouseClickState() == MOUSE_RELEASED) { pMenu->SetFocusItem(this); CGameMenuEvent event = { kMenuEventEnter, 0 }; gGameMenuMgr.m_mousecaught = 1; if (Event(event)) gGameMenuMgr.PostPop(); } } extern void Restart(CGameMenuItemChain *pItem); extern void Quit(CGameMenuItemChain *pItem); bool CGameMenuItemYesNoQuit::Event(CGameMenuEvent &event) { switch (event.at0) { case kMenuEventKey: if (event.at2 == sc_Y) { if (m_nRestart) Restart(NULL); else Quit(NULL); } else if (event.at2 == sc_N) gGameMenuMgr.Pop(); return false; case kMenuEventEnter: if (m_nRestart) Restart(NULL); else Quit(NULL); return false; } return CGameMenuItem::Event(event); } CGameMenuItemPicCycle::CGameMenuItemPicCycle() { m_pzText = NULL; at24 = 0; m_nItems = 0; atb0 = 0; at2c = 0; atb4 = 0; } CGameMenuItemPicCycle::CGameMenuItemPicCycle(int a1, int a2, void(*a3)(CGameMenuItemPicCycle *), int *a4, int a5, int a6) { m_nWidth = 0; at24 = 0; m_nItems = 0; m_nX = a1; m_nY = a2; atb0 = a3; atb4 = 0; SetPicArray(a4, a5, a6); } void CGameMenuItemPicCycle::Draw(void) { videoSetViewableArea(0, 0, xdim - 1, ydim - 1); if (atb4) rotatesprite(0, 0, 65536, 0, atb4, 0, 0, 82, 0, 0, xdim - 1, ydim - 1); if (at30[at24]) rotatesprite(0, 0, 65536, 0, at30[at24], 0, 0, 82, 0, 0, xdim - 1, ydim - 1); } bool CGameMenuItemPicCycle::Event(CGameMenuEvent &event) { switch (event.at0) { case kMenuEventRight: case kMenuEventEnter: case kMenuEventSpace: Next(); if (atb0) atb0(this); return false; case kMenuEventLeft: Prev(); if (atb0) atb0(this); return false; } return CGameMenuItem::Event(event); } void CGameMenuItemPicCycle::Add(int nItem, bool active) { dassert(m_nItems < kMaxPicCycleItems); at30[m_nItems] = nItem; if (active) at24 = m_nItems; m_nItems++; } void CGameMenuItemPicCycle::Next(void) { if (m_nItems > 0) { at24++; if (at24 >= m_nItems) at24 = 0; } } void CGameMenuItemPicCycle::Prev(void) { if (m_nItems > 0) { at24--; if (at24 < 0) at24 += m_nItems; } } void CGameMenuItemPicCycle::Clear(void) { m_nItems = at24 = 0; memset(at30, 0, sizeof(at30)); at2c = 0; } void CGameMenuItemPicCycle::SetPicArray(int *pArray, int nTileCount, int nIndex) { Clear(); at2c = 0; dassert(nTileCount <= kMaxPicCycleItems); for (int i = 0; i < nTileCount; i++) Add(pArray[i], false); SetPicIndex(nIndex); } void CGameMenuItemPicCycle::SetPicIndex(int nIndex) { at24 = ClipRange(nIndex, 0, m_nItems); } CGameMenuItemPassword::CGameMenuItemPassword() { at37 = 0; m_pzText = NULL; at36 = 0; at32 = 0; at5b = 0; } CGameMenuItemPassword::CGameMenuItemPassword(const char *a1, int a2, int a3, int a4) { at37 = 0; m_nWidth = 0; at36 = 0; at32 = 0; at5b = 0; m_pzText = a1; m_nFont = a2; m_nX = a3; m_nY = a4; } const char *kCheckPasswordMsg = "ENTER PASSWORD: "; const char *kOldPasswordMsg = "ENTER OLD PASSWORD: "; const char *kNewPasswordMsg = "ENTER NEW PASSWORD: "; const char *kInvalidPasswordMsg = "INVALID PASSWORD."; void CGameMenuItemPassword::Draw(void) { bool focus = pMenu->IsFocusItem(this); int shade = 32; int shadef = 32-((int)totalclock&63); int width; switch (at37) { case 1: case 2: case 3: switch (at37) { case 1: strcpy(at3b, kCheckPasswordMsg); break; case 2: strcpy(at3b, kOldPasswordMsg); break; case 3: strcpy(at3b, kNewPasswordMsg); break; } for (int i = 0; i < at32; i++) strcat(at3b, "*"); strcat(at3b, "_"); gMenuTextMgr.GetFontInfo(m_nFont, at3b, &width, NULL); gMenuTextMgr.DrawText(at3b, m_nFont, m_nX-width/2, m_nY+20, shadef, 0, false); shadef = 32; break; case 4: if (((int)totalclock - at5b) & 32) { gMenuTextMgr.GetFontInfo(m_nFont, kInvalidPasswordMsg, &width, NULL); gMenuTextMgr.DrawText(kInvalidPasswordMsg, m_nFont, m_nX - width / 2, m_nY + 20, shade, 0, false); } if (at5b && totalclock-at5b > 256) { at5b = 0; at37 = 0; } break; } gMenuTextMgr.GetFontInfo(m_nFont, m_pzText, &width, NULL); gMenuTextMgr.DrawText(m_pzText, m_nFont, m_nX-width/2, m_nY, focus ? shadef : shade, 0, false); } bool CGameMenuItemPassword::Event(CGameMenuEvent &event) { switch (at37) { case 0: case 4: if (event.at0 == kMenuEventEnter) { at29[0] = 0; if (strcmp(at20, "")) at37 = 2; else at37 = 3; return false; } return CGameMenuItem::Event(event); case 1: case 2: case 3: switch (event.at0) { case kMenuEventEnter: switch (at37) { case 1: at36 = strcmp(at20,at29) == 0; if (at36) at37 = 0; else at37 = 4; if (!at36) { at5b = (int)totalclock; pMenu->FocusPrevItem(); } else { at5f->at20 = 0; at5f->Draw(); gbAdultContent = false; // NUKE-TODO: //CONFIG_WriteAdultMode(); pMenu->FocusPrevItem(); } return false; case 2: at36 = strcmp(at20,at29) == 0; if (at36) at37 = 0; else at37 = 4; if (at36) { strcpy(at20, ""); strcpy(gzAdultPassword, ""); // NUKE-TODO: //CONFIG_WriteAdultMode(); at37 = 0; } else at5b = (int)totalclock; return false; case 3: strcpy(at20, at29); strcpy(at20, gzAdultPassword); strcpy(gzAdultPassword, ""); // NUKE-TODO: //CONFIG_WriteAdultMode(); at37 = 0; return false; } break; case kMenuEventEscape: at37 = 0; Draw(); return false; case kMenuEventKey: if (at32 < 8) { char key = Btoupper(g_keyAsciiTable[event.at2]); if (isalnum(key) || ispunct(key) || isspace(key)) { at29[at32++] = key; at29[at32] = 0; } } return false; case kMenuEventBackSpace: if (at32 > 0) at29[--at32] = 0; return false; case kMenuEventLeft: case kMenuEventRight: case kMenuEventSpace: return false; } } return CGameMenuItem::Event(event); } bool GameInterface::mouseInactiveConditional(bool condition) { return MOUSEINACTIVECONDITIONAL(condition); } END_BLD_NS