Files
scummvm-cursorfix/engines/mm/xeen/dialogs/dialogs_party.cpp
2026-02-02 04:50:13 +01:00

532 lines
16 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 "mm/xeen/dialogs/dialogs_char_info.h"
#include "mm/xeen/dialogs/dialogs_create_char.h"
#include "mm/xeen/dialogs/dialogs_party.h"
#include "mm/xeen/dialogs/dialogs_input.h"
#include "mm/xeen/dialogs/dialogs_query.h"
#include "mm/xeen/character.h"
#include "mm/xeen/events.h"
#include "mm/xeen/party.h"
#include "mm/xeen/xeen.h"
namespace MM {
namespace Xeen {
#ifdef USE_TTS
static const uint8 kPartyDialogFaceCount = 4;
static const uint8 kPartyDialogButtonCount = 6;
static const uint8 kPartyDialogInfoPerFaceCount = 4;
#endif
enum PartyDialogButtonTTSTextIndex {
kPartyDialogUp = 0,
kPartyDialogDown = 1,
kPartyDialogDelete = 2,
kPartyDialogRemove = 3,
kPartyDialogCreate = 4,
kPartyDialogExit = 5,
kPartyDialogFace1 = 6,
kPartyDialogFace2 = 7,
kPartyDialogFace3 = 8,
kPartyDialogFace4 = 9
};
PartyDialog::PartyDialog(XeenEngine *vm) : ButtonContainer(vm),
PartyDrawer(vm), _vm(vm) {
#ifdef USE_TTS
_faceCount = 0;
#endif
initDrawStructs();
}
void PartyDialog::show(XeenEngine *vm) {
PartyDialog *dlg = new PartyDialog(vm);
dlg->execute();
delete dlg;
}
void PartyDialog::execute() {
EventsManager &events = *_vm->_events;
FileManager &files = *_vm->_files;
Interface &intf = *_vm->_interface;
Party &party = *_vm->_party;
Screen &screen = *_vm->_screen;
Sound &sound = *_vm->_sound;
Windows &windows = *_vm->_windows;
bool modeFlag = false;
int startingChar = 0;
sound.playSong(files._ccNum ? "newbrigh.m" : "inn.m");
loadButtons();
setupBackground();
Common::String ttsMessage;
#ifdef USE_TTS
bool returnFromCreate = false;
#endif
while (!_vm->shouldExit()) {
_vm->_mode = MODE_INTERACTIVE;
// Build up a list of available characters in the Roster that are on the
// same side of Xeen as the player is currently on
loadCharacters();
Window &w = windows[11];
w.open();
setupFaces(startingChar, false);
w.writeString(Common::String::format(Res.PARTY_DIALOG_TEXT, _partyDetails.c_str()), false, &ttsMessage);
#ifdef USE_TTS
speakText(ttsMessage, returnFromCreate);
returnFromCreate = false;
#endif
w.drawList(&_faceDrawStructs[0], 4);
_uiSprites.draw(w, 0, Common::Point(16, 100));
_uiSprites.draw(w, 2, Common::Point(52, 100));
_uiSprites.draw(w, 4, Common::Point(87, 100));
_uiSprites.draw(w, 6, Common::Point(122, 100));
_uiSprites.draw(w, 8, Common::Point(157, 100));
_uiSprites.draw(w, 10, Common::Point(192, 100));
if (g_vm->getGameID() == GType_Swords)
Res._logoSprites.draw(1, 0, Common::Point(232, 9));
screen.loadPalette("mm4.pal");
if (modeFlag) {
windows[0].update();
events.setCursor(0);
screen.fadeIn();
} else {
if (_vm->getGameID() == GType_DarkSide) {
screen.fadeOut();
windows[0].update();
}
doScroll(false, false);
events.setCursor(0);
if (_vm->getGameID() == GType_DarkSide) {
screen.fadeIn();
}
}
bool breakFlag = false;
while (!_vm->shouldExit() && !breakFlag) {
do {
events.pollEventsAndWait();
checkEvents(_vm);
} while (!_vm->shouldExit() && !_buttonValue);
if (Common::KEYCODE_F1 == _buttonValue ||
Common::KEYCODE_F2 == _buttonValue ||
Common::KEYCODE_F3 == _buttonValue ||
Common::KEYCODE_F4 == _buttonValue ||
Common::KEYCODE_F5 == _buttonValue ||
Common::KEYCODE_F6 == _buttonValue) {
// Show character info
_buttonValue -= Common::KEYCODE_F1;
if (_buttonValue < (int)party._activeParty.size())
CharacterInfo::show(_vm, _buttonValue);
} else if (Common::KEYCODE_1 == _buttonValue ||
Common::KEYCODE_2 == _buttonValue ||
Common::KEYCODE_3 == _buttonValue ||
Common::KEYCODE_4 == _buttonValue) {
_buttonValue -= Common::KEYCODE_1 - 7;
if ((_buttonValue - 7 + startingChar) < (int)_charList.size()) {
// Check if the selected character is already in the party
uint idx = 0;
for (; idx < party._activeParty.size(); ++idx) {
if (_charList[_buttonValue - 7 + startingChar] ==
party._activeParty[idx]._rosterId)
break;
}
// Only add the character if they're not already in the party
if (idx == party._activeParty.size()) {
if (party._activeParty.size() == MAX_ACTIVE_PARTY) {
sound.playFX(21);
ErrorScroll::show(_vm, Res.YOUR_PARTY_IS_FULL);
} else {
// Add the character to the active party
party._activeParty.push_back(party._roster[_charList[_buttonValue - 7 + startingChar]]);
startingCharChanged(startingChar);
}
}
}
} else if (Common::KEYCODE_UP == _buttonValue ||
Common::KEYCODE_KP8 == _buttonValue) {
// Up arrow
if (startingChar > 0) {
startingChar -= 4;
startingCharChanged(startingChar);
}
} else if (Common::KEYCODE_DOWN == _buttonValue ||
Common::KEYCODE_KP2 == _buttonValue) {
// Down arrow
if (startingChar < ((int)_charList.size() - 4)) {
startingChar += 4;
startingCharChanged(startingChar);
}
} else if (Res.KeyConstants.DialogsParty.KEY_EXIT == _buttonValue ||
Common::KEYCODE_ESCAPE == _buttonValue ||
Common::KEYCODE_SPACE == _buttonValue) {
if (party._activeParty.size() == 0) {
ErrorScroll::show(_vm, Res.NO_ONE_TO_ADVENTURE_WITH);
} else {
if (_vm->_mode != MODE_STARTUP) {
for (int idx = OBSCURITY_NONE; idx >= OBSCURITY_BLACK; --idx) {
events.updateGameCounter();
intf.obscureScene((Obscurity)idx);
w.update();
while (events.timeElapsed() < 1)
events.pollEventsAndWait();
}
}
w.close();
party._mazeId = party._priorMazeId;
party.copyPartyToRoster();
//_vm->_saves->writeCharFile();
return;
}
} else if (Res.KeyConstants.DialogsParty.KEY_CREATE == _buttonValue) {
// Create
if (_charList.size() == XEEN_TOTAL_CHARACTERS) {
ErrorScroll::show(_vm, Res.YOUR_ROSTER_IS_FULL);
} else {
// Save Background
Graphics::ManagedSurface savedBg;
savedBg.copyFrom(screen);
screen.fadeOut();
w.close();
// Show the create character dialog
CreateCharacterDialog::show(_vm);
party.copyPartyToRoster();
//_vm->_saves->writeCharFile();
// Restore Background
screen.fadeOut();
screen.blitFrom(savedBg);
windows[0].update();
#ifdef USE_TTS
returnFromCreate = true;
#endif
modeFlag = true;
breakFlag = true;
}
} else if (Res.KeyConstants.DialogsParty.KEY_DELETE == _buttonValue) {
// Delete character
if (_charList.size() > 0) {
int charButtonValue = selectCharacter(true, startingChar);
if (charButtonValue != 0) {
int charIndex = charButtonValue - Common::KEYCODE_1 + startingChar;
Character &c = party._roster[_charList[charIndex]];
if (c.hasSlayerSword()) {
ErrorScroll::show(_vm, Res.HAS_SLAYER_SWORD);
} else {
Common::String msg = Common::String::format(Res.SURE_TO_DELETE_CHAR,
c._name.c_str(), Res.CLASS_NAMES[c._class]);
if (Confirm::show(_vm, msg)) {
// If the character is in the party, remove it
for (uint idx = 0; idx < party._activeParty.size(); ++idx) {
if (party._activeParty[idx]._rosterId == c._rosterId) {
party._activeParty.remove_at(idx);
break;
}
}
// Empty the character in the roster
c.clear();
loadCharacters();
startingChar = 0;
startingCharChanged(startingChar);
}
}
}
}
} else if (Res.KeyConstants.DialogsParty.KEY_REMOVE == _buttonValue) {
// Remove character
if (party._activeParty.size() > 0) {
int charButtonValue = selectCharacter(false, startingChar);
if (charButtonValue != 0) {
party.copyPartyToRoster();
party._activeParty.remove_at(charButtonValue - Common::KEYCODE_F1);
}
startingCharChanged(startingChar);
}
}
}
}
}
void PartyDialog::loadCharacters() {
Map &map = *_vm->_map;
Party &party = *_vm->_party;
_charList.clear();
for (int i = 0; i < XEEN_TOTAL_CHARACTERS; ++i) {
Character &player = party._roster[i];
if (player._name.empty() || player._xeenSide != map._loadCcNum)
continue;
_charList.push_back(i);
}
}
void PartyDialog::loadButtons() {
_uiSprites.load("inn.icn");
addButton(Common::Rect(16, 100, 40, 120), Common::KEYCODE_UP, &_uiSprites, kPartyDialogUp);
addButton(Common::Rect(52, 100, 76, 120), Common::KEYCODE_DOWN, &_uiSprites, kPartyDialogDown);
addButton(Common::Rect(87, 100, 111, 120), Res.KeyConstants.DialogsParty.KEY_DELETE, &_uiSprites, kPartyDialogDelete);
addButton(Common::Rect(122, 100, 146, 120), Res.KeyConstants.DialogsParty.KEY_REMOVE, &_uiSprites, kPartyDialogRemove);
addButton(Common::Rect(157, 100, 181, 120), Res.KeyConstants.DialogsParty.KEY_CREATE, &_uiSprites, kPartyDialogCreate);
addButton(Common::Rect(192, 100, 216, 120), Res.KeyConstants.DialogsParty.KEY_EXIT, &_uiSprites, kPartyDialogExit);
addButton(Common::Rect(0, 0, 0, 0), Common::KEYCODE_ESCAPE);
}
void PartyDialog::initDrawStructs() {
_faceDrawStructs[0] = DrawStruct(0, 0, 0);
_faceDrawStructs[1] = DrawStruct(0, 101, 0);
_faceDrawStructs[2] = DrawStruct(0, 0, 43);
_faceDrawStructs[3] = DrawStruct(0, 101, 43);
}
void PartyDialog::setupBackground() {
_vm->_screen->loadBackground("back.raw");
_vm->_interface->assembleBorder();
}
void PartyDialog::setupFaces(int firstDisplayChar, bool updateFlag) {
Party &party = *_vm->_party;
Common::String charNames[4];
Common::String charRaces[4];
Common::String charSex[4];
Common::String charClasses[4];
int posIndex;
int charId;
// Reset the button areas for the display character images
while (_buttons.size() > 7)
_buttons.remove_at(7);
addButton(Common::Rect(16, 16, 48, 48), Common::KEYCODE_1, nullptr, kPartyDialogFace1);
addButton(Common::Rect(117, 16, 149, 48), Common::KEYCODE_2, nullptr, kPartyDialogFace2);
addButton(Common::Rect(59, 59, 91, 91), Common::KEYCODE_3, nullptr, kPartyDialogFace3);
addButton(Common::Rect(117, 59, 151, 91), Common::KEYCODE_4, nullptr, kPartyDialogFace4);
#ifdef USE_TTS
_faceCount = 0;
#endif
for (posIndex = 0; posIndex < 4; ++posIndex) {
charId = (firstDisplayChar + posIndex) >= (int)_charList.size() ? -1 :
_charList[firstDisplayChar + posIndex];
bool isInParty = party.isInParty(charId);
if (charId == -1) {
while ((int)_buttons.size() >(7 + posIndex))
_buttons.remove_at(_buttons.size() - 1);
break;
}
#ifdef USE_TTS
_faceCount++;
#endif
Common::Rect &b = _buttons[7 + posIndex]._bounds;
b.moveTo((posIndex & 1) ? 117 : 16, b.top);
Character &ps = party._roster[_charList[firstDisplayChar + posIndex]];
charNames[posIndex] = isInParty ? Res.IN_PARTY : ps._name;
charRaces[posIndex] = Res.RACE_NAMES[ps._race];
charSex[posIndex] = Res.SEX_NAMES[ps._sex];
charClasses[posIndex] = Res.CLASS_NAMES[ps._class];
}
drawParty(updateFlag);
// Set up the sprite set to use for each face
for (posIndex = 0; posIndex < 4; ++posIndex) {
if ((firstDisplayChar + posIndex) >= (int)_charList.size())
_faceDrawStructs[posIndex]._sprites = nullptr;
else
_faceDrawStructs[posIndex]._sprites = party._roster[
_charList[firstDisplayChar + posIndex]]._faceSprites;
}
_partyDetails = Common::String::format(Res.PARTY_DETAILS,
charNames[0].c_str(), charRaces[0].c_str(), charSex[0].c_str(), charClasses[0].c_str(),
charNames[1].c_str(), charRaces[1].c_str(), charSex[1].c_str(), charClasses[1].c_str(),
charNames[2].c_str(), charRaces[2].c_str(), charSex[2].c_str(), charClasses[2].c_str(),
charNames[3].c_str(), charRaces[3].c_str(), charSex[3].c_str(), charClasses[3].c_str()
);
}
void PartyDialog::startingCharChanged(int firstDisplayChar) {
Windows &windows = *_vm->_windows;
Window &w = windows[11];
setupFaces(firstDisplayChar, true);
Common::String ttsMessage;
w.writeString(Common::String::format(Res.PARTY_DIALOG_TEXT, _partyDetails.c_str()), false, &ttsMessage);
#ifdef USE_TTS
speakText(ttsMessage, false);
#endif
w.drawList(_faceDrawStructs, 4);
_uiSprites.draw(w, 0, Common::Point(16, 100));
_uiSprites.draw(w, 2, Common::Point(52, 100));
_uiSprites.draw(w, 4, Common::Point(87, 100));
_uiSprites.draw(w, 6, Common::Point(122, 100));
_uiSprites.draw(w, 8, Common::Point(157, 100));
_uiSprites.draw(w, 10, Common::Point(192, 100));
w.update();
}
int PartyDialog::selectCharacter(bool isDelete, int firstDisplayChar) {
EventsManager &events = *_vm->_events;
Party &party = *_vm->_party;
Windows &windows = *_vm->_windows;
Window &w = windows[28];
SpriteResource iconSprites;
iconSprites.load("esc.icn");
w.setBounds(Common::Rect(50, isDelete ? 112 : 76, 266, isDelete ? 148 : 112));
w.open();
w.writeString(Common::String::format(Res.REMOVE_OR_DELETE_WHICH,
Res.REMOVE_DELETE[isDelete ? 1 : 0]));
iconSprites.draw(w, 0, Common::Point(225, isDelete ? 120 : 84));
w.update();
saveButtons();
addButton(Common::Rect(225, isDelete ? 120 : 84, 249, isDelete ? 140 : 104),
Common::KEYCODE_ESCAPE, &iconSprites);
addButton(Common::Rect(16, 16, 48, 48), Common::KEYCODE_1, nullptr, kPartyDialogFace1);
addButton(Common::Rect(117, 16, 149, 48), Common::KEYCODE_2, nullptr, kPartyDialogFace2);
addButton(Common::Rect(16, 59, 48, 91), Common::KEYCODE_3, nullptr, kPartyDialogFace3);
addButton(Common::Rect(117, 59, 149, 91), Common::KEYCODE_4, nullptr, kPartyDialogFace4);
addPartyButtons(_vm);
int result = -1, v;
while (!_vm->shouldExit() && result == -1) {
_buttonValue = 0;
while (!_vm->shouldExit() && !_buttonValue) {
events.pollEventsAndWait();
checkEvents(_vm);
}
switch (_buttonValue) {
case Common::KEYCODE_ESCAPE:
result = 0;
break;
case Common::KEYCODE_F1:
case Common::KEYCODE_F2:
case Common::KEYCODE_F3:
case Common::KEYCODE_F4:
case Common::KEYCODE_F5:
case Common::KEYCODE_F6:
if (!isDelete) {
v = _buttonValue - Common::KEYCODE_F1;
if (v < (int)party._activeParty.size())
result = _buttonValue;
}
break;
case Common::KEYCODE_1:
case Common::KEYCODE_2:
case Common::KEYCODE_3:
case Common::KEYCODE_4:
if (isDelete) {
v = _buttonValue - Common::KEYCODE_1;
if ((firstDisplayChar + v) < (int)_charList.size())
result = _buttonValue;
}
break;
default:
break;
}
}
w.close();
restoreButtons();
return result == -1 ? 0 : result;
}
#ifdef USE_TTS
void PartyDialog::speakText(const Common::String &text, bool isCharacterCreation) {
uint index = 0;
// After returning from the character creation screen, the text is duplicated, with the first iteration being
// outdated
if (isCharacterCreation) {
for (uint i = 0; i < _faceCount; ++i) {
getNextTextSection(text, index, kPartyDialogInfoPerFaceCount);
}
getNextTextSection(text, index, kPartyDialogButtonCount);
}
_buttonTexts.clear();
Common::String faceTexts[kPartyDialogFaceCount];
for (uint8 i = 0; i < _faceCount; ++i) {
faceTexts[i] = getNextTextSection(text, index, kPartyDialogInfoPerFaceCount);
_vm->sayText(faceTexts[i]);
}
Common::String buttonTexts = addNextTextToButtons(text, index, kPartyDialogButtonCount);
_vm->sayText(buttonTexts);
// Since the number of faces can vary, sort them after the bottom buttons
for (uint8 i = 0; i < _faceCount; ++i) {
_buttonTexts.push_back(faceTexts[i]);
}
}
#endif
} // End of namespace Xeen
} // End of namespace MM