scummvm/engines/agi/menu.cpp
2021-12-26 18:48:43 +01:00

737 lines
21 KiB
C++

/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* 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, see <http://www.gnu.org/licenses/>.
*
*/
#include "common/config-manager.h"
#include "agi/agi.h"
#include "agi/graphics.h"
#include "agi/text.h"
#include "agi/keyboard.h"
#include "agi/menu.h"
namespace Agi {
GfxMenu::GfxMenu(AgiEngine *vm, GfxMgr *gfx, PictureMgr *picture, TextMgr *text) {
_vm = vm;
_gfx = gfx;
_picture = picture;
_text = text;
_allowed = true;
_submitted = false;
_delayedExecuteViaKeyboard = false;
_delayedExecuteViaMouse = false;
if (!_vm->isLanguageRTL())
_setupMenuColumn = 1;
else
_setupMenuColumn = FONT_COLUMN_CHARACTERS - 2;
_setupMenuItemColumn = 1;
_lastSelectedMenuNr = 0;
_mouseModeItemNr = -1;
_drawnMenuNr = -1;
_drawnMenuHeight = 0;
_drawnMenuWidth = 0;
_drawnMenuY = 0;
_drawnMenuX = 0;
}
GfxMenu::~GfxMenu() {
for (GuiMenuArray::iterator itemIter = _array.begin(); itemIter != _array.end(); ++itemIter)
delete *itemIter;
_array.clear();
for (GuiMenuItemArray::iterator menuIter = _itemArray.begin(); menuIter != _itemArray.end(); ++menuIter)
delete *menuIter;
_itemArray.clear();
}
void GfxMenu::addMenu(const char *menuText) {
int16 curColumnEnd = _setupMenuColumn;
// already submitted? in that case no further changes possible
if (_submitted)
return;
GuiMenuEntry *menuEntry = new GuiMenuEntry();
menuEntry->text = menuText;
// WORKAROUND: Apple II gs Goldrush! Speed menu exceeds screen width, because of a redundant space at 'Special' menu, remove it
if (_vm->getPlatform() == Common::kPlatformApple2GS && ConfMan.getBool("apple2gs_speedmenu") && _vm->getGameID() == GID_GOLDRUSH)
if (menuEntry->text == " Special ")
menuEntry->text = "Special ";
menuEntry->textLen = menuEntry->text.size();
if (!_vm->isLanguageRTL()) {
// Cut menu name in case menu bar is full
// Happens in at least the fan game Get Outta Space Quest
// Original interpreter had graphical issues in this case
// TODO: this whole code needs to get reworked anyway to support different types of menu bars depending on platform
curColumnEnd += menuEntry->textLen;
while ((menuEntry->textLen) && (curColumnEnd > 40)) {
menuEntry->text.deleteLastChar();
menuEntry->textLen--;
curColumnEnd--;
}
}
menuEntry->row = 0;
menuEntry->column = _setupMenuColumn;
if (_vm->isLanguageRTL())
menuEntry->column -= menuEntry->textLen;
menuEntry->itemCount = 0;
menuEntry->firstItemNr = _itemArray.size();
menuEntry->selectedItemNr = menuEntry->firstItemNr;
menuEntry->maxItemTextLen = 0;
_array.push_back(menuEntry);
if (!_vm->isLanguageRTL())
_setupMenuColumn += menuEntry->textLen + 1;
else
_setupMenuColumn -= menuEntry->textLen + 1;
}
void GfxMenu::addMenuItem(const char *menuItemText, uint16 controllerSlot) {
int16 arrayCount = _array.size();
// already submitted? in that case no further changes possible
if (_submitted)
return;
if (arrayCount == 0)
error("tried to add a menu item before adding an actual menu");
// go to latest menu entry
GuiMenuEntry *curMenuEntry = _array.back();
GuiMenuItemEntry *menuItemEntry = new GuiMenuItemEntry();
menuItemEntry->enabled = true;
menuItemEntry->text = menuItemText;
menuItemEntry->textLen = menuItemEntry->text.size();
menuItemEntry->controllerSlot = controllerSlot;
// Original interpreter on PC used the length of the first item for drawing
// At least in KQ2 on Apple IIgs follow-up items are longer, which would result in graphic glitches.
// That's why we remember the longest item and draw according to that
if (curMenuEntry->maxItemTextLen < menuItemEntry->textLen) {
curMenuEntry->maxItemTextLen = menuItemEntry->textLen;
}
if (curMenuEntry->itemCount == 0) {
if (!_vm->isLanguageRTL()) {
// for first menu item of menu calculated column
if (menuItemEntry->textLen + curMenuEntry->column < (FONT_COLUMN_CHARACTERS - 1)) {
_setupMenuItemColumn = curMenuEntry->column;
} else {
_setupMenuItemColumn = (FONT_COLUMN_CHARACTERS - 1) - menuItemEntry->textLen;
}
} else {
_setupMenuItemColumn = curMenuEntry->column + curMenuEntry->textLen - menuItemEntry->textLen;
if (_setupMenuItemColumn < 2)
_setupMenuItemColumn = 2;
}
}
menuItemEntry->row = 2 + curMenuEntry->itemCount;
menuItemEntry->column = _setupMenuItemColumn;
_itemArray.push_back(menuItemEntry);
curMenuEntry->itemCount++;
}
void GfxMenu::submit() {
GuiMenuEntry *menuEntry = nullptr;
GuiMenuItemEntry *menuItemEntry = nullptr;
int16 menuCount = _array.size();
int16 menuNr = 0;
int16 menuItemNr = 0;
int16 menuItemLastNr = 0;
if ((_array.size() == 0) || (_itemArray.size() == 0))
return;
// WORKAROUND: For Apple II gs we add a Speed menu
if (_vm->getPlatform() == Common::kPlatformApple2GS && ConfMan.getBool("apple2gs_speedmenu")) {
uint16 maxControllerSlot = 0;
for (GuiMenuItemArray::iterator menuIter = _itemArray.begin(); menuIter != _itemArray.end(); ++menuIter)
if ((*menuIter)->controllerSlot > maxControllerSlot)
maxControllerSlot = (*menuIter)->controllerSlot;
for (uint16 curMapping = 0; curMapping < MAX_CONTROLLER_KEYMAPPINGS; curMapping++)
if (_vm->_game.controllerKeyMapping[curMapping].controllerSlot > maxControllerSlot)
maxControllerSlot = _vm->_game.controllerKeyMapping[curMapping].controllerSlot;
if (maxControllerSlot >= 0xff - 4)
warning("GfxMenu::submit : failed to add 'Speed' menu");
else {
_vm->_game.appleIIgsSpeedControllerSlot = maxControllerSlot + 1;
addMenu("Speed");
addMenuItem("Normal", _vm->_game.appleIIgsSpeedControllerSlot + 2);
addMenuItem("Slow", _vm->_game.appleIIgsSpeedControllerSlot + 3);
addMenuItem("Fast", _vm->_game.appleIIgsSpeedControllerSlot + 1);
addMenuItem("Fastest", _vm->_game.appleIIgsSpeedControllerSlot + 0);
}
}
_submitted = true;
// WORKAROUND: For Apple II gs we try to fix the menu text
// On this platform it seems a system font was used and the menu was drawn differently (probably system menu?)
// Still the text was misaligned anyway, but it looks worse in our (the original PC) implementation
// Atari ST SQ1 had one bad menu entry as well, we fix that too.
switch (_vm->getPlatform()) {
case Common::kPlatformApple2GS:
case Common::kPlatformAtariST:
// Go through all menus
for (menuNr = 0; menuNr < menuCount; menuNr++) {
menuEntry = _array[menuNr];
menuItemLastNr = menuEntry->firstItemNr + menuEntry->itemCount;
// Go through all items of current menu
for (menuItemNr = menuEntry->firstItemNr; menuItemNr < menuItemLastNr; menuItemNr++) {
menuItemEntry = _itemArray[menuItemNr];
if (menuItemEntry->textLen < menuEntry->maxItemTextLen) {
// current item text is shorter than the maximum?
int16 missingCharCount = menuEntry->maxItemTextLen - menuItemEntry->textLen;
if (menuItemEntry->text.contains('>')) {
// text contains '>', we now try to find a '<'
// and then add spaces in case this item is shorter than the first item
int16 textPos = menuItemEntry->textLen - 1;
while (textPos > 0) {
if (menuItemEntry->text[textPos] == '<')
break;
textPos--;
}
if (textPos > 0) {
while (missingCharCount) {
menuItemEntry->text.insertChar(' ', textPos);
missingCharCount--;
}
}
} else {
// Also check if text consists only of '-', which is the separator
// These were sometimes also too small
int16 separatorCount = 0;
int16 charPos = 0;
while (charPos < menuItemEntry->textLen) {
if (menuItemEntry->text[charPos] != '-')
break;
separatorCount++;
charPos++;
}
if (separatorCount == menuItemEntry->textLen) {
// Separator detected
while (missingCharCount) {
menuItemEntry->text.insertChar('-', 0);
missingCharCount--;
}
} else {
// Append spaces to the end to fill it up
int16 textPos = menuItemEntry->textLen;
while (missingCharCount) {
menuItemEntry->text.insertChar(' ', textPos);
textPos++;
missingCharCount--;
}
}
}
menuItemEntry->textLen = menuItemEntry->text.size();
}
}
}
break;
default:
break;
}
}
void GfxMenu::itemEnable(uint16 controllerSlot) {
itemEnableDisable(controllerSlot, true);
}
void GfxMenu::itemDisable(uint16 controllerSlot) {
itemEnableDisable(controllerSlot, false);
}
void GfxMenu::itemEnableDisable(uint16 controllerSlot, bool enabled) {
GuiMenuItemArray::iterator listIterator;
GuiMenuItemArray::iterator listEnd = _itemArray.end();
GuiMenuItemEntry *menuItemEntry;
listIterator = _itemArray.begin();
while (listIterator != listEnd) {
menuItemEntry = *listIterator;
if (menuItemEntry->controllerSlot == controllerSlot) {
menuItemEntry->enabled = enabled;
}
listIterator++;
}
}
void GfxMenu::itemEnableAll() {
GuiMenuItemArray::iterator listIterator;
GuiMenuItemArray::iterator listEnd = _itemArray.end();
GuiMenuItemEntry *menuItemEntry;
listIterator = _itemArray.begin();
while (listIterator != listEnd) {
menuItemEntry = *listIterator;
menuItemEntry->enabled = true;
listIterator++;
}
}
// return true, in case a menu was actually created and submitted by the scripts
bool GfxMenu::isAvailable() {
return _submitted;
}
void GfxMenu::accessAllow() {
_allowed = true;
}
void GfxMenu::accessDeny() {
_allowed = false;
}
void GfxMenu::delayedExecuteViaKeyboard() {
_delayedExecuteViaKeyboard = true;
_delayedExecuteViaMouse = false;
}
void GfxMenu::delayedExecuteViaMouse() {
_delayedExecuteViaKeyboard = false;
_delayedExecuteViaMouse = true;
}
bool GfxMenu::delayedExecuteActive() {
return _delayedExecuteViaKeyboard | _delayedExecuteViaMouse;
}
void GfxMenu::execute() {
bool viaKeyboard = _delayedExecuteViaKeyboard;
bool viaMouse = _delayedExecuteViaMouse;
_delayedExecuteViaKeyboard = false;
_delayedExecuteViaMouse = false;
// got submitted? -> safety check
if (!_submitted)
return;
// access allowed at the moment?
if (!_allowed)
return;
_text->charPos_Push();
_text->charAttrib_Push();
_text->clearLine(0, _text->calculateTextBackground(15));
// Draw all menus
for (uint16 menuNr = 0; menuNr < _array.size(); menuNr++) {
drawMenuName(menuNr, false);
}
// Draw last selected menu
_drawnMenuNr = _lastSelectedMenuNr;
// Unless we are in "via mouse" mode. In that case check current mouse position
if (viaMouse) {
int16 mouseRow = _vm->_mouse.pos.y;
int16 mouseColumn = _vm->_mouse.pos.x;
_gfx->translateDisplayPosToFontScreen(mouseColumn, mouseRow);
mouseFindMenuSelection(mouseRow, mouseColumn, _drawnMenuNr, _mouseModeItemNr);
}
if (_drawnMenuNr >= 0) {
if (viaKeyboard) {
drawMenu(_drawnMenuNr, _array[_drawnMenuNr]->selectedItemNr);
}
if (viaMouse) {
drawMenu(_drawnMenuNr, _mouseModeItemNr);
}
}
if (viaKeyboard) {
_vm->cycleInnerLoopActive(CYCLE_INNERLOOP_MENU_VIA_KEYBOARD);
} else if (viaMouse) {
_vm->cycleInnerLoopActive(CYCLE_INNERLOOP_MENU_VIA_MOUSE);
}
do {
_vm->processAGIEvents();
} while (_vm->cycleInnerLoopIsActive() && !(_vm->shouldQuit() || _vm->_restartGame));
if (_drawnMenuNr >= 0) {
removeActiveMenu(_drawnMenuNr);
}
if (viaKeyboard) {
// In "via Keyboard" mode, remember last selection
_lastSelectedMenuNr = _drawnMenuNr;
}
_text->charAttrib_Pop();
_text->charPos_Pop();
// Restore status line
if (_text->statusEnabled()) {
_text->statusDraw();
} else {
if (_text->getWindowRowMin() == 0) {
// WORKAROUND: Playarea starts right at the stop, so instead of clearing that part, render it from playarea
// Required for at least Donald Duck
// This was not done by original AGI, which means the upper pixel line were cleared in this case.
_gfx->render_Block(0, 0, SCRIPT_WIDTH, FONT_VISUAL_HEIGHT);
} else {
_text->clearLine(0, 0);
}
}
}
void GfxMenu::drawMenuName(int16 menuNr, bool inverted) {
GuiMenuEntry *menuEntry = _array[menuNr];
bool disabledLook = false;
// Don't draw in case there is no text
if (!menuEntry->text.size())
return;
if (!inverted) {
_text->charAttrib_Set(0, _text->calculateTextBackground(15));
} else {
_text->charAttrib_Set(15, _text->calculateTextBackground(0));
}
_text->charPos_Set(menuEntry->row, menuEntry->column);
if (menuEntry->itemCount == 0)
disabledLook = true;
_text->displayText(menuEntry->text.c_str(), disabledLook);
}
void GfxMenu::drawItemName(int16 itemNr, bool inverted) {
GuiMenuItemEntry *itemEntry = _itemArray[itemNr];
bool disabledLook = false;
if (!inverted) {
_text->charAttrib_Set(0, _text->calculateTextBackground(15));
} else {
_text->charAttrib_Set(15, _text->calculateTextBackground(0));
}
_text->charPos_Set(itemEntry->row, itemEntry->column);
if (itemEntry->enabled == false)
disabledLook = true;
_text->displayText(itemEntry->text.c_str(), disabledLook);
}
void GfxMenu::drawMenu(int16 selectedMenuNr, int16 selectedMenuItemNr) {
GuiMenuEntry *menuEntry = _array[selectedMenuNr];
GuiMenuItemEntry *itemEntry = _itemArray[menuEntry->firstItemNr];
int16 itemNr = menuEntry->firstItemNr;
int16 itemCount = menuEntry->itemCount;
// draw menu name as inverted
drawMenuName(selectedMenuNr, true);
// calculate active menu dimensions
_drawnMenuHeight = (menuEntry->itemCount + 2) * FONT_VISUAL_HEIGHT;
_drawnMenuWidth = (menuEntry->maxItemTextLen * FONT_VISUAL_WIDTH) + 8;
_drawnMenuY = (1 - _text->getWindowRowMin()) * FONT_VISUAL_HEIGHT;
//(menuEntry->itemCount + 3 - _text->getWindowRowMin()) * FONT_VISUAL_HEIGHT - 1;
_drawnMenuX = (itemEntry->column - 1) * FONT_VISUAL_WIDTH;
_gfx->drawBox(_drawnMenuX, _drawnMenuY, _drawnMenuWidth, _drawnMenuHeight, 15, 0);
while (itemCount) {
if (itemNr == selectedMenuItemNr) {
drawItemName(itemNr, true);
} else {
drawItemName(itemNr, false);
}
itemNr++;
itemCount--;
}
}
void GfxMenu::removeActiveMenu(int16 selectedMenuNr) {
// draw menu name normally again
drawMenuName(selectedMenuNr, false);
// overwrite actual menu items by rendering play screen
_gfx->render_Block(_drawnMenuX, _drawnMenuY, _drawnMenuWidth, _drawnMenuHeight);
}
void GfxMenu::keyPress(uint16 newKey) {
GuiMenuEntry *menuEntry = _array[_drawnMenuNr];
GuiMenuItemEntry *itemEntry = _itemArray[menuEntry->selectedItemNr];
int16 newMenuNr = _drawnMenuNr;
int16 newItemNr = menuEntry->selectedItemNr;
switch (newKey) {
case AGI_KEY_ENTER:
// check, if current item is actually enabled
if (!itemEntry->enabled)
return;
// Trigger controller
_vm->_game.controllerOccurred[itemEntry->controllerSlot] = true;
_vm->cycleInnerLoopInactive(); // exit execute-loop
break;
case AGI_KEY_ESCAPE:
_vm->cycleInnerLoopInactive(); // exit execute-loop
break;
// these here change menu item
case AGI_KEY_UP:
newItemNr--;
break;
case AGI_KEY_DOWN:
newItemNr++;
break;
case AGI_KEY_PAGE_UP:
// select first item of current menu
newItemNr = menuEntry->firstItemNr;
break;
case AGI_KEY_PAGE_DOWN:
// select last item of current menu
newItemNr = menuEntry->firstItemNr + menuEntry->itemCount - 1;
break;
case AGI_KEY_LEFT:
if (!_vm->isLanguageRTL())
newMenuNr--;
else
newMenuNr++;
break;
case AGI_KEY_RIGHT:
if (!_vm->isLanguageRTL())
newMenuNr++;
else
newMenuNr--;
break;
case AGI_KEY_HOME:
// select first menu
newMenuNr = 0;
break;
case AGI_KEY_END:
// select last menu
newMenuNr = _array.size() - 1;
break;
default:
break;
}
if (newMenuNr != _drawnMenuNr) {
// selected menu was changed
int16 lastMenuNr = _array.size() - 1;
if (newMenuNr < 0) {
newMenuNr = lastMenuNr;
} else if (newMenuNr > lastMenuNr) {
newMenuNr = 0;
}
if (newMenuNr != _drawnMenuNr) {
removeActiveMenu(_drawnMenuNr);
_drawnMenuNr = newMenuNr;
drawMenu(_drawnMenuNr, _array[_drawnMenuNr]->selectedItemNr);
}
}
if (newItemNr != menuEntry->selectedItemNr) {
// selected item was changed
int16 lastItemNr = menuEntry->firstItemNr + menuEntry->itemCount - 1;
if (newItemNr < menuEntry->firstItemNr) {
newItemNr = lastItemNr;
} else if (newItemNr > lastItemNr) {
newItemNr = menuEntry->firstItemNr;
}
if (newItemNr != menuEntry->selectedItemNr) {
// still changed after clip -> draw changes
drawItemName(menuEntry->selectedItemNr, false);
drawItemName(newItemNr, true);
menuEntry->selectedItemNr = newItemNr;
}
}
}
// This gets called:
// During "via keyboard" mode in case user actively clicks on something
// During "via mouse" mode all the time, so that current mouse cursor position modifies active selection
// In "via mouse" mode, we check if user let go of the left mouse button and then select the item that way
void GfxMenu::mouseEvent(uint16 newKey) {
// Find out, where current mouse cursor actually is
int16 mouseRow = _vm->_mouse.pos.y;
int16 mouseColumn = _vm->_mouse.pos.x;
_gfx->translateDisplayPosToFontScreen(mouseColumn, mouseRow);
int16 activeMenuNr, activeItemNr;
mouseFindMenuSelection(mouseRow, mouseColumn, activeMenuNr, activeItemNr);
switch (newKey) {
case AGI_MOUSE_BUTTON_LEFT:
// User clicked somewhere, in this case check if user clicked on status bar or on one of the currently shown menu items
// Happens in "via keyboard" mode only
// We do not close menu in case user clicked on something invalid
if (activeItemNr >= 0) {
GuiMenuItemEntry *itemEntry = _itemArray[activeItemNr];
if (!itemEntry->enabled)
return;
// Trigger controller
_vm->_game.controllerOccurred[itemEntry->controllerSlot] = true;
_vm->cycleInnerLoopInactive(); // exit execute-loop
return;
}
if (activeMenuNr >= 0) {
// User clicked on a menu, check if that menu is already active
if (activeMenuNr != _drawnMenuNr) {
removeActiveMenu(_drawnMenuNr);
_drawnMenuNr = activeMenuNr;
drawMenu(_drawnMenuNr, _array[_drawnMenuNr]->selectedItemNr);
}
}
return; // exit all the time, we do not want to change the user selection while in "via keyboard" mode
break;
default:
break;
}
// If mouse is not selecting any menu, just use the last menu instead
if (activeMenuNr < 0) {
activeMenuNr = _drawnMenuNr;
}
if (activeMenuNr != _drawnMenuNr) {
if (_drawnMenuNr >= 0) {
removeActiveMenu(_drawnMenuNr);
}
_drawnMenuNr = activeMenuNr;
if (_drawnMenuNr >= 0) {
drawMenu(_drawnMenuNr, activeItemNr);
}
_mouseModeItemNr = activeItemNr;
}
if (activeItemNr != _mouseModeItemNr) {
if (_mouseModeItemNr >= 0) {
drawItemName(_mouseModeItemNr, false);
}
if (activeItemNr >= 0) {
drawItemName(activeItemNr, true);
}
_mouseModeItemNr = activeItemNr;
}
if (_vm->_mouse.button == kAgiMouseButtonUp) {
// User has stopped pressing the mouse button, if any item number is selected -> execute it
if (activeItemNr >= 0) {
GuiMenuItemEntry *itemEntry = _itemArray[activeItemNr];
if (itemEntry->enabled) {
// Trigger controller
_vm->_game.controllerOccurred[itemEntry->controllerSlot] = true;
}
}
_vm->cycleInnerLoopInactive(); // exit execute-loop
return;
}
}
void GfxMenu::mouseFindMenuSelection(int16 mouseRow, int16 mouseColumn, int16 &activeMenuNr, int16 &activeMenuItemNr) {
GuiMenuEntry *menuEntry = nullptr;
int16 menuCount = _array.size();
for (int16 menuNr = 0; menuNr < menuCount; menuNr++) {
menuEntry = _array[menuNr];
if (mouseRow == menuEntry->row) {
// line match
if ((mouseColumn >= menuEntry->column) && (mouseColumn < (menuEntry->column + menuEntry->textLen))) {
// full match
activeMenuNr = menuNr;
activeMenuItemNr = -1; // no item selected
return;
}
}
}
// Now also check current menu
if (_drawnMenuNr >= 0) {
// A menu is currently shown
menuEntry = _array[_drawnMenuNr];
int16 itemNr = menuEntry->firstItemNr;
int16 itemCount = menuEntry->itemCount;
while (itemCount) {
GuiMenuItemEntry *itemEntry = _itemArray[itemNr];
if (mouseRow == itemEntry->row) {
// line match
if ((mouseColumn >= itemEntry->column) && (mouseColumn < (itemEntry->column + itemEntry->textLen))) {
// full match
if (itemEntry->enabled) {
// Only see it, when it's currently enabled
activeMenuNr = _drawnMenuNr;
activeMenuItemNr = itemNr;
return;
}
}
}
itemNr++;
itemCount--;
}
}
activeMenuNr = -1;
activeMenuItemNr = -1;
return;
}
} // End of namespace Agi