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

586 lines
15 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/>.
*
* Copyright 2020 Google
*
*/
#include "common/file.h"
#include "hadesch/video.h"
#include "hadesch/pod_image.h"
#include "hadesch/tag_file.h"
#include "hadesch/hadesch.h"
#include "common/system.h"
#include "video/smk_decoder.h"
#include "audio/decoders/aiff.h"
#include "hadesch/pod_file.h"
#include "hadesch/baptr.h"
#include "common/translation.h"
namespace Hadesch {
static const int kHeroBeltMinY = 378;
static const int kHeroBeltMaxY = 471;
static const TranscribedSound powerSounds[3][2] = {
{
{ "g0280nc0", _hs("That's where hero power of strength will be stored when you earn it") },
{ "g0280ng0", _hs("The power of strength will let you overcome obstacles but you'll need a complete set of hero powers to use it") }
},
{
{ "g0280nb0", _hs("That's where hero power of stealth will be stored when you earn it") },
{ "g0280nf0", _hs("The power of stealth allows you to sneak past things but you'll need a complete set of hero powers to use it") }
},
{
{ "g0280ne0", _hs("That's where hero power of wisdom will be stored when you earn it") },
{ "g0280nh0", _hs("The power of wisdom will let you outwit and avoid deception but you'll need a complete set of hero powers to use it") }
},
};
static Common::Array <PodImage> loadImageArray(const Common::String &name) {
Common::SharedPtr<Common::SeekableReadStream> rs(g_vm->getWdPodFile()->getFileStream(name + ".pod"));
PodFile pf2(name);
pf2.openStore(rs);
return pf2.loadImageArray();
}
static PodImage loadImage(const Common::String &name) {
return loadImageArray(name)[0];
}
static const struct {
const char *background;
const char *iconNames;
const char *icons;
// TODO: figure out how the original handles
// cursors.
const char *iconCursors;
const char *iconCursorsBright;
const char *scrollBg;
const char *scrollBgHades;
const char *scrollTextCrete;
const char *scrollTextTroyFemale;
const char *scrollTextTroyMale;
const char *scrollTextMedusa;
const char *scrollTextHades;
const char *powerImageWisdom;
const char *powerImageStrength;
const char *powerImageStealth;
} assetTable[3] = {
// Warm
{
"g0010oa0",
"g0091ba0",
"g0091bb0",
"g0091bc0",
"g0093bc0",
"g0015oy0",
"g0015oy3",
"g0015td0",
"g0015te0",
"g0015te3",
"g0015tf0",
"g0015tg0",
"g0290oa0",
"g0290ob0",
"g0290oc0"
},
// Cold
{
"g0010ob0",
"g0092ba0",
"g0092bb0",
"g0091bc0",
"g0093bc0",
"g0015oy2",
"g0015oy5",
"g0015td2",
"g0015te2",
"g0015te5",
"g0015tf2",
"g0015tg2",
"g0092oa0",
"g0092ob0",
"g0092oc0"
},
// Cool
{
"g0010oc0",
"g0093ba0",
"g0093bb0",
"g0091bc0",
"g0093bc0",
"g0015oy1",
"g0015oy4",
"g0015td1",
"g0015te1",
"g0015te4",
"g0015tf1",
"g0015tg1",
"g0093oa0",
"g0093ob0",
"g0093oc0"
}
};
HeroBelt::HeroBelt() {
for (int i = 0; i < 3; i++) {
_background[i] = loadImage(assetTable[i].background);
_iconNames[i] = loadImageArray(assetTable[i].iconNames);
_icons[i] = loadImageArray(assetTable[i].icons);
_iconCursors[i] = loadImageArray(assetTable[i].iconCursors);
_iconCursorsBright[i] = loadImageArray(assetTable[i].iconCursorsBright);
_powerImages[0][i] = loadImageArray(assetTable[i].powerImageStrength);
_powerImages[1][i] = loadImageArray(assetTable[i].powerImageStealth);
_powerImages[2][i] = loadImageArray(assetTable[i].powerImageWisdom);
_scrollBg[i] = loadImage(assetTable[i].scrollBg);
_scrollBgHades[i] = loadImage(assetTable[i].scrollBgHades);
_scrollTextCrete[i] = loadImage(assetTable[i].scrollTextCrete);
_scrollTextTroyFemale[i] = loadImage(assetTable[i].scrollTextTroyFemale);
_scrollTextTroyMale[i] = loadImage(assetTable[i].scrollTextTroyMale);
_scrollTextMedusa[i] = loadImage(assetTable[i].scrollTextMedusa);
_scrollTextHades[i] = loadImage(assetTable[i].scrollTextHades);
}
_branchOfLife = loadImageArray("v7150ba0");
_overHeroBelt = false;
_heroBeltY = kHeroBeltMaxY;
_bottomEdge = false;
_heroBeltSpeed = 0;
_edgeStartTime = 0;
_animateItem = kNone;
_animateItemTargetSlot = -1;
_hotZone = -1;
_holdingItem = kNone;
_holdingSlot = -1;
_highlightTextIdx = -1;
_showScroll = false;
_hotZones.readHotzones(
Common::SharedPtr<Common::SeekableReadStream>(g_vm->getWdPodFile()->getFileStream("HeroBelt.HOT")),
true);
}
void HeroBelt::computeHotZone(int time, Common::Point mousePos, bool mouseEnabled) {
bool wasBottomEdge = _bottomEdge;
_overHeroBelt = false;
_bottomEdge = false;
_mousePos = mousePos;
if (!mouseEnabled) {
return;
}
_bottomEdge = mousePos.y > 460;
_overHeroBelt = (_bottomEdge && _heroBeltSpeed < 0) || mousePos.y > _heroBeltY;
if (!wasBottomEdge && _bottomEdge)
_edgeStartTime = time;
_currentTime = time;
int wasHotZone = _hotZone;
_hotZone = _hotZones.pointToIndex(mousePos);
if (_hotZone >= 0 && wasHotZone < 0) {
_startHotTime = time;
}
computeHighlight();
}
static bool isInFrieze() {
Persistent *persistent = g_vm->getPersistent();
return (persistent->_currentRoomId == kMinotaurPuzzle && persistent->_quest != kCreteQuest)
|| (persistent->_currentRoomId == kTrojanHorsePuzzle && persistent->_quest != kTroyQuest)
|| (persistent->_currentRoomId == kMedusaPuzzle && persistent->_quest != kMedusaQuest)
|| (persistent->_currentRoomId == kFerrymanPuzzle && persistent->_quest != kRescuePhilQuest)
|| (persistent->_currentRoomId == kMonsterPuzzle && persistent->_quest != kRescuePhilQuest);
}
void HeroBelt::render(Common::SharedPtr<GfxContext> context, int time, Common::Point viewPoint) {
Persistent *persistent = g_vm->getPersistent();
Common::Point beltPoint;
// TODO: use beltopen hotzone instead?
if (persistent->_currentRoomId == kMonsterPuzzle) {
_heroBeltY = kHeroBeltMinY;
} else {
if (_bottomEdge && _heroBeltY == kHeroBeltMaxY
&& (time > _edgeStartTime + 500)) {
_heroBeltSpeed = -10;
}
if (!_overHeroBelt && _heroBeltY != kHeroBeltMaxY
&& _animateItemTargetSlot == -1) {
_showScroll = false;
_heroBeltSpeed = +10;
}
if (_heroBeltSpeed != 0) {
_heroBeltY += _heroBeltSpeed;
if (_heroBeltY <= kHeroBeltMinY) {
_heroBeltY = kHeroBeltMinY;
_heroBeltSpeed = 0;
}
if (_heroBeltY >= kHeroBeltMaxY) {
_heroBeltY = kHeroBeltMaxY;
_heroBeltSpeed = 0;
}
}
}
_currentTime = time;
beltPoint = Common::Point(0, _heroBeltY) + viewPoint;
_background[_colour].render(context, beltPoint);
if (_animateItem != kNone) {
if (_currentTime > _animateItemStartTime + _animItemTime) {
_animateItem = kNone;
_animateItemTargetSlot = -1;
_animItemCallbackEvent();
}
}
if (_animateItem != kNone) {
Common::Point target = computeSlotPoint(_animateItemTargetSlot, true);
Common::Point delta1 = target - _animateItemStartPoint;
int elapsed = _currentTime - _animateItemStartTime;
double fraction = ((elapsed + 0.0) / _animItemTime);
Common::Point deltaPartial = fraction * delta1;
Common::Point cur = _animateItemStartPoint + deltaPartial;
_iconCursors[_colour][_animateItem - 1].render(
context, cur + viewPoint);
}
if (persistent->_currentRoomId == kMonsterPuzzle) {
_icons[_colour][kNumberI].render(
context,
computeSlotPoint(0, false) + viewPoint
+ Common::Point(0, 4));
_icons[_colour][kNumberII].render(
context,
computeSlotPoint(1, false) + viewPoint);
_icons[_colour][kNumberIII].render(
context,
computeSlotPoint(2, false) + viewPoint
+ Common::Point(0, 4));
for (unsigned i = 0; i < 3; i++) {
_icons[_colour][_thunderboltFrame].render(
context,
computeSlotPoint(3 + i, false) + viewPoint);
}
_branchOfLife[_branchOfLifeFrame].render(
context, beltPoint);
} else {
for (int i = 0; i < inventorySize; i++) {
if (i == _animateItemTargetSlot || i == _holdingSlot
|| persistent->_inventory[i] == kNone) {
continue;
}
_icons[_colour][persistent->_inventory[i] - 1].render(
context,
computeSlotPoint(i, false) + viewPoint);
}
}
_icons[_colour][persistent->_hintsAreEnabled ? kActiveHints : kInactiveHints].render(
context,
computeSlotPoint(6, false) + viewPoint);
if (isInFrieze())
_icons[_colour][kReturnToWall].render(
context,
computeSlotPoint(7, false) + viewPoint);
else
_icons[_colour][kQuestScroll].render(
context,
computeSlotPoint(7, false) + viewPoint);
_icons[_colour][kOptionsButton].render(
context,
computeSlotPoint(8, false) + viewPoint);
if (_highlightTextIdx >= 0) {
_iconNames[_colour][_highlightTextIdx].render(
context, beltPoint);
}
for (unsigned i = 0; i < ARRAYSIZE(persistent->_powerLevel); i++)
if (persistent->_powerLevel[i] > 0) {
_powerImages[i][_colour][(int)i == _selectedPower].render(
context, beltPoint);
}
if (_showScroll) {
PodImage *bg = _scrollBg, *text = nullptr;
switch (persistent->_quest) {
case kCreteQuest:
text = _scrollTextCrete;
break;
case kTroyQuest:
if (persistent->_gender == kMale)
text = _scrollTextTroyMale;
else
text = _scrollTextTroyFemale;
break;
case kMedusaQuest:
text = _scrollTextMedusa;
break;
case kRescuePhilQuest:
text = _scrollTextHades;
bg = _scrollBgHades;
break;
case kEndGame:
case kNoQuest:
case kNumQuests:
break;
}
bg[_colour].render(context, viewPoint);
if (text != nullptr)
text[_colour].render(
context, viewPoint);
}
}
bool HeroBelt::isPositionOverHeroBelt(Common::Point mousePos) const {
return mousePos.y > _heroBeltY;
}
void HeroBelt::handleClick(Common::Point mousePos) {
Persistent *persistent = g_vm->getPersistent();
Common::String q = _hotZones.pointToName(mousePos);
debug("handling belt click on <%s>", q.c_str());
for (int i = 0; i < inventorySize; i++) {
if (q == inventoryName(i)) {
if (_holdingItem != kNone) {
if (persistent->_inventory[i] != kNone &&
_holdingSlot != i) {
g_vm->fallbackClick();
return;
}
persistent->_inventory[_holdingSlot] = kNone;
persistent->_inventory[i] = _holdingItem;
_holdingItem = kNone;
_holdingSlot = -1;
return;
}
if (i == _animateItemTargetSlot
|| persistent->_inventory[i] == kNone)
return;
_holdingItem = persistent->_inventory[i];
_holdingSlot = i;
return;
}
}
if (q == "quest scroll") {
if (isInFrieze())
g_vm->moveToRoom(kWallOfFameRoom);
else
_showScroll = true;
}
if (q == "hints") {
persistent->_hintsAreEnabled = !persistent->_hintsAreEnabled;
}
if (q == "options") {
g_vm->enterOptions();
}
if (q == "strength")
clickPower(kPowerStrength);
if (q == "stealth")
clickPower(kPowerStealth);
if (q == "wisdom")
clickPower(kPowerWisdom);
}
void HeroBelt::clickPower(HeroPower pwr) {
Persistent *persistent = g_vm->getPersistent();
Common::SharedPtr<VideoRoom> room = g_vm->getVideoRoom();
if (persistent->_currentRoomId == kMonsterPuzzle) {
_selectedPower = pwr;
return;
}
if (persistent->_quest == kRescuePhilQuest)
return;
room->playSpeech(powerSounds[pwr][!!persistent->_powerLevel[pwr]]);
}
void HeroBelt::computeHighlight() {
Common::String hotZone = _hotZones.indexToName(_hotZone);
Persistent *persistent = g_vm->getPersistent();
for (int i = 0; i < inventorySize; i++) {
if (hotZone == inventoryName(i)) {
if (persistent->_currentRoomId == kMonsterPuzzle) {
_highlightTextIdx = i < 3 ? kNumberI + i : kLightning1 + i - 3;
return;
} else {
if (persistent->_inventory[i] != kNone &&
_holdingSlot != i) {
_highlightTextIdx = persistent->_inventory[i] - 1;
return;
}
}
}
}
if (hotZone == "quest scroll") {
// TODO: what's with 30 and 28?
if (isInFrieze())
_highlightTextIdx = kReturnToWall;
else if (persistent->_quest == kRescuePhilQuest)
_highlightTextIdx = kHadesScroll;
else
_highlightTextIdx = kQuestScroll;
return;
}
if (hotZone == "stealth") {
_highlightTextIdx = kPowerOfStealth;
return;
}
if (hotZone == "strength") {
_highlightTextIdx = kPowerOfStrength;
return;
}
if (hotZone == "wisdom") {
_highlightTextIdx = kPowerOfWisdom;
return;
}
if (hotZone == "hints") {
_highlightTextIdx = persistent->_hintsAreEnabled ? kActiveHints : kInactiveHints;
return;
}
if (hotZone == "options") {
_highlightTextIdx = kOptionsButton;
return;
}
_highlightTextIdx = -1;
}
void HeroBelt::placeToInventory(InventoryItem item, EventHandlerWrapper callbackEvent) {
Persistent *persistent = g_vm->getPersistent();
unsigned i;
for (i = 0; i < inventorySize; i++) {
if (persistent->_inventory[i] == kNone)
break;
}
if (i == inventorySize) {
debug("Out of inventory space");
return;
}
persistent->_inventory[i] = item;
_animateItem = item;
_animItemCallbackEvent = callbackEvent;
_animateItemStartPoint = _mousePos;
_animateItemTargetSlot = i;
_animateItemStartTime = _currentTime;
_animItemTime = 2000;
_heroBeltSpeed = -10;
}
void HeroBelt::removeFromInventory(InventoryItem item) {
Persistent *persistent = g_vm->getPersistent();
for (unsigned i = 0; i < inventorySize; i++) {
if (persistent->_inventory[i] == item) {
persistent->_inventory[i] = kNone;
}
}
if (_holdingItem == item) {
_holdingItem = kNone;
_holdingSlot = -1;
}
if (_animateItem == item) {
_animateItem = kNone;
_animateItemTargetSlot = -1;
}
}
void HeroBelt::clearHold() {
_holdingItem = kNone;
_holdingSlot = -1;
}
Common::Point HeroBelt::computeSlotPoint(int slot, bool fullyExtended) {
Common::Point ret = Common::Point(19, 35);
ret += Common::Point(0, (fullyExtended ? kHeroBeltMinY : _heroBeltY));
ret += Common::Point(39 * slot, slot % 2 ? 4 : 0);
if (slot >= 6) {
ret += Common::Point(253, 0);
}
return ret;
}
int HeroBelt::getCursor(int time) {
Common::String q = _hotZones.indexToName(_hotZone);
Persistent *persistent = g_vm->getPersistent();
if (q == "")
return 0;
for (unsigned i = 0; i < inventorySize; i++) {
if (q == inventoryName(i)) {
if ((int) i != _animateItemTargetSlot
&& persistent->_inventory[i] != kNone)
return (time - _startHotTime) / kDefaultSpeed % 3;
return 0;
}
}
return (time - _startHotTime) / kDefaultSpeed % 3;
}
Common::String HeroBelt::inventoryName(int slot) {
return Common::String::format("inventory%d", slot);
}
bool HeroBelt::isHoldingItem() const {
return _holdingItem != kNone;
}
const Graphics::Cursor *HeroBelt::getHoldingItemCursor(int cursorAnimationFrame) const {
if ((cursorAnimationFrame / 2) % 2 == 1)
return &_iconCursorsBright[_colour][_holdingItem - 1];
else
return &_iconCursors[_colour][_holdingItem - 1];
}
}