scummvm/engines/saga2/intrface.cpp
2022-10-29 00:35:07 +02:00

2965 lines
87 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/>.
*
*
* Based on the original sources
* Faery Tale II -- The Halls of the Dead
* (c) 1993-1996 The Wyrmkeep Entertainment Co.
*/
#include "saga2/saga2.h"
#include "saga2/detection.h"
#include "saga2/blitters.h"
#include "saga2/objects.h"
#include "saga2/contain.h"
#include "saga2/intrface.h"
#include "saga2/grabinfo.h"
#include "saga2/uidialog.h"
#include "saga2/motion.h"
#include "saga2/enchant.h"
#include "saga2/display.h"
#include "saga2/localize.h"
#include "saga2/imagcach.h"
#include "saga2/fontlib.h"
#include "saga2/uimetrcs.h"
namespace Saga2 {
/* ===================================================================== *
External delarations
* ===================================================================== */
extern ReadyContainerView *TrioCviews[kNumViews];
extern ReadyContainerView *indivCviewTop, *indivCviewBot;
extern gPanelList *trioControls, *indivControls;
extern gPanelList *playControls;
extern const uint32 imageGroupID;
extern gPanelList *tileControls;
extern BackWindow *mainWindow;
extern uint8 fixedColors[16];
/* ===================================================================== *
Classes
* ===================================================================== */
// Private subclass of GfxCompImage for armor display
class gArmorIndicator : public GfxCompImage {
public:
ArmorAttributes _attr;
void drawClipped(gPort &,
const Point16 &,
const Rect16 &) override;
void setValue(PlayerActorID pID);
gArmorIndicator(gPanelList &list, const Rect16 &box, void *img, uint16 ident, AppFunc *cmd = nullptr)
: GfxCompImage(list, box, img, ident, cmd) {
_attr.damageAbsorbtion = 0;
_attr.damageDivider = 1;
_attr.defenseBonus = 0;
}
};
// Private subclass of gControl for enchantment display
// REM: Some of this should probably be moved to a header file, even though no-one
// is ever going to need it other than this file.
enum enchantmentIcons {
iconInvisible = 0,
iconDetectPoison,
iconSoulSight,
iconPoisoned,
iconParalysed,
iconClumsy,
iconFrozen,
iconConstrained,
iconAfraid,
iconSeawalk,
iconHaste,
iconFirewalk,
iconSurestrike,
iconAdrenalFervor,
iconInnerBalance,
iconShadowWalk,
iconBattleFever,
iconSunWard,
iconNetherWard,
iconSpellBarrier,
iconIronskin,
iconNumbscent,
iconProtectEvil,
iconProtectUndead,
iconProtectGhosts,
iconCushionAir,
iconResistImpact,
iconResistSlash,
iconResistProjectile,
iconResistFire,
iconResistAcid,
iconResistHeat,
iconResistCold,
iconResistLightning,
iconResistPoison,
iconResistPsionic,
iconResistDirectMagic,
iconImmuneFire,
iconImmuneAcid,
iconImmuneHeat,
iconImmuneCold,
iconImmuneLightning,
iconImmunePoison,
iconImmunePsionic,
iconCount
};
const char *enchantmentNames[] = {
"Invisible",
"Detect Poison",
"Soul Sight",
"Poisoned",
"Paralysed",
"Clumsy",
"Frozen",
"Constrained",
"Afraid",
"Seawalk",
"Haste",
"Fire Walk",
"Surestrike",
"Adrenal Fervor",
"Inner Balance",
"Shadow Walk",
"Battle Fever",
"Sun Ward",
"Nether Ward",
"Spell Barrier",
"Ironskin",
"Numbscent",
"Protection from Evil",
"Protection from Undead",
"Protection from Ghosts",
"Cushion of Air",
"Resist impact damage",
"Resist slashing damage",
"Resist projectile damage",
"Resist Fire / Yellow Magic",
"Resist Acid / Violet Magic",
"Resist Heat / Red Magic",
"Resist Cold / Blue Magic",
"Resist Lightning / Orange Magic",
"Resist Poison / Green Magic",
"Resist Mental Damage",
"Resist Direct Magic",
"Immunity to Fire / Yellow Magic",
"Immunity to Acid / Violet Magic",
"Immunity to Heat / Red Magic",
"Immunity to Cold / Blue Magic",
"Immunity to Lightning / Orange Magic",
"Immunity to Poison / Green Magic",
"Immunity to Mental Damage",
nullptr
};
class gEnchantmentDisplay : public gControl {
uint8 _iconFlags[iconCount];
void drawClipped(gPort &, const Point16 &, const Rect16 &) override;
void pointerMove(gPanelMessage &msg) override;
public:
void setValue(PlayerActorID pID);
gEnchantmentDisplay(gPanelList &list, uint16 ident, AppFunc *cmd = nullptr)
: gControl(list, Rect16(0, 0, 630, 18), nullptr, ident, cmd) {
memset(_iconFlags, 0, sizeof(_iconFlags));
}
};
/* ===================================================================== *
Imports
* ===================================================================== */
// this is a redeclare of the struct in playmode.cpp
typedef struct {
Point16 size;
int16 compress;
int8 data[2];
} ImageHeader;
// external appfuncs
APPFUNC(cmdBrain);
/* ===================================================================== *
User interface application functions
* ===================================================================== */
APPFUNC(cmdPortrait);
APPFUNC(cmdAggressive);
//APPFUNC( cmdJump );
APPFUNC(cmdArmor);
APPFUNC(cmdCenter);
APPFUNC(cmdBand);
APPFUNC(cmdOptions);
APPFUNC(cmdBroChange);
APPFUNC(cmdHealthStar);
APPFUNC(cmdMassInd);
APPFUNC(cmdBulkInd);
APPFUNC(cmdManaInd);
/* ===================================================================== *
User control metrics
* ===================================================================== */
// position arrays for all _buttons on the individual panels
static const StaticRect topBox[numButtons] = {
/* portrait */ { 489, 22 + (yContOffset * 0), 65, 72 },
/* agress */ { 559, 86 + (yContOffset * 0), 28, 27 },
/* jump */ { 592, 86 + (yContOffset * 0), 28, 27 },
/* center */ { 559, 56 + (yContOffset * 0), 28, 27 },
/* banding */ { 592, 56 + (yContOffset * 0), 28, 27 },
/* namePlates */ { 488, 94 + (yFaceOffset * 0), 65, 15 },
/* namePlateFrames */ { 487, 20 + (yFaceOffset * 0), 69, 92 }
};
static const StaticRect midBox[numButtons] = {
{ 489, 22 + (yContOffset * 1), 65, 72 },
{ 559, 86 + (yContOffset * 1), 28, 27 },
{ 592, 86 + (yContOffset * 1), 28, 27 },
{ 559, 56 + (yContOffset * 1), 28, 27 },
{ 592, 56 + (yContOffset * 1), 28, 27 },
{ 488, 94 + (yFaceOffset * 1), 65, 15 },
{ 487, 20 + (yFaceOffset * 1), 69, 92 }
};
static const StaticRect botBox[numButtons] = {
{ 489, 22 + (yContOffset * 2), 65, 72 },
{ 559, 86 + (yContOffset * 2), 28, 27 },
{ 592, 86 + (yContOffset * 2), 28, 27 },
{ 559, 56 + (yContOffset * 2), 28, 27 },
{ 592, 56 + (yContOffset * 2), 28, 27 },
{ 488, 94 + (yFaceOffset * 2), 65, 15 },
{ 487, 20 + (yFaceOffset * 2), 69, 92 }
};
// options button
GfxCompButton *optBtn;
gEnchantmentDisplay *enchDisp;
// brother _buttons
GfxOwnerSelCompButton *julBtn;
GfxOwnerSelCompButton *phiBtn;
GfxOwnerSelCompButton *kevBtn;
GfxCompImage *broBtnFrame;
// trio controls
GfxMultCompButton *portBtns[kNumViews];
GfxOwnerSelCompButton *aggressBtns[kNumViews];
//GfxCompButton *jumpBtns[kNumViews];
GfxOwnerSelCompButton *centerBtns[kNumViews];
GfxOwnerSelCompButton *bandingBtns[kNumViews];
GfxCompImage *namePlates[kNumViews];
GfxCompImage *namePlateFrames[kNumViews];
gArmorIndicator *armorInd[kNumViews];
// individual
GfxMultCompButton *indivPortBtn;
GfxOwnerSelCompButton *indivAggressBtn;
//GfxCompButton *indivJumpBtn;
GfxOwnerSelCompButton *indivCenterBtn;
GfxOwnerSelCompButton *indivBandingBtn;
GfxCompImage *indivNamePlate;
GfxCompImage *indivNamePlateFrame;
gArmorIndicator *indivArmorInd;
// mental button/indicators
GfxCompButton *menConBtn;
// [brother panels] compressed image non-allocated pointer arrays
void **portImag[kNumViews];
void **aggressImag;
//void **jumpImag;
void **centerImag;
void **bandingImag;
void **menConBtnImag;
void **optBtnImag;
void **julBtnImag;
void **phiBtnImag;
void **kevBtnImag;
// portrait name plate things
void *namePlateImages[kNumViews];
void *namePlateFrameImag;
void *armorImag;
// brother selection button frame
void *broBtnFrameImag;
// name plate resource indexes
int16 namePlateResNum[] = { 8, 9, 10 };
// number of plate images
int16 numNamePlateRes[] = { 1, 1, 1 };
// brother resource names
ResName broNames[] = {
{ 'J', 'U', 'L', 0 },
{ 'P', 'H', 'I', 0 },
{ 'K', 'E', 'V', 0 }
};
const uint32 containerGroupID = MKTAG('C', 'O', 'N', 'T');
// button position views
// topBox, midBox, and botBox are defined in uimetrcs.h
static const StaticRect *views[] = {
topBox,
midBox,
botBox
};
// individual indicators/_buttons
static const StaticRect menConBtnRect = {485, 265, 44, 43};
// options button
static const StaticRect optBtnRect = {20, 445, 26, 15};
// brother _buttons and frame
static const StaticRect broBtnRect = {481, 450, 144, 11};
static const StaticRect julBtnRect = {482, 451, 44, 9};
static const StaticRect phiBtnRect = {531, 451, 44, 9};
static const StaticRect kevBtnRect = {580, 451, 44, 9};
StaticTextPallete genericTextPal = {9 + 15, 20, 14, 11, 23, 17};
/* uint8 dlPen;
uint8 urPen;
uint8 inPen;
uint8 dlHilitePen;
uint8 urHilitePen;
uint8 inHilitePen; */
// [Added here by Talin to fix problem in Win32. I don't know if it's the right place]
class CPortrait *Portrait;
/* ===================================================================== *
ui controls
* ===================================================================== */
// which brother is set in the individual controls
uint16 indivBrother;
/* ===================================================================== *
Global class declarations
* ===================================================================== */
// these objhets have to be initialized after resource are initialized
CStatusLine *StatusLine = nullptr;
CMassWeightIndicator *MassWeightIndicator = nullptr;
CHealthIndicator *HealthIndicator = nullptr;
CManaIndicator *ManaIndicator = nullptr;
/* ===================================================================== *
Plaq writing class
* ===================================================================== */
CPlaqText::CPlaqText(gPanelList &list,
const Rect16 &box,
const char *msg,
gFont *font,
int16 textPos,
textPallete &pal,
int16 ident,
AppFunc *cmd)
: gControl(list, box, msg, ident, cmd) {
if (strlen(msg) < kBufSize) {
Common::strcpy_s(_lineBuf, msg);
} else {
*_lineBuf = '\0';
}
_textFacePal = pal;
_buttonFont = font;
_textRect = box;
_textPosition = textPos;
_oldFont = nullptr;
}
void CPlaqText::enable(bool abled) {
gPanel::enable(abled);
}
void CPlaqText::invalidate(Rect16 *) {
_window.update(_extent);
}
void CPlaqText::draw() {
gPort &port = _window._windowPort;
Rect16 rect = _window.getExtent();
// save pen color, etc.
SAVE_GPORT_STATE(port);
_oldFont = port._font;
// setup the port
port.setMode(kDrawModeMatte);
port.setFont(_buttonFont);
g_vm->_pointer->hide(port, _extent); // hide mouse pointer
drawClipped(port, Point16(0, 0), Rect16(0, 0, rect.width, rect.height));
g_vm->_pointer->show(port, _extent); // show mouse pointer
// reset the old font
port.setFont(_oldFont);
}
void CPlaqText::drawClipped(gPort &port,
const Point16 &offset,
const Rect16 &r) {
if (_extent.overlap(r)) {
if (*_lineBuf) {
_textRect = _extent;
_textRect.x -= offset.x;
_textRect.y -= offset.y;
writePlaqText(port, _textRect, _buttonFont, _textPosition, _textFacePal, _selected, _lineBuf);
}
}
}
/* ===================================================================== *
Portrait control class
* ===================================================================== */
CPortrait::CPortrait(GfxMultCompButton **portraits,
GfxMultCompButton *indivPort,
const uint16 numPorts,
uint16 numBrothers) { // numBrothers = post 1
// do some checking
assert(portraits);
assert(indivPort);
for (uint16 i = 0; i < numBrothers; i++) {
assert(portraits[i]);
};
_buttons = portraits; // set the pointer for class
_indivButton = indivPort; // set the individual portrait
_numButtons = numPorts; // number of buttons per pointer
_numViews = numBrothers; // number of pointers for whole array
// start off in a normal facial state
for (uint16 i = 0; i < _numViews + 1; i++) {
_currentState[i] = kPortraitNormal;
}
}
void CPortrait::setPortrait(uint16 brotherID) {
assert(brotherID < _numViews + 1);
// tell button to select and display new image
if (brotherID == kUiIndiv) {
WriteStatusF(4, " Brother id %d", brotherID);
_indivButton->setCurrent(_currentState[brotherID]);
_indivButton->invalidate();
} else {
_buttons[brotherID]->setCurrent(_currentState[brotherID]);
_buttons[brotherID]->invalidate();
}
}
void CPortrait::set(uint16 brotherID, PortraitType type) {
_currentState[brotherID] = type;
setPortrait(brotherID);
}
void CPortrait::ORset(uint16 brotherID, PortraitType type) { // brotherID = post 0
assert(brotherID < _numViews + 1);
if (type == _currentState[brotherID]) {
_currentState[brotherID] = kPortraitNormal;
} else {
_currentState[brotherID] = type;
}
// set this button to the new state
setPortrait(brotherID);
}
void CPortrait::getStateString(char buf[], int8 size, uint16 brotherID) {
static char asleepStr[] = ASLEEP_STATE;
static char paralysedStr[] = PARALY_STATE;
static char blindStr[] = BLIND_STATE ;
static char afraidStr[] = AFRAID_STATE;
static char angryStr[] = ANGRY_STATE ;
static char badlyWoundedStr[] = BADWND_STATE;
static char hurtStr[] = HURT_STATE ;
static char poisonedStr[] = POISON_STATE;
static char diseasedStr[] = DISEAS_STATE;
static char normalStr[] = NORMAL_STATE;
static char commaStr[] = ", ";
PlayerActor *pa = getPlayerActorAddress(brotherID);
Actor *a = pa->getActor();
ActorAttributes &stats = pa->getBaseStats();
buf[size - 1] = '\0';
if (a->isDead()) {
Common::strlcpy(buf, DEAD_STATE, size);
return;
}
buf[0] = '\0';
if (a->_enchantmentFlags & (1 << kActorAsleep)) {
Common::strlcat(buf, asleepStr, size);
} else if (a->_enchantmentFlags & (1 << kActorParalyzed)) {
Common::strlcat(buf, paralysedStr, size);
} else if (a->_enchantmentFlags & (1 << kActorBlind)) {
Common::strlcat(buf, blindStr, size);
} else if (a->_enchantmentFlags & (1 << kActorFear)) {
Common::strlcat(buf, afraidStr, size);
} else if (pa->isAggressive()) {
Common::strlcat(buf, angryStr, size);
}
if (stats.vitality >= a->_effectiveStats.vitality * 3) {
if (buf[0] != '\0') // strlen(buf) > 0
Common::strlcat(buf, commaStr, size);
Common::strlcat(buf, badlyWoundedStr, size);
} else if (stats.vitality * 2 > a->_effectiveStats.vitality * 3) {
if (buf[0] != '\0') // strlen(buf) > 0
Common::strlcat(buf, commaStr, size);
Common::strlcat(buf, hurtStr, size);
}
if (a->_enchantmentFlags & (1 << kActorPoisoned)) {
if (buf[0] != '\0') // strlen(buf) > 0
Common::strlcat(buf, commaStr, size);
Common::strlcat(buf, poisonedStr, size);
} else if (a->_enchantmentFlags & (1 << kActorDiseased)) {
if (buf[0] != '\0') // strlen(buf) > 0
Common::strlcat(buf, commaStr, size);
Common::strlcat(buf, diseasedStr, size);
}
if (buf[0] == '\0') // strlen(buf) == 0
Common::strlcat(buf, normalStr, size);
}
/* ===================================================================== *
status line gadget
* ===================================================================== */
// status line class
CStatusLine::CStatusLine(gPanelList &list,
const Rect16 &box,
const char *msg,
gFont *font,
int16 textPos,
textPallete pal,
int32 /*frameTime*/,
int16 ident,
AppFunc *cmd) :
CPlaqText(list, box, msg, font, textPos, pal, ident, cmd) {
_lineDisplayed = false;
_queueHead = _queueTail = 0;
for (int i = 0; i < ARRAYSIZE(_lineQueue); i++) {
_lineQueue[i].text = nullptr;
_lineQueue[i].frameTime = 0;
}
_waitAlarm._basetime = _waitAlarm._duration = 0;
_minWaitAlarm._basetime = _minWaitAlarm._duration = 0;
}
CStatusLine::~CStatusLine() {
while (_queueTail != _queueHead) {
assert(_lineQueue[_queueTail].text != nullptr);
delete[] _lineQueue[_queueTail].text;
_queueTail = bump(_queueTail);
}
}
void CStatusLine::setLine(char *msg, uint32 frameTime) { // frametime def
uint8 newHead = bump(_queueHead);
if (newHead != _queueTail) {
size_t msgLen = strlen(msg);
_lineQueue[_queueHead].text = new char[msgLen + 1];
Common::strcpy_s(_lineQueue[_queueHead].text, msgLen + 1, msg);
_lineQueue[_queueHead].frameTime = frameTime;
_queueHead = newHead;
}
}
void CStatusLine::experationCheck() {
if (_lineDisplayed
&& (_waitAlarm.check()
|| (_queueTail != _queueHead && _minWaitAlarm.check()))) {
enable(false);
_window.update(_extent);
_lineDisplayed = false;
}
if (!_lineDisplayed && _queueTail != _queueHead) {
// enable the control
enable(true);
// set up the time for this message
_waitAlarm.set(_lineQueue[_queueTail].frameTime);
_minWaitAlarm.set(_lineQueue[_queueTail].frameTime / 5);
// copy upto the buffer's size in chars
Common::strlcpy(_lineBuf, _lineQueue[_queueTail].text,kBufSize);
_lineBuf[kBufSize - 1] = '\0';
// free the queue text buffer
delete[] _lineQueue[_queueTail].text;
_lineQueue[_queueTail].text = nullptr;
// bump the queue tail
_queueTail = bump(_queueTail);
// draw the new textline
_window.update(_extent);
_lineDisplayed = true;
}
}
void CStatusLine::clear() {
enable(false);
_window.update(_extent);
_lineDisplayed = false;
_queueHead = _queueTail = 0;
}
/* ===================================================================== *
CMassWeightInterface: mass and weight allowence indicators
* ===================================================================== */
bool CMassWeightIndicator::_bRedraw;
CMassWeightIndicator::CMassWeightIndicator(gPanelList *panel, const Point16 &pos, uint16 type, bool death) {
// set up the position of this indicator
_backImagePos = pos;
_massPiePos = _backImagePos;
_bulkPiePos = _backImagePos;
_massPiePos.x += kMassPieXOffset;
_massPiePos.y += kMassPieYOffset;
_bulkPiePos.x += kBulkPieXOffset;
_bulkPiePos.y += kBulkPieYOffset;
_bRedraw = true; // this MUST be true or the indicators will not draw the first time
// attach the resource context
_containerRes = resFile->newContext(containerGroupID, "container context");
// setup mass/bulk indicator imagery
if (death) {
_massBulkImag = g_vm->_imageCache->requestImage(_containerRes, MKTAG('D', 'J', 'B', massBulkResNum));
_pieIndImag = loadImageRes(_containerRes, pieIndResNum, numPieIndImages, 'D', 'A', 'J');
} else {
_massBulkImag = g_vm->_imageCache->requestImage(_containerRes, MKTAG('G', 'J', 'B', massBulkResNum));
_pieIndImag = loadImageRes(_containerRes, pieIndResNum, numPieIndImages, 'G', 'A', 'J');
}
// attach controls to the indivControls panel
// these butttons will get deactivated along with the panel
_pieMass = new GfxCompImage(*panel,
Rect16(_massPiePos.x, _massPiePos.y, kPieXSize, kPieYSize),
_pieIndImag,
numPieIndImages,
0,
type,
cmdMassInd);
_pieBulk = new GfxCompImage(*panel,
Rect16(_bulkPiePos.x, _bulkPiePos.y, kPieXSize, kPieYSize),
_pieIndImag,
numPieIndImages,
0,
type,
cmdBulkInd);
// mass/bulk back image
new GfxCompImage(*panel,
Rect16(_backImagePos.x, _backImagePos.y, kBackImageXSize, kBackImageYSize),
_massBulkImag,
kUiIndiv,
nullptr);
// release resource context
if (_containerRes) {
resFile->disposeContext(_containerRes);
_containerRes = nullptr;
}
_currentMass = 0;
_currentBulk = 0;
// if this is something other then the ready containers
if (type > 1) {
_containerObject = (GameObject *)panel->_userData;
} else {
_containerObject = nullptr;
}
g_vm->_indList.push_back(this);
}
CMassWeightIndicator::~CMassWeightIndicator() {
g_vm->_indList.remove(this);
unloadImageRes(_pieIndImag, numPieIndImages);
g_vm->_imageCache->releaseImage(_massBulkImag);
}
/*****************************************************************************
** Method: recalculate()
** Description: This will recalculate the values to be displayed for the
** mass and bulk of the current ( single mode ) player
**/
void CMassWeightIndicator::recalculate() {
assert(_pieMass);
assert(_pieBulk);
uint16 mass = getMassPieDiv();
uint16 bulk = getBulkPieDiv();
uint16 retMass, retBulk;
if (_containerObject) {
setMassPie(retMass = getWeightRatio(_containerObject, mass, false));
setBulkPie(retBulk = getBulkRatio(_containerObject, bulk, false));
} else {
setMassPie(retMass = getWeightRatio(g_vm->_playerList[getCenterActorPlayerID()]->getActor(), mass, false));
setBulkPie(retBulk = getBulkRatio(g_vm->_playerList[getCenterActorPlayerID()]->getActor(), bulk, false));
}
}
/*****************************************************************************
** Method: update()
** Description: This will call recalculate and then invalidate the entire
** weight/bulk control ( so it refreshes )
**/
void CMassWeightIndicator::update() {
if (_bRedraw == true) {
for (Common::List<CMassWeightIndicator *>::iterator it = g_vm->_indList.begin(); it != g_vm->_indList.end(); ++it) {
(*it)->recalculate();
(*it)->invalidate();
}
_bRedraw = false;
}
}
/* ===================================================================== *
CManaIndicator: magic stuff indicator
* ===================================================================== */
// setup the mana color tables
static uint8 manaColorMap[CManaIndicator::kManaNumManaTypes][CManaIndicator::kResNumManaColors + 9] = {
// each row has 9 leading zero's to acount for windows colors
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x18, 0x74, 0x73, 0x72, 0x80, 0x84, 0x83, 0x82, 0x82, 0x81, 0x81, 0xF4, 0xE9 }, // Red
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x18, 0x74, 0x73, 0x72, 0x78, 0x77, 0x76, 0x75, 0x6B, 0x6A, 0x69, 0x3A, 0x39 }, // Orange
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x18, 0x64, 0x63, 0x62, 0x61, 0x60, 0x5F, 0x5E, 0x5D, 0x5C, 0x5B, 0x5A, 0x59 }, // Yellow
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x18, 0xE8, 0xE7, 0xE6, 0xE5, 0xE4, 0xE3, 0xE2, 0xE1, 0xE0, 0xDF, 0xDE, 0xC9 }, // Green
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x18, 0xA0, 0x9F, 0x9E, 0xB2, 0xB1, 0xB0, 0xAF, 0xAE, 0xAD, 0xAC, 0xAA, 0xA9 }, // Blue
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x18, 0xA0, 0x98, 0x97, 0x96, 0x95, 0x94, 0x93, 0x93, 0x92, 0x92, 0x91, 0x0C } // Violet
};
CManaIndicator::CManaIndicator(gPanelList &list) : GfxCompImage(list,
Rect16(kAreaX, kAreaY, kAreaXSize, kAreaYSize),
nullptr,
0,
cmdManaInd) {
assert(resFile);
// init the resource handle with the mana resource group
_resContext = resFile->newContext(MKTAG('M', 'A', 'N', 'A'), "mana context");
// load star images
_starImages = loadImageRes(_resContext, kResStarResNum, kResNumStars, 'S', 'T', 'A');
// load in the ring images
_ringImages = loadImageRes(_resContext, kResRingResNum, kResNumRings, 'R', 'N', 'G');
_backImage = g_vm->_imageCache->requestImage(_resContext, MKTAG('B', 'A', 'C', 'K'));
_wellImage = g_vm->_imageCache->requestImage(_resContext, MKTAG('W', 'E', 'L', 'L'));
// hmm this could be cleaner...
_starRingEndPos[0] = Point16(kManaRedEndX, kManaRedEndY);
_starRingEndPos[1] = Point16(kManaOrangeEndX, kManaOrangeEndY);
_starRingEndPos[2] = Point16(kManaYellowEndX, kManaYellowEndY);
_starRingEndPos[3] = Point16(kManaGreenEndX, kManaGreenEndY);
_starRingEndPos[4] = Point16(kManaBlueEndX, kManaBlueEndY);
_starRingEndPos[5] = Point16(kManaVioletEndX, kManaVioletEndY);
_starSizes[0] = Point16(kStar1XSize, kStar1YSize);
_starSizes[1] = Point16(kStar2XSize, kStar2YSize);
_starSizes[2] = Point16(kStar3XSize, kStar3YSize);
_starSizes[3] = Point16(kStar4XSize, kStar4YSize);
_starSizes[4] = Point16(kStar5XSize, kStar5YSize);
_starSizes[5] = Point16(kStar6XSize, kStar6YSize);
_starSizes[6] = Point16(kStar7XSize, kStar7YSize);
_ringSizes[0] = Point16(kRing1XSize, kRing1YSize);
_ringSizes[1] = Point16(kRing2XSize, kRing2YSize);
_ringSizes[2] = Point16(kRing3XSize, kRing3YSize);
_ringSizes[3] = Point16(kRing4XSize, kRing4YSize);
_ringSizes[4] = Point16(kRing5XSize, kRing5YSize);
_ringSizes[5] = Point16(kRing6XSize, kRing6YSize);
_ringSizes[6] = Point16(kRing7XSize, kRing7YSize);
// get rid of resource context
resFile->disposeContext(_resContext);
_resContext = nullptr;
// set update checks to nominal values
for (uint16 i = 0; i < kManaNumManaTypes; i++) {
_currentMana[i] = -1;
_currentBaseMana[i] = -1;
}
// init the save map
_savedMap._size = Extent16(kAreaXSize, kAreaYSize);
_savedMap._data = new uint8[_savedMap.bytes()];
}
CManaIndicator::~CManaIndicator() {
// release images
unloadImageRes(_starImages, kResNumStars);
unloadImageRes(_ringImages, kResNumRings);
// release back image
g_vm->_imageCache->releaseImage(_backImage);
g_vm->_imageCache->releaseImage(_wellImage);
// release the saved map
if (_savedMap._data)
delete[] _savedMap._data;
}
// this method provides a rect for any of the six mana regions of the control
// region numbers:
// -------
// |0 1 2|
// |3 4 5|
// -------
Rect16 CManaIndicator::getManaRegionRect(int8 nRegion) {
assert(nRegion >= 0 && nRegion < kManaNumManaRegions);
int boxSizeX = kAreaXSize / 3;
int boxSizeY = kAreaYSize / 2;
static Rect16 manaRegionRects[kManaNumManaRegions] = {
Rect16(kAreaX, kAreaY, boxSizeX, boxSizeY),
Rect16(kAreaX + boxSizeX, kAreaY, boxSizeX, boxSizeY),
Rect16(kAreaX + boxSizeX * 2, kAreaY, boxSizeX, boxSizeY),
Rect16(kAreaX, kAreaY + boxSizeY, boxSizeX, boxSizeY),
Rect16(kAreaX + boxSizeX, kAreaY + boxSizeY, boxSizeX, boxSizeY),
Rect16(kAreaX + boxSizeX * 2, kAreaY + boxSizeY, boxSizeX, boxSizeY)
};
return manaRegionRects[nRegion];
}
void CManaIndicator::draw() {
gPort &port = _window._windowPort;
// save pen color, etc.
SAVE_GPORT_STATE(port);
// setup the port
port.setMode(kDrawModeMatte);
g_vm->_pointer->hide(port, _extent); // hide mouse pointer
drawClipped(port, Point16(0, 0), Rect16(0, 0, kAreaXSize, kAreaYSize));
g_vm->_pointer->show(port, _extent); // show mouse pointer
}
void CManaIndicator::drawClipped(gPort &port,
const Point16 &offset,
const Rect16 &clipRect) {
bool calcDraw;
// Do an update to the mana star info if needed,
// if not, do not draw stuff
calcDraw = update(g_vm->_playerList[getCenterActorPlayerID()]);
if (!calcDraw) {
if (!_extent.overlap(clipRect)) return;
// draw the saved image to the port
port.setMode(kDrawModeMatte);
port.bltPixels(_savedMap, 0, 0,
_extent.x - offset.x, _extent.y - offset.y,
kAreaXSize, kAreaYSize);
// draw the frame
drawCompressedImage(port, Point16(_extent.x - offset.x, _extent.y - offset.y), _backImage);
// and finish
return;
}
// otherwise continue with the update
g_vm->_pointer->hide();
// create a temporary gPort to blit stuff to
gPort tempPort;
gPixelMap ringMap, starMap, mixMap, tempMap;
if (!NewTempPort(tempPort, kAreaXSize, kAreaYSize))
return;
// set the blit surface to a flat black
memset(tempPort._map->_data, 24, tempPort._map->bytes());
// draw the well
drawCompressedImage(tempPort, Point16(kWellX, kWellY), _wellImage);
// make a mixing plane and blank it
mixMap._size = Extent16(kAreaXSize, kAreaYSize);
mixMap._data = new uint8[mixMap.bytes()]();
// make a temp plane and blank it
tempMap._size = Extent16(kAreaXSize, kAreaYSize);
tempMap._data = new uint8[tempMap.bytes()]();
// clear out the blit surfaces
memset(mixMap._data, 0, mixMap.bytes());
memset(tempMap._data, 0, tempMap.bytes());
// draw as glyph
tempPort.setMode(kDrawModeMatte);
// draw each star and ring with color remap
for (uint16 i = 0; i < kManaNumManaTypes; i++) {
// get the header for the image pointer passed
ImageHeader *starHdr = (ImageHeader *)_starImages[_manaLines[i].starImageIndex];
ImageHeader *ringHdr = (ImageHeader *)_ringImages[_manaLines[i].ringImageIndex];
// set the buffer blit area to the image size
starMap._size = starHdr->size;
ringMap._size = ringHdr->size;
// see if it's compressed
if (starHdr->compress) {
// allocation of the temp buffer
starMap._data = new uint8[starMap.bytes()]();
// if it is then upack it to spec'ed coords.
unpackImage(&starMap, starMap._size.x, starMap._size.y, starHdr->data);
} else starMap._data = (uint8 *)starHdr->data;
// see if it's compressed
if (ringHdr->compress) {
// allocation of the temp buffer
ringMap._data = new uint8[ringMap.bytes()]();
// if it is then upack it to spec'ed coords.
unpackImage(&ringMap, ringMap._size.x, ringMap._size.y, ringHdr->data);
} else ringMap._data = (uint8 *)ringHdr->data;
// now blit the rings to the mixing surface
TBlit(&mixMap, &ringMap, _manaLines[i].ringPos.x, _manaLines[i].ringPos.y);
TBlit(&tempMap, &starMap, _manaLines[i].starPos.x, _manaLines[i].starPos.y);
// now do a peusdo-log additive thing to the images
uint8 *dst = (uint8 *)mixMap._data;
uint8 *src = (uint8 *)tempMap._data;
// get the least common dinominator for size ( should be equal )
uint16 bufferSize = MIN(mixMap.bytes(), tempMap.bytes());
for (uint16 j = 0; j < bufferSize; j++) {
// image bug fix/kludge
if (dst[j] > 21) dst[j] = 10;
if (src[j] > 21) src[j] = 10;
// if the tempMap pixel is greater then zero
if (src[j] != 0 && src[j] > dst[j]) {
dst[j] += src[j] - dst[j];
}
}
// for each color index possible, match correct color value
// at dest buffer
compositePixels(
tempPort._map,
&mixMap,
0,
0,
manaColorMap[i]);
// clear out the mixing surfaces
memset(mixMap._data, 0, mixMap.bytes());
memset(tempMap._data, 0, tempMap.bytes());
// dispose the temporary gPixelMap
if (starHdr->compress)
delete[] starMap._data;
if (ringHdr->compress)
delete[] ringMap._data;
}
// save this frame
TBlit(&_savedMap, tempPort._map, 0, 0);
// Blit the pixelmap to the main screen
port.setMode(kDrawModeMatte);
port.bltPixels(*tempPort._map, 0, 0,
_extent.x - offset.x, _extent.y - offset.y,
kAreaXSize, kAreaYSize);
// now blit the frame on top of it all.
drawCompressedImage(port, Point16(_extent.x - offset.x, _extent.y - offset.y), _backImage);
// dispose of temporary pixelmap
DisposeTempPort(tempPort);
if (mixMap._data)
delete[] mixMap._data;
if (tempMap._data)
delete[] tempMap._data;
g_vm->_pointer->show();
}
bool CManaIndicator::needUpdate(PlayerActor *player) {
assert(player);
// get the ability scores for this character
ActorAttributes *stats = player->getEffStats();
ActorAttributes baseStatsRef = player->getBaseStats();
int16 manaAmount;
int16 baseManaAmount;
uint16 i;
// this could do more array checking, but
// the number of mana type should remain stable
for (i = 0; i < kManaNumManaTypes; i++) {
manaAmount = stats->mana(i);
baseManaAmount = baseStatsRef.mana(i);
// check for new data
if (manaAmount != _currentMana[i] || baseManaAmount != _currentBaseMana[i]) {
return true;
}
}
return false;
}
bool CManaIndicator::update(PlayerActor *player) {
assert(player);
// get the ability scores for this character
ActorAttributes *stats = player->getEffStats();
ActorAttributes baseStatsRef = player->getBaseStats();
int16 manaAmount;
int16 baseManaAmount;
uint16 i;
bool newData = false;
// this could do more array checking, but
// the number of mana type should remain stable
for (i = 0; i < kManaNumManaTypes; i++) {
manaAmount = stats->mana(i);
baseManaAmount = baseStatsRef.mana(i);
// check for new data
if (manaAmount != _currentMana[i] || baseManaAmount != _currentBaseMana[i]) {
newData = true;
_currentMana[i] = manaAmount;
_currentBaseMana[i] = baseManaAmount;
}
// get manaLine info ( which star/ring image, and position on screen )
// from getStarInfo which takes the mana type index ( i ),
// current mana total, and the player base mana
if (newData == true) {
getManaLineInfo(i, manaAmount, baseManaAmount, &_manaLines[i]);
}
}
// return the state of data change
if (newData == false) {
return false;
} else {
return true;
}
}
// A generalized interpolation template
template<class T> inline T LERP(T p1, T p2, int32 steps, int32 stepNum) {
return p1 + (((p2 - p1) * stepNum) / steps);
}
void CManaIndicator::getManaLineInfo(uint16 index,
int16 manaAmount,
int16 baseManaAmount,
manaLineInfo *info) {
Point16 basePos = Point16(kAreaXSize / 2, kAreaYSize / 2);
// div zero prevention
if (manaAmount == 0) manaAmount = 1;
if (baseManaAmount == 0) baseManaAmount = 1;
manaLineInfo manaInfo;
// Calculate the positions of the mana stars, and which images to use.
manaInfo.starPos = LERP(basePos,
_starRingEndPos[index],
(int32)kManaMaxLevel,
(int32)manaAmount);
manaInfo.ringPos = LERP(basePos,
_starRingEndPos[index],
(int32)kManaMaxLevel,
(int32)baseManaAmount);
manaInfo.starImageIndex = clamp(0, manaAmount * kResNumStars / kManaMaxLevel, kResNumStars - 1);
manaInfo.ringImageIndex = clamp(0, baseManaAmount * kResNumStars / kManaMaxLevel, kResNumRings - 1);
// now do centering correct for images
manaInfo.starPos.x -= _starSizes[manaInfo.starImageIndex].x / 2;
manaInfo.starPos.y -= _starSizes[manaInfo.starImageIndex].y / 2;
manaInfo.ringPos.x -= _ringSizes[manaInfo.ringImageIndex].x / 2;
manaInfo.ringPos.y -= _ringSizes[manaInfo.ringImageIndex].y / 2;
// return the manaLineInfo struct info about mana star ring
*info = manaInfo;
}
/* ===================================================================== *
CHealthIndicator: Health star indicator
* ===================================================================== */
CHealthIndicator::CHealthIndicator(AppFunc *cmd) {
uint16 i;
// init the resource handle with the image group context
_healthRes = resFile->newContext(imageGroupID, "health imagery context");
// load in health star imagery
_starImag = loadButtonRes(_healthRes, kHealthStarStart, kHealthStarNum, 'S', 'T', 'A');
// load in the health star border
_starFrameImag = g_vm->_imageCache->requestImage(_healthRes, MKTAG('B', 'T', 'N', kHealthStarFrameResNum));
// set the image indexes to nominal startup values
for (i = 0; i < kHealthNumControls + 1; i++) {
_imageIndexMemory[i] = -1;
}
// setup the id's for each of the stars
_starIDs[0] = kUiJulian;
_starIDs[1] = kUiPhillip;
_starIDs[2] = kUiKevin;
// health controls for the trio view
// deallocated with panel
for (i = 0; i < kHealthNumControls; i++) {
_starBtns[i] = new GfxCompImage(*trioControls,
Rect16(kHealthStarXPos,
kHealthStarYPos + kHealthStarYOffset * i,
kHealthStarXSize,
kHealthStarYSize),
_starImag,
kHealthStarNum,
kHealthStarInitial,
_starIDs[i],
cmd);
// image control for the star border/frame trio mode
new GfxCompImage(*trioControls,
Rect16(kHealthFrameXPos,
kHealthFrameYPos + kHealthStarYOffset * i,
kHealthFrameXSize,
kHealthFrameYSize),
_starFrameImag,
0,
nullptr);
}
// health control for individual mode
// deallocated with panel
_indivStarBtn = new GfxCompImage(*indivControls,
Rect16(kHealthStarXPos,
kHealthStarYPos,
kHealthStarXSize,
kHealthStarYSize),
_starImag,
kHealthStarNum,
kHealthStarInitial,
kUiIndiv,
cmd);
// image control for the star border/frame indiv mode
new GfxCompImage(*indivControls,
Rect16(kHealthFrameXPos,
kHealthFrameYPos,
kHealthFrameXSize,
kHealthFrameYSize),
_starFrameImag,
0,
nullptr);
// release resource context
if (_healthRes) {
resFile->disposeContext(_healthRes);
_healthRes = nullptr;
}
}
CHealthIndicator::~CHealthIndicator() {
// release star imagery
unloadImageRes(_starImag, kHealthStarNum);
// release star frame imagery
g_vm->_imageCache->releaseImage(_starFrameImag);
}
// Recalculate and update the health star for a particular brother
void CHealthIndicator::updateStar(GfxCompImage *starCtl, int32 bro, int32 baseVitality, int32 curVitality) {
assert(baseVitality >= 0);
int16 maxStar, imageIndex;
// prevent div zero
if (baseVitality == 0) baseVitality = 1;
maxStar = clamp(0, baseVitality / 10 + 14, kHealthStarLevels - 1);
// imageIndex = (int16)( sqrt( sqrt( (double)curVitality ) ) * maxStar) / sqrt( sqrt( (double)baseVitality ) );
imageIndex = (int16)(sqrt((double)MAX((int32)0, curVitality)) * maxStar) / sqrt((double)baseVitality);
// prevent needless draws
if (_imageIndexMemory[bro] != imageIndex) {
starCtl->setCurrent(imageIndex);
starCtl->invalidate();
_imageIndexMemory[bro] = imageIndex;
}
}
void CHealthIndicator::update() {
if (g_vm->_indivControlsFlag) {
// get the stats for the selected brother
int16 baseVitality = g_vm->_playerList[translatePanID(kUiIndiv)]->getBaseStats().vitality;
int16 currVitality = g_vm->_playerList[translatePanID(kUiIndiv)]->getEffStats()->vitality;
updateStar(_indivStarBtn, kUiIndiv, baseVitality, currVitality);
} else {
for (uint16 i = 0; i < kHealthNumControls; i++) {
// get the stats for the selected brother
int16 baseVitality = g_vm->_playerList[i]->getBaseStats().vitality;
int16 currVitality = g_vm->_playerList[i]->getEffStats()->vitality;
updateStar(_starBtns[i], i, baseVitality, currVitality);
}
}
}
/* ===================================================================== *
Plaq style text writing function
* ===================================================================== */
void writePlaqText(gPort &port,
const Rect16 &r,
gFont *font,
int16 textPos,
textPallete &pal,
bool hiLite,
const char *msg, ...) {
char lineBuf[128];
va_list argptr;
Rect16 workRect;
int16 cnt;
gFont *_oldFont = port._font;
va_start(argptr, msg);
cnt = Common::vsprintf_s(lineBuf, msg, argptr);
va_end(argptr);
SAVE_GPORT_STATE(port);
port.setMode(kDrawModeMatte);
port.setFont(font);
workRect = r;
port.setColor(hiLite ? pal.dlHilitePen : pal.dlPen);
workRect.x--;
port.drawTextInBox(lineBuf, cnt, workRect, textPos, Point16(0, 0));
workRect.y++;
port.drawTextInBox(lineBuf, cnt, workRect, textPos, Point16(0, 0));
workRect.x++;
port.drawTextInBox(lineBuf, cnt, workRect, textPos, Point16(0, 0));
port.setColor(hiLite ? pal.urHilitePen : pal.urPen);
workRect.x++;
workRect.y--;
port.drawTextInBox(lineBuf, cnt, workRect, textPos, Point16(0, 0));
workRect.y--;
port.drawTextInBox(lineBuf, cnt, workRect, textPos, Point16(0, 0));
workRect.x--;
port.drawTextInBox(lineBuf, cnt, workRect, textPos, Point16(0, 0));
port.setColor(hiLite ? pal.inHilitePen : pal.inPen);
workRect.y++;
port.drawTextInBox(lineBuf, cnt, workRect, textPos, Point16(0, 0));
port.setFont(_oldFont);
}
void writePlaqTextPos(gPort &port,
const Point16 &pos,
gFont *font,
int16 /*textPos*/,
textPallete &pal,
bool hiLite,
const char *msg, ...) {
char lineBuf[128];
va_list argptr;
Point16 drawPos;
gFont *_oldFont = port._font;
va_start(argptr, msg);
Common::vsprintf_s(lineBuf, msg, argptr);
va_end(argptr);
SAVE_GPORT_STATE(port);
port.setMode(kDrawModeMatte);
port.setFont(font);
drawPos = pos;
port.setColor(hiLite ? pal.dlHilitePen : pal.dlPen);
drawPos.x--;
port.moveTo(drawPos);
port.drawText(lineBuf, -1);
drawPos.y++;
port.moveTo(drawPos);
port.drawText(lineBuf, -1);
drawPos.x++;
port.moveTo(drawPos);
port.drawText(lineBuf, -1);
port.setColor(hiLite ? pal.urHilitePen : pal.urPen);
drawPos.x++;
drawPos.y--;
port.moveTo(drawPos);
port.drawText(lineBuf, -1);
drawPos.y--;
port.moveTo(drawPos);
port.drawText(lineBuf, -1);
drawPos.x--;
port.moveTo(drawPos);
port.drawText(lineBuf, -1);
port.setColor(hiLite ? pal.inHilitePen : pal.inPen);
drawPos.y++;
port.moveTo(drawPos);
port.drawText(lineBuf, -1);
port.setFont(_oldFont);
}
///////////////////////////////////////////////////////////////////
/* functions for loading and unloading sets of compressed images */
// creates an array of button images
// passed a resource context, resource ref ID and the number of images
// to sequentially load in
void **loadButtonRes(hResContext *con, int16 resID, int16 numRes) {
int16 i, k;
void **images = (void **)malloc(sizeof(void *)*numRes);
for (i = 0, k = resID; i < numRes; i++, k++) {
// get an image from the image cache
images[i] = g_vm->_imageCache->requestImage(con, MKTAG('B', 'T', 'N', k));
}
return images;
}
// creates an array of images
// passed a resource context, resource ref ID, the number of images
// to sequentially load in, and the context id's
void **loadButtonRes(hResContext *con, int16 resID, int16 numRes, char a, char b, char c) {
int16 i, k;
void **images = (void **)malloc(sizeof(void *)*numRes);
for (i = 0, k = resID; i < numRes; i++, k++) {
images[i] = g_vm->_imageCache->requestImage(con, MKTAG(a, b, c, k));
}
return images;
}
// see LoadButtonRes
void **loadImageRes(hResContext *con, int16 resID, int16 numRes, char a, char b, char c) {
return loadButtonRes(con, resID, numRes, a, b, c);
}
// correctly deletes any image arrays allocated with the LoadButtonRes or
// LoadImageRes sets
void unloadImageRes(void **images, int16 numRes) {
int16 i;
if (images) {
for (i = 0; i < numRes; i++) {
g_vm->_imageCache->releaseImage(images[i]);
}
free(images);
}
}
// defined for setup off all button based user controls
void SetupUserControls() {
if (g_vm->getGameId() == GID_DINO) {
warning("TODO: SetupUserControls() for Dino");
return;
}
// index variables
uint16 n;
uint8 index = 0;
// resource handle
hResContext *imageRes;
// brother panel id's
uint16 brotherIDs[kNumViews] = { kUiJulian, kUiPhillip, kUiKevin };
// portrait resource indexes
int16 portResNum[] = { 0, 0, 0 };
// image setup
// init the resource handle with the image group context
imageRes = resFile->newContext(imageGroupID, "image context");
// set up the control button images
aggressImag = loadButtonRes(imageRes, aggressResNum, numBtnImages);
centerImag = loadButtonRes(imageRes, centerResNum, numBtnImages);
bandingImag = loadButtonRes(imageRes, bandingResNum, numBtnImages);
menConBtnImag = loadButtonRes(imageRes, menConBtnResNum, numBtnImages);
// setup the options button imagery
optBtnImag = loadButtonRes(imageRes, optBtnResNum, numBtnImages);
// setup the brother selector button imagery and button frame
julBtnImag = loadButtonRes(imageRes, julBtnResNum, numBtnImages);
phiBtnImag = loadButtonRes(imageRes, phiBtnResNum, numBtnImages);
kevBtnImag = loadButtonRes(imageRes, kevBtnResNum, numBtnImages);
broBtnFrameImag = g_vm->_imageCache->requestImage(imageRes, MKTAG('F', 'R', 'A', 'M'));
// set up the portrait name plates
for (n = 0; n < kNumViews; n++) {
namePlateImages[n] = g_vm->_imageCache->requestImage(imageRes, MKTAG('B', 'T', 'N', namePlateResNum[n]));
}
// get the frame image
namePlateFrameImag = g_vm->_imageCache->requestImage(imageRes, MKTAG('B', 'T', 'N', 15));
armorImag = g_vm->_imageCache->requestImage(imageRes, MKTAG('B', 'T', 'N', 34));
// clean out the old context
if (imageRes) resFile->disposeContext(imageRes);
imageRes = nullptr;
// init the resource handle with the face group context
imageRes = resFile->newContext(faceGroupID, "face resources");
// set up the portrait button images
for (n = 0; n < kNumViews; n++) {
portImag[n] = loadButtonRes(imageRes, portResNum[n], numPortImages, broNames[n].a, broNames[n].b, broNames[n].c);
}
// setup stand alone controls
optBtn = new GfxCompButton(*playControls, optBtnRect, optBtnImag, numBtnImages, 0, cmdOptions);
enchDisp = new gEnchantmentDisplay(*playControls, 0);
// setup the trio user cntl _buttons
for (n = 0; n < kNumViews; n++) {
// portrait button
portBtns[n] = new GfxMultCompButton(*trioControls, views[n][index++],
portImag[n], numPortImages, 0, false, brotherIDs[n], cmdPortrait);
portBtns[n]->setMousePoll(true);
// aggressive button
aggressBtns[n] = new GfxOwnerSelCompButton(*trioControls, views[n][index++],
aggressImag, numBtnImages, brotherIDs[n], cmdAggressive);
// name plates that go under the portraits
armorInd[n] = new gArmorIndicator(*trioControls, views[n][index++],
armorImag, brotherIDs[n], cmdArmor);
// center on brother
centerBtns[n] = new GfxOwnerSelCompButton(*trioControls, views[n][index++],
centerImag, numBtnImages, brotherIDs[n], cmdCenter);
// banding
bandingBtns[n] = new GfxOwnerSelCompButton(*trioControls, views[n][index++],
bandingImag, numBtnImages, brotherIDs[n], cmdBand);
// name plates that go under the portraits
namePlates[n] = new GfxCompImage(*trioControls, views[n][index++],
namePlateImages[n], 0, nullptr);
// the frames for the name plates
namePlateFrames[n] = new GfxCompImage(*trioControls, views[n][index++],
namePlateFrameImag, 0, nullptr);
index = 0;
}
// individual control _buttons
// portrait button
indivPortBtn = new GfxMultCompButton(*indivControls, views[0][index++],
portImag[0], numPortImages, 0, false, kUiIndiv, cmdPortrait);
indivPortBtn->setMousePoll(true);
// aggressive button
indivAggressBtn = new GfxOwnerSelCompButton(*indivControls, views[0][index++],
aggressImag, numBtnImages, kUiIndiv, cmdAggressive);
indivArmorInd = new gArmorIndicator(*indivControls, views[0][index++],
armorImag, kUiIndiv, cmdArmor);
// center on brother
indivCenterBtn = new GfxOwnerSelCompButton(*indivControls, views[0][index++],
centerImag, numBtnImages, kUiIndiv, cmdCenter);
// banding
indivBandingBtn = new GfxOwnerSelCompButton(*indivControls, views[0][index++],
bandingImag, numBtnImages, kUiIndiv, cmdBand);
// name plates that go under the portraits
indivNamePlate = new GfxCompImage(*indivControls, views[0][index++],
namePlateImages[0], 0, nullptr);
// the frames for the name plates
indivNamePlateFrame = new GfxCompImage(*indivControls, views[0][index++],
namePlateFrameImag, 0, nullptr);
// setup the portrait object
Portrait = new CPortrait(portBtns, // portrait _buttons
indivPortBtn,
numPortImages,// num of images per button
kNumViews); // number of brothers
// mental container button
menConBtn = new GfxCompButton(*indivControls, menConBtnRect, menConBtnImag, numBtnImages, kUiIndiv, cmdBrain);
// brother selection _buttons >>> need to replace these with sticky _buttons
julBtn = new GfxOwnerSelCompButton(*indivControls, julBtnRect, julBtnImag, numBtnImages, kUiJulian, cmdBroChange);
phiBtn = new GfxOwnerSelCompButton(*indivControls, phiBtnRect, phiBtnImag, numBtnImages, kUiPhillip, cmdBroChange);
kevBtn = new GfxOwnerSelCompButton(*indivControls, kevBtnRect, kevBtnImag, numBtnImages, kUiKevin, cmdBroChange);
// frame for brother _buttons
broBtnFrame = new GfxCompImage(*indivControls, broBtnRect, broBtnFrameImag, kUiIndiv, nullptr);
// make the mana indicator
ManaIndicator = new CManaIndicator(*indivControls);
ManaIndicator->setMousePoll(true);
// get rid of the resource contexts
if (imageRes) {
resFile->disposeContext(imageRes);
imageRes = nullptr;
}
//The controls need to be enabled but undrawn at startup
//if ( displayEnabled() )
// g_vm->_userControlsSetup = true;
updateAllUserControls();
}
void enableUserControls() {
g_vm->_userControlsSetup = true;
}
void disableUserControls() {
g_vm->_userControlsSetup = false;
}
// defines the cleanup for ALL user interface controls
void CleanupUserControls() {
g_vm->_userControlsSetup = false;
CleanupButtonImages();
}
// defines the cleaup for the global button image array
void CleanupButtonImages() {
int16 i;
// deallocates the images in the array and the arrays themselves
unloadImageRes(aggressImag, numBtnImages);
// unloadImageRes( jumpImag , numBtnImages );
unloadImageRes(centerImag, numBtnImages);
unloadImageRes(bandingImag, numBtnImages);
unloadImageRes(menConBtnImag, numBtnImages);
// dealloc the imag for the option button
unloadImageRes(optBtnImag, numBtnImages);
// dealloc brother's indiv mode _buttons
unloadImageRes(julBtnImag, numBtnImages);
unloadImageRes(phiBtnImag, numBtnImages);
unloadImageRes(kevBtnImag, numBtnImages);
// portraits
for (i = 0; i < kNumViews; i++) {
unloadImageRes(portImag[i], numPortImages);
}
// name plate frame
g_vm->_imageCache->releaseImage(namePlateFrameImag);
g_vm->_imageCache->releaseImage(armorImag);
// release name frames
g_vm->_imageCache->releaseImage(broBtnFrameImag);
// name plates
for (i = 0; i < kNumViews; i++) {
g_vm->_imageCache->releaseImage(namePlateImages[i]);
}
}
void updateIndicators() {
HealthIndicator->update();
MassWeightIndicator->update();
// mana indicator update check
if (isIndivMode()) {
if (ManaIndicator->needUpdate(g_vm->_playerList[getCenterActorPlayerID()])) {
// redraw the region that is not covered by any other window
ManaIndicator->invalidate();
}
}
}
//>>>
/* ===================================================================== *
These should become methods of the playerActor class
* ===================================================================== */
template< class T >
inline T GetRatio(T curUnits, T maxUnits, T ratio) {
uint16 delt;
if (maxUnits < ratio) {
delt = ratio / maxUnits;
curUnits *= delt;
} else {
delt = maxUnits / ratio;
curUnits /= delt;
}
return clamp(0, curUnits, ratio);
}
uint16 getWeightRatio(GameObject *obj, uint16 &maxRatio, bool bReturnMaxRatio = true) {
assert(isObject(obj) || isActor(obj));
uint16 weight;
uint16 maxWeight;
// get the mass capacity for this container
maxWeight = obj->massCapacity();;
// get the total mass this container is holding
weight = obj->totalContainedMass();
if (bReturnMaxRatio) {
maxRatio = maxWeight;
return weight;
} else {
return maxWeight != unlimitedCapacity
? GetRatio(weight, maxWeight, maxRatio)
: 0;
}
}
uint16 getBulkRatio(GameObject *obj, uint16 &maxRatio, bool bReturnMaxRatio = true) {
assert(isObject(obj) || isActor(obj));
uint16 maxBulk;
uint16 bulk;
// get the bulk capacity for this container
maxBulk = obj->bulkCapacity();
// get the total bulk this container is holding
bulk = obj->totalContainedBulk();
if (bReturnMaxRatio) {
maxRatio = maxBulk;
return bulk;
} else {
return maxBulk != unlimitedCapacity
? GetRatio(bulk, maxBulk, maxRatio)
: 0;
}
}
/* ===================================================================== *
Application functions
* ===================================================================== */
void updateReadyContainers() {
// if in individual mode
if (g_vm->_indivControlsFlag) {
indivCviewTop->invalidate();
indivCviewBot->invalidate();
} else if (TrioCviews[getCenterActorPlayerID()]) {
TrioCviews[getCenterActorPlayerID()]->invalidate();
}
}
void setEnchantmentDisplay() {
if (enchDisp) enchDisp->setValue(getCenterActorPlayerID());
}
// sets the individual brother control state _buttons
void setIndivBtns(uint16 brotherID) { // top = 0, mid = 1, bot = 2
g_vm->_indivControlsFlag = true;
// set the indiv bro
indivBrother = brotherID;
// set all the individual brother _buttons to the correct states
indivCenterBtn->select(centerBtns[brotherID]->isSelected());
indivCenterBtn->ghost(centerBtns[brotherID]->isGhosted());
//_indivStarBtn->setCurrent( ( uint16 )_starBtns[brotherID]->getCurrent() );
indivPortBtn->setImages(portImag[brotherID]);
indivNamePlate->setImage(namePlateImages[brotherID]);
Portrait->set(kUiIndiv, Portrait->getCurrentState(brotherID));
indivBandingBtn->select(bandingBtns[brotherID]->isSelected());
indivBandingBtn->ghost(bandingBtns[brotherID]->isGhosted());
indivAggressBtn->select(aggressBtns[brotherID]->isSelected());
indivAggressBtn->ghost(aggressBtns[brotherID]->isGhosted());
indivArmorInd->setValue(brotherID);
indivArmorInd->ghost(armorInd[brotherID]->isGhosted());
setEnchantmentDisplay();
// point the read containers to the correct brother
if (brotherID >= kPlayerActors)
brotherID = kPlayerActors - 1;
indivCviewTop->setContainer(GameObject::objectAddress(ActorBaseID + brotherID));
indivCviewTop->ghost(TrioCviews[brotherID]->isGhosted());
indivCviewBot->setContainer(GameObject::objectAddress(ActorBaseID + brotherID));
indivCviewBot->ghost(TrioCviews[brotherID]->isGhosted());
// now set the indicators for mass and bulk
uint16 pieWeightRatio = MassWeightIndicator->getMassPieDiv();
uint16 _pieBulkRatio = MassWeightIndicator->getBulkPieDiv();
PlayerActor *brother = g_vm->_playerList[brotherID];
MassWeightIndicator->setMassPie(getWeightRatio(brother->getActor(), pieWeightRatio, false));
MassWeightIndicator->setBulkPie(getBulkRatio(brother->getActor(), _pieBulkRatio, false));
}
// sets the trio brothers control state _buttons
void setTrioBtns() {
g_vm->_indivControlsFlag = false;
// reset any value that might have changed in idividual mode
centerBtns[indivBrother]->select(indivCenterBtn->isSelected());
bandingBtns[indivBrother]->select(indivBandingBtn->isSelected());
Portrait->set(indivBrother, Portrait->getCurrentState(kUiIndiv));
aggressBtns[indivBrother]->select(indivAggressBtn->isSelected());
armorInd[indivBrother]->setValue(indivBrother);
setEnchantmentDisplay();
}
void setControlPanelsToIndividualMode(uint16 brotherID) {
// copy the button/indicator states to the indiv _buttons
setIndivBtns(brotherID);
// set the mode controls
trioControls->show(false, false);
indivControls->show(true, true);
trioControls->show(false, true);
}
void setControlPanelsToTrioMode() {
setTrioBtns();
indivControls->show(false, false);
trioControls->show(true, true);
indivControls->show(false, true);
}
void toggleIndivMode() {
if (g_vm->_indivControlsFlag) setControlPanelsToTrioMode();
else setControlPanelsToIndividualMode(getCenterActorPlayerID());
}
void setCenterBrother(uint16 whichBrother) {
// If we picked up anything, then put it back.
g_vm->_mouseInfo->replaceObject();
// set the new center actor
setCenterActor(g_vm->_playerList[whichBrother]);
}
uint16 translatePanID(uint16 panID) {
// individual mode brother id translation
if (panID == kUiIndiv) {
panID = indivBrother;
}
return panID;
}
void updateBrotherPortrait(uint16 brotherID, int16 pType) {
if (g_vm->_userControlsSetup) {
Portrait->set(brotherID, (PortraitType)pType);
if (brotherID == indivBrother)
Portrait->set(kUiIndiv, (PortraitType)pType);
}
}
void updateBrotherAggressionButton(uint16 brotherID, bool aggressive) {
if (g_vm->_userControlsSetup) {
aggressBtns[brotherID]->select(aggressive);
aggressBtns[brotherID]->ghost(isBrotherDead(brotherID));
if (brotherID == indivBrother) {
indivAggressBtn->select(aggressive);
indivAggressBtn->ghost(isBrotherDead(brotherID));
}
// possibly change portrait type
recalcPortraitType(brotherID);
}
}
void updateBrotherBandingButton(uint16 brotherID, bool banded) {
if (g_vm->_userControlsSetup) {
bandingBtns[brotherID]->select(banded);
bandingBtns[brotherID]->ghost(isBrotherDead(brotherID));
if (brotherID == indivBrother) {
indivBandingBtn->select(banded);
indivBandingBtn->ghost(isBrotherDead(brotherID));
}
}
}
void updateBrotherRadioButtons(uint16 brotherID) {
if (g_vm->_userControlsSetup) {
bool jul = (kUiJulian == brotherID);
bool phi = (kUiPhillip == brotherID);
bool kev = (kUiKevin == brotherID);
// set the selection _buttons to the correct states
julBtn->select(jul);
phiBtn->select(phi);
kevBtn->select(kev);
julBtn->ghost(isBrotherDead(kUiJulian));
phiBtn->ghost(isBrotherDead(kUiPhillip));
kevBtn->ghost(isBrotherDead(kUiKevin));
// set the center brother _buttons
centerBtns[kUiJulian]->select(jul);
centerBtns[kUiPhillip]->select(phi);
centerBtns[kUiKevin]->select(kev);
centerBtns[kUiJulian]->ghost(isBrotherDead(kUiJulian));
centerBtns[kUiPhillip]->ghost(isBrotherDead(kUiPhillip));
centerBtns[kUiKevin]->ghost(isBrotherDead(kUiKevin));
if (brotherID == indivBrother) {
indivCenterBtn->select(true);
indivCenterBtn->ghost(isBrotherDead(brotherID));
}
if (g_vm->_indivControlsFlag)
setControlPanelsToIndividualMode(brotherID);
}
}
void updateBrotherArmor(uint16 brotherID) {
if (g_vm->_userControlsSetup) {
armorInd[brotherID]->setValue(brotherID);
armorInd[brotherID]->ghost(isBrotherDead(brotherID));
if (brotherID == indivBrother) {
indivArmorInd->setValue(brotherID);
indivArmorInd->ghost(isBrotherDead(brotherID));
}
}
}
void updateAllUserControls() {
if (displayEnabled()) {
if (g_vm->_userControlsSetup && g_vm->getGameId() == GID_FTA2) {
uint16 centerBrotherID = getCenterActorPlayerID(),
brotherID;
if (g_vm->_indivControlsFlag)
setControlPanelsToIndividualMode(indivBrother);
else
setControlPanelsToTrioMode();
updateBrotherRadioButtons(centerBrotherID);
for (brotherID = 0; brotherID < kNumViews; brotherID++) {
bool dead = isBrotherDead(brotherID);
updateBrotherBandingButton(brotherID, isBanded(brotherID));
updateBrotherAggressionButton(brotherID, isAggressive(brotherID));
updateBrotherPortrait(brotherID, getPortraitType(brotherID));
updateBrotherArmor(brotherID);
// if in individual mode, ghost containers if he's dead
if (brotherID == indivBrother) {
indivCviewTop->ghost(dead);
indivCviewBot->ghost(dead);
}
// Ghost the ready container if he's dead.
TrioCviews[brotherID]->ghost(dead);
}
}
} else {
reDrawScreen();
}
}
void updateBrotherControls(PlayerActorID brotherID) {
if (g_vm->_userControlsSetup) {
bool dead = isBrotherDead(brotherID);
updateBrotherRadioButtons(getCenterActorPlayerID());
updateBrotherBandingButton(brotherID, isBanded(brotherID));
updateBrotherAggressionButton(brotherID, isAggressive(brotherID));
updateBrotherPortrait(brotherID, getPortraitType(brotherID));
updateBrotherArmor(brotherID);
// if in individual mode, ghost containers if he's dead
if (brotherID == indivBrother) {
indivCviewTop->ghost(dead);
indivCviewBot->ghost(dead);
}
// Ghost the ready container if he's dead.
TrioCviews[brotherID]->ghost(dead);
}
}
// button call backs
APPFUNC(cmdPortrait) {
const int bufSize = 80;
const int stateBufSize = 60;
uint16 panID = ev.panel->_id;
GameObject *mouseObject = g_vm->_mouseInfo->getObject(); // object being dragged
switch (ev.eventType) {
case gEventNewValue:
if (mouseObject != nullptr) {
PlayerActor *pa = getPlayerActorAddress(translatePanID(panID));
Actor *centerActorPtr = getCenterActor();
// we dropped the object onto another object
if (g_vm->_mouseInfo->getDoable()) {
int16 intent = g_vm->_mouseInfo->getIntent();
g_vm->_mouseInfo->replaceObject();
if (intent == GrabInfo::kIntUse) {
// If we are using an intangible object (spell) then consider
// the owner of the spell to be the center actor for the rest
// of this action.
if (mouseObject->proto()->containmentSet() & ProtoObj::isIntangible) {
ObjectID possessor = mouseObject->possessor();
if (possessor != Nothing) {
centerActorPtr = (Actor *)GameObject::objectAddress(possessor);
}
}
MotionTask::useObjectOnObject(
*centerActorPtr,
*mouseObject,
*pa->getActor());
} else if (intent == GrabInfo::kIntDrop) {
MotionTask::dropObjectOnObject(
*centerActorPtr,
*mouseObject,
*pa->getActor(),
g_vm->_mouseInfo->getMoveCount());
}
// ( ( gGenericControl * )ev.panel )->disableDblClick();
// clickActionDone = true;
} else if (g_vm->_mouseInfo->getIntent() == GrabInfo::kIntUse) {
g_vm->_mouseInfo->replaceObject();
// clickActionDone = true;
}
} else if (panID != kUiIndiv) {
if (!isBrotherDead(panID)) {
setCenterBrother(panID);
setControlPanelsToIndividualMode(panID);
}
} else if (panID == kUiIndiv) {
setControlPanelsToTrioMode();
}
break;
case gEventMouseMove:
if (ev.value == GfxCompImage::kLeave) {
g_vm->_mouseInfo->setText(nullptr);
g_vm->_mouseInfo->setDoable(true);
break;
}
// if (ev.value == gCompImage::enter)
{
if (mouseObject != nullptr) {
PlayerActor *pa = getPlayerActorAddress(translatePanID(panID));
Actor *targetActor = pa->getActor(),
*enactor = getCenterActor();
g_vm->_mouseInfo->setText(nullptr);
if ((enactor->getLocation() - targetActor->getLocation()).quickHDistance() > 96) {
g_vm->_mouseInfo->setDoable(false);
} else {
g_vm->_mouseInfo->setDoable(true);
}
} else {
// working buffer
char buf[bufSize];
char state[stateBufSize];
uint16 brotherID = translatePanID(panID);
Portrait->getStateString(state, stateBufSize, brotherID);
switch (brotherID) {
case kUiJulian:
Common::sprintf_s(buf, "%s %s", JULIAN_BROSTATE, state);
break;
case kUiPhillip:
Common::sprintf_s(buf, "%s %s", PHILLIP_BROSTATE, state);
break;
case kUiKevin:
Common::sprintf_s(buf, "%s %s", KEVIN_BROSTATE, state);
break;
}
// set the text in the cursor
g_vm->_mouseInfo->setText(buf);
}
}
break;
default:
break;
}
}
void toggleAgression(PlayerActorID bro, bool all) {
int16 wasAggressive = isAggressive(bro);
if (all) {
for (int i = 0; i < kPlayerActors; i++)
setAggression(i, !wasAggressive);
} else setAggression(bro, !wasAggressive);
}
APPFUNC(cmdAggressive) {
uint16 transBroID = translatePanID(ev.panel->_id);
// check for message update stuff
// and aggression update
if (ev.eventType == gEventNewValue) {
toggleAgression(transBroID, rightButtonState());
// int16 wasAggressive = isAggressive( transBroID );
// if (rightButtonState())
// {
// for (int i = 0; i < kPlayerActors; i++)
// setAggression( i, !wasAggressive );
// }
// else setAggression( transBroID, !wasAggressive );
} else if (ev.eventType == gEventMouseMove) {
if (ev.value == GfxCompImage::kEnter) {
// set the text in the cursor
g_vm->_mouseInfo->setText(isAggressive(transBroID)
? ON_AGRESS
: OFF_AGRESS);
} else if (ev.value == GfxCompImage::kLeave) {
g_vm->_mouseInfo->setText(nullptr);
}
}
}
/*
APPFUNC( cmdJump )
{
if( ev.eventType == gEventNewValue )
{
jump = !jump;
if( jump )
{
setMouseImage( CloseBx1Image, 0, 0 );
}
else
{
setMouseImage( ArrowImage, 0, 0 );
}
}
}
*/
APPFUNC(cmdArmor) {
if (ev.eventType == gEventMouseMove) {
if (ev.value == GfxCompImage::kEnter) {
gArmorIndicator *gai = (gArmorIndicator *)ev.panel;
char buf[128];
if (gai->_attr.damageAbsorbtion == 0
&& gai->_attr.defenseBonus == 0) {
g_vm->_mouseInfo->setText(NO_ARMOR);
} else {
Common::sprintf_s(buf,
DESC_ARMOR,
gai->_attr.damageAbsorbtion,
gai->_attr.damageDivider,
gai->_attr.defenseBonus);
// set the text in the cursor
g_vm->_mouseInfo->setText(buf);
}
} else if (ev.value == GfxCompImage::kLeave) {
g_vm->_mouseInfo->setText(nullptr);
}
}
}
APPFUNC(cmdCenter) {
uint16 transBroID = translatePanID(ev.panel->_id);
if (ev.eventType == gEventNewValue) {
if (rightButtonState())
setCenterBrother((transBroID + 1) % kPlayerActors);
else setCenterBrother(transBroID);
}
if (ev.eventType == gEventMouseMove) {
if (ev.value == GfxCompImage::kEnter) {
// set the text in the cursor
g_vm->_mouseInfo->setText(getCenterActorPlayerID() == transBroID
? ON_CENTER
: OFF_CENTER);
} else if (ev.value == GfxCompImage::kLeave) {
g_vm->_mouseInfo->setText(nullptr);
}
}
}
void toggleBanding(PlayerActorID bro, bool all) {
int16 wasBanded = isBanded(bro);
if (all) {
for (int i = 0; i < kPlayerActors; i++)
setBanded(i, !wasBanded);
} else setBanded(bro, !wasBanded);
}
APPFUNC(cmdBand) {
uint16 transBroID = translatePanID(ev.panel->_id);
if (ev.eventType == gEventNewValue) {
toggleBanding(transBroID, rightButtonState());
// int16 wasBanded = isBanded( transBroID );
//
// if (rightButtonState())
// {
// for (int i = 0; i < kPlayerActors; i++)
// setBanded( i, !wasBanded );
// }
// else setBanded( transBroID, !wasBanded );
} else if (ev.eventType == gEventMouseMove) {
if (ev.value == GfxCompImage::kEnter) {
// set the text in the cursor
g_vm->_mouseInfo->setText(isBanded(transBroID)
? ON_BANDED
: OFF_BANDED);
} else if (ev.value == GfxCompImage::kLeave) {
g_vm->_mouseInfo->setText(nullptr);
}
}
}
APPFUNC(cmdOptions) {
if (ev.eventType == gEventNewValue) {
OptionsDialog();
//openOptionsPanel();
} else if (ev.eventType == gEventMouseMove) {
if (ev.value == GfxCompImage::kEnter) g_vm->_mouseInfo->setText(OPTIONS_PANEL);
else if (ev.value == GfxCompImage::kLeave) g_vm->_mouseInfo->setText(nullptr);
}
}
APPFUNC(cmdBroChange) {
if (ev.eventType == gEventNewValue) {
if (!isBrotherDead(ev.panel->_id)) {
setCenterBrother(ev.panel->_id);
// this sets up the _buttons in trio mode to the correct
// state ( must be called before indiv mode switchtes )
setTrioBtns();
setControlPanelsToIndividualMode(ev.panel->_id);
}
} else if (ev.eventType == gEventMouseMove) {
const int bufSize = 80;
const int stateBufSize = 60;
uint16 panID = ev.panel->_id;
if (ev.value == GfxCompImage::kEnter) {
// working buffer
char buf[bufSize];
char state[stateBufSize];
uint16 brotherID = translatePanID(panID);
Portrait->getStateString(state, stateBufSize, brotherID);
switch (brotherID) {
case kUiJulian:
Common::sprintf_s(buf, "%s %s", JULIAN_BROSTATE, state);
break;
case kUiPhillip:
Common::sprintf_s(buf, "%s %s", PHILLIP_BROSTATE, state);
break;
case kUiKevin:
Common::sprintf_s(buf, "%s %s", KEVIN_BROSTATE, state);
break;
}
// set the text in the cursor
g_vm->_mouseInfo->setText(buf);
} else if (ev.value == GfxCompImage::kLeave) {
g_vm->_mouseInfo->setText(nullptr);
}
}
}
APPFUNC(cmdHealthStar) {
uint16 transBroID = translatePanID(ev.panel->_id);
if (ev.eventType == gEventMouseMove) {
if (ev.value == GfxCompImage::kLeave) {
g_vm->_mouseInfo->setText(nullptr);
return;
}
if (ev.value == GfxCompImage::kEnter) {
ev.panel->setMousePoll(true);
}
// get the stats for the selected brother
int16 baseVitality = g_vm->_playerList[transBroID]->getBaseStats().vitality;
int16 currVitality = g_vm->_playerList[transBroID]->getEffStats()->vitality;
char buf[40];
Common::sprintf_s(buf, "%s %d/%d", HEALTH_HINT, currVitality, baseVitality);
g_vm->_mouseInfo->setText(buf);
}
}
APPFUNC(cmdMassInd) {
gWindow *win = nullptr;
GameObject *_containerObject = nullptr;
if (ev.eventType == gEventMouseMove) {
if (ev.value == GfxCompImage::kEnter) {
const int bufSize = 40;
int curWeight;
uint16 baseWeight;
char buf[bufSize];
win = ev.panel->getWindow(); // get the window pointer
assert(win);
// is it something other than the brother's indicators?
if (ev.panel->_id > 1) {
_containerObject = (GameObject *)win->_userData;
} else {
_containerObject = (GameObject *)g_vm->_playerList[getCenterActorPlayerID()]->getActor();
}
assert(_containerObject);
curWeight = getWeightRatio(_containerObject, baseWeight);
if (baseWeight != unlimitedCapacity) {
Common::sprintf_s(buf, "%s %d/%d", WEIGHT_HINT, curWeight, baseWeight);
g_vm->_mouseInfo->setText(buf);
} else
g_vm->_mouseInfo->setText(UNK_WEIGHT_HINT);
} else if (ev.value == GfxCompImage::kLeave) {
g_vm->_mouseInfo->setText(nullptr);
}
}
}
APPFUNC(cmdBulkInd) {
gWindow *win = nullptr;
GameObject *_containerObject = nullptr;
if (ev.eventType == gEventMouseMove) {
if (ev.value == GfxCompImage::kEnter) {
const int bufSize = 40;
uint16 baseBulk = 100;
char buf[bufSize];
int curBulk;
win = ev.panel->getWindow(); // get the window pointer
assert(win);
// is it something other than the brother's indicators?
if (ev.panel->_id > 1) {
_containerObject = (GameObject *)win->_userData;
} else {
_containerObject = (GameObject *)g_vm->_playerList[getCenterActorPlayerID()]->getActor();
}
assert(_containerObject);
curBulk = getBulkRatio(_containerObject, baseBulk);
if (baseBulk != unlimitedCapacity) {
Common::sprintf_s(buf, "%s %d/%d", BULK_HINT, curBulk, baseBulk);
g_vm->_mouseInfo->setText(buf);
} else
g_vm->_mouseInfo->setText(UNK_BULK_HINT);
} else if (ev.value == GfxCompImage::kLeave) {
g_vm->_mouseInfo->setText(nullptr);
}
}
}
APPFUNC(cmdManaInd) {
if (ev.eventType == gEventMouseMove) {
if (ev.value != GfxCompImage::kLeave) {
const int BUF_SIZE = 64;
char textBuffer[BUF_SIZE];
int manaType = -1;
int numManaRegions = ManaIndicator->getNumManaRegions();
int i;
int curMana = 0, baseMana = 0;
PlayerActor *player = g_vm->_playerList[getCenterActorPlayerID()];
ActorAttributes *stats = player->getEffStats();
ActorAttributes baseStatsRef = player->getBaseStats();
Point16 pos = ev.mouse;
pos.x += ManaIndicator->getExtent().x;
pos.y += ManaIndicator->getExtent().y;
for (i = 0; i < numManaRegions; i++) {
Rect16 regionRect = ManaIndicator->getManaRegionRect(i);
if (regionRect.ptInside(pos)) {
manaType = i;
break;
}
}
if (manaType != -1) {
curMana = stats->mana(clamp(0, manaType, numManaRegions));
baseMana = baseStatsRef.mana(clamp(0, manaType, numManaRegions));
}
switch (manaType) {
case 0:
Common::sprintf_s(textBuffer, "%s %d/%d", "Red Mana:", curMana, baseMana);
break;
case 1:
Common::sprintf_s(textBuffer, "%s %d/%d", "Orange Mana:", curMana, baseMana);
break;
case 2:
Common::sprintf_s(textBuffer, "%s %d/%d", "Yellow Mana:", curMana, baseMana);
break;
case 3:
Common::sprintf_s(textBuffer, "%s %d/%d", "Green Mana:", curMana, baseMana);
break;
case 4:
Common::sprintf_s(textBuffer, "%s %d/%d", "Blue Mana:", curMana, baseMana);
break;
case 5:
Common::sprintf_s(textBuffer, "%s %d/%d", "Purple Mana:", curMana, baseMana);
break;
case -1:
textBuffer[0] = 0;
textBuffer[1] = 0;
break;
default:
assert(false); // should never get here
break;
}
// set the text in the cursor
g_vm->_mouseInfo->setText(textBuffer);
} else
g_vm->_mouseInfo->setText(nullptr);
}
}
bool isIndivMode() {
return g_vm->_indivControlsFlag;
}
void initUIState() {
g_vm->_indivControlsFlag = false;
indivBrother = 0;
//updateAllUserControls();
}
void saveUIState(Common::OutSaveFile *outS) {
debugC(2, kDebugSaveload, "Saving UIState");
outS->write("UIST", 4);
CHUNK_BEGIN;
out->writeUint16LE(g_vm->_indivControlsFlag);
out->writeUint16LE(indivBrother);
CHUNK_END;
debugC(3, kDebugSaveload, "..._indivControlsFlag = %d", g_vm->_indivControlsFlag);
debugC(3, kDebugSaveload, "... indivBrother = %d", indivBrother);
}
void loadUIState(Common::InSaveFile *in) {
debugC(2, kDebugSaveload, "Loading UIState");
g_vm->_indivControlsFlag = in->readUint16LE();
indivBrother = in->readUint16LE();
debugC(3, kDebugSaveload, "... _indivControlsFlag = %d", g_vm->_indivControlsFlag);
debugC(3, kDebugSaveload, "... indivBrother = %d", indivBrother);
updateAllUserControls();
}
void cleanupUIState() {
if (StatusLine != nullptr)
StatusLine->clear();
}
void gArmorIndicator::setValue(PlayerActorID brotherID) {
Actor *bro = g_vm->_playerList[brotherID]->getActor();
bro->totalArmorAttributes(_attr);
invalidate();
}
// getCurrentCompImage() is virtual function that should return
// the current image to be displayed (to be used across all sub-classes)
void gArmorIndicator::drawClipped(gPort &port,
const Point16 &offset,
const Rect16 &r) {
if (!_extent.overlap(r)) return;
SAVE_GPORT_STATE(port);
// get the current image
void *dispImage = getCurrentCompImage();
// make sure the image is valid
if (dispImage) {
// will part of this be drawn on screen?
if (_extent.overlap(r)) {
char buf[8];
// offset the image?
Point16 pos(_extent.x - offset.x,
_extent.y - offset.y
);
// draw the compressed image
if (isGhosted()) {
drawCompressedImageGhosted(port, pos, dispImage);
return;
} else drawCompressedImage(port, pos, dispImage);
// draw the armor numebrs
port.setFont(&Helv11Font);
port.setColor(11); // set color to white
port.setStyle(kTextStyleThickOutline);
port.setOutlineColor(24); // set outline color to black
port.setMode(kDrawModeMatte);
if (_attr.damageAbsorbtion == 0 && _attr.defenseBonus == 0)
Common::sprintf_s(buf, "-");
else if (_attr.damageDivider > 1)
Common::sprintf_s(buf, "%d/%d", _attr.damageAbsorbtion, _attr.damageDivider);
else Common::sprintf_s(buf, "%d", _attr.damageAbsorbtion);
port.drawTextInBox(buf, -1, Rect16(pos.x, pos.y, _extent.width, _extent.height),
kTextPosRight | kTextPosHigh, Point16(0, 2));
if (_attr.damageAbsorbtion == 0 && _attr.defenseBonus == 0)
Common::sprintf_s(buf, "-");
else Common::sprintf_s(buf, "%d", _attr.defenseBonus);
port.drawTextInBox(buf, -1, Rect16(pos.x, pos.y, _extent.width, _extent.height),
kTextPosRight | kTextPosLow, Point16(0, 2));
}
}
}
void gEnchantmentDisplay::drawClipped(gPort &port, const Point16 &offset, const Rect16 &r) {
Point16 pos(_extent.x + _extent.width - 10, _extent.y + 1);
pos += offset;
if (!_extent.overlap(r)) return;
for (int i = 0; i < iconCount; i++) {
if (_iconFlags[i]) {
Sprite *sp = mentalSprites->sprite(i + 162);
pos.x -= sp->size.x + 2;
DrawSprite(port, pos, sp);
}
}
}
void gEnchantmentDisplay::pointerMove(gPanelMessage &msg) {
if (msg._pointerLeave) {
g_vm->_mouseInfo->setText(nullptr);
} else {
int16 x = _extent.width - 10;
setMousePoll(true);
setValue(getCenterActorPlayerID());
for (int i = 0; i < iconCount; i++) {
if (_iconFlags[i]) {
Sprite *sp = mentalSprites->sprite(i + 162);
x -= sp->size.x + 2;
if (msg._pickPos.x >= x) {
// set the text in the cursor
char buf[128];
if (_iconFlags[i] == 255)
Common::sprintf_s(buf, "%s", enchantmentNames[i]);
else Common::sprintf_s(buf, "%s : %d", enchantmentNames[i], _iconFlags[i]);
g_vm->_mouseInfo->setText(buf);
return;
}
}
}
}
}
void gEnchantmentDisplay::setValue(PlayerActorID pID) {
Actor *a = g_vm->_playerList[pID]->getActor();
uint8 newIconFlags[iconCount];
EnchantmentIterator iter(a);
ContainerIterator cIter(a);
GameObject *obj = nullptr;
memset(newIconFlags, 0, sizeof newIconFlags);
/*
x iconHaste,
iconFirewalk,
x iconAdrenalFervor,
x iconShadowWalk,
iconSunWard,
iconSpellBarrier,
*/
for (ObjectID id1 = iter.first(&obj); id1 != Nothing; id1 = iter.next(&obj)) {
uint16 enchantmentID = obj->getExtra();
uint16 eType = getEnchantmentType(enchantmentID);
uint16 eSubType = getEnchantmentSubType(enchantmentID);
int16 eAmount = getEnchantmentAmount(enchantmentID);
uint8 duration = obj->getHitPoints(); // get hitpoints of enchant
switch (eType) {
case kEffectAttrib:
switch (eSubType) {
// case kSkillIDArchery:
// case kSkillIDShieldcraft:
case kSkillIDBludgeon:
case kSkillIDSwordcraft:
if (eAmount > 0) newIconFlags[iconSurestrike] = duration;
else newIconFlags[iconClumsy] = duration;
break;
case kSkillIDAgility:
if (eAmount > 0) newIconFlags[iconInnerBalance] = duration;
else newIconFlags[iconClumsy] = duration;
break;
case kSkillIDBrawn:
newIconFlags[iconBattleFever] = duration;
break;
}
break;
case kEffectResist:
switch (eSubType) {
case kDamageImpact:
newIconFlags[iconResistImpact] = duration;
break;
case kDamageSlash:
newIconFlags[iconResistSlash] = duration;
break;
case kDamageProjectile:
newIconFlags[iconResistProjectile] = duration;
break;
case kDamageFire:
newIconFlags[iconResistFire] = duration;
break;
case kDamageAcid:
newIconFlags[iconResistAcid] = duration;
break;
case kDamageHeat:
newIconFlags[iconResistHeat] = duration;
break;
case kDamageCold:
newIconFlags[iconResistCold] = duration;
break;
case kDamageLightning:
newIconFlags[iconResistLightning] = duration;
break;
case kDamagePoison:
newIconFlags[iconResistPoison] = duration;
break;
case kDamageMental:
newIconFlags[iconResistPsionic] = duration;
break;
case kDamageDirMagic:
newIconFlags[iconResistDirectMagic] = duration;
break;
}
break;
case kEffectImmune:
switch (eSubType) {
case kDamageImpact:
newIconFlags[iconIronskin] = duration;
break;
case kDamageSlash:
newIconFlags[iconIronskin] = duration;
break;
case kDamageFire:
newIconFlags[iconImmuneFire] = duration;
break;
case kDamageAcid:
newIconFlags[iconImmuneAcid] = duration;
break;
case kDamageHeat:
newIconFlags[iconImmuneHeat] = duration;
break;
case kDamageCold:
newIconFlags[iconImmuneCold] = duration;
break;
case kDamageLightning:
newIconFlags[iconImmuneLightning] = duration;
break;
case kDamagePoison:
newIconFlags[iconImmunePoison] = duration;
break;
case kDamageMental:
newIconFlags[iconImmunePsionic] = duration;
break;
}
break;
case kEffectOthers:
switch (eSubType) {
case kActorPoisoned:
case kActorDiseased:
newIconFlags[iconPoisoned] = duration;
break;
case kActorFear:
newIconFlags[iconAfraid] = duration;
break;
case kActorParalyzed:
newIconFlags[iconParalysed] = duration;
break; // iconFrozen ??
case kActorSlowFall:
newIconFlags[iconCushionAir] = duration;
break;
case kActorImmobile:
newIconFlags[iconConstrained] = duration;
break;
case kActorSeeInvis:
newIconFlags[iconSoulSight] = duration;
break;
case kActorInvisible:
newIconFlags[iconInvisible] = duration;
break;
case kActorUndetectable:
newIconFlags[iconNumbscent] = duration;
break;
case kActorDetPoison:
newIconFlags[iconDetectPoison] = duration;
break;
case kActorNoDrain:
newIconFlags[iconNetherWard] = duration;
break;
case kActorWaterBreathe:
newIconFlags[iconSeawalk] = duration;
break;
case kActorRepelEvil:
newIconFlags[iconProtectEvil] = duration;
break;
// case kActorRepelUndead: newIconFlags[iconProtectUndead] = duration; break;
// case actorRepelGhosts: newIconFlags[iconProtectGhosts] = duration; break;
}
}
}
while (cIter.next(&obj)) {
ProtoObj *proto = obj->proto();
uint16 cSet = proto->containmentSet();
if ((cSet & (ProtoObj::isArmor | ProtoObj::isWeapon | ProtoObj::isWearable))
&& proto->isObjectBeingUsed(obj)) {
if (proto->immunity & (1 << kResistImpact)) newIconFlags[iconResistImpact] = 255;
else if (proto->resistance & (1 << kResistImpact)) newIconFlags[iconResistImpact] = 255;
if (proto->immunity & (1 << kResistSlash)) newIconFlags[iconResistSlash] = 255;
else if (proto->resistance & (1 << kResistSlash)) newIconFlags[iconResistSlash] = 255;
if (proto->immunity & (1 << kResistProjectile)) newIconFlags[iconResistProjectile] = 255;
else if (proto->resistance & (1 << kResistProjectile)) newIconFlags[iconResistProjectile] = 255;
if (proto->immunity & (1 << kImmuneFire)) newIconFlags[iconImmuneFire] = 255;
else if (proto->resistance & (1 << kResistFire)) newIconFlags[iconResistFire] = 255;
if (proto->immunity & (1 << kImmuneAcid)) newIconFlags[iconImmuneAcid] = 255;
else if (proto->resistance & (1 << kResistAcid)) newIconFlags[iconResistAcid] = 255;
if (proto->immunity & (1 << kImmuneHeat)) newIconFlags[iconImmuneHeat] = 255;
else if (proto->resistance & (1 << kResistHeat)) newIconFlags[iconResistHeat] = 255;
if (proto->immunity & (1 << kImmuneCold)) newIconFlags[iconImmuneCold] = 255;
else if (proto->resistance & (1 << kResistCold)) newIconFlags[iconResistCold] = 255;
if (proto->immunity & (1 << kImmuneLightning)) newIconFlags[iconImmuneLightning] = 255;
else if (proto->resistance & (1 << kResistLightning)) newIconFlags[iconResistLightning] = 255;
if (proto->immunity & (1 << kImmunePoison)) newIconFlags[iconImmunePoison] = 255;
else if (proto->resistance & (1 << kResistPoison)) newIconFlags[iconResistPoison] = 255;
if (proto->immunity & (1 << kImmuneMental)) newIconFlags[iconImmunePsionic] = 255;
else if (proto->resistance & (1 << kResistMental)) newIconFlags[iconResistPsionic] = 255;
if (proto->immunity & (1 << kResistDirMagic)) newIconFlags[iconResistDirectMagic] = 255;
else if (proto->resistance & (1 << kResistDirMagic))newIconFlags[iconResistDirectMagic] = 255;
}
}
// Compute icon flags for resistances and immunities
#if 0
enum effectOthersTypes {
// Movement flags
o kActorNoncorporeal = 1, // The creature can walk through things
x kActorWaterBreathe = 2, // death spell
x kActorSlowFall = 3, // the creature is not harmed by falling (but falls none the less)
- kActorLevitate = 4, // flying with no height control ?
- kActorFlying = 5, // the creature flys
// speed flags
- kActorFastMove = 6, //
- kActorFastAttack = 7, //
kActorSlowAttack = 8, // come... back... here... lit... tle... bun... ny...
x actorSlowMove = 9, // I thought I told you to leave the piano at home
// ill effects
- kActorAsleep = 10, // Zzzzzzzzzzz
x kActorParalyzed = 11, // the creature can't move an inch
x kActorFear = 12, // run away! run away
x kActorDiseased = 13, // cannot heal
x kActorPoisoned = 14, // death spell
// perception & perceivability flags
x kActorBlind = 15, // can't see
x kActorSeeInvis = 16, // can see invisible
- kActorClairvoyant = 17, // unknown effects
x kActorInvisible = 18, // is invisible
x kActorUndetectable = 19, // can't be seen, smelled
x kActorDetPoison = 20, // poison things glow green
// flags preventing changes to other flags
kActorNoEnchant = 21, // no bad enchantments
x kActorNoDrain = 22, // no mana / food drains
// flags that make things run away
x kActorRepelEvil = 23, // death spell
x kActorRepelGood = 24, // death spell
x kActorRepelUndead = 25, // death spell
// dead or moved flags
// actorMapping =15, //
// actorLandWalk =0 , // someone else had this I have no idea what it is
// actorFloat =2 , // the creature can travel through malts shakes & sundaes
kActorWaterWalk, // can walk on water (same as float ?)
// actorPanic =13, // creature takes off randomly
// actorSpotHidden =17, // can see hidden
// actorDetTraps =22, // traps glow green
// actorFlameAura =23, // has a flaming aura
// actorDead =25, // death spell
};
#endif
// If icon flags changed, then redraw the control.
if (memcmp(_iconFlags, newIconFlags, sizeof _iconFlags)) {
memcpy(_iconFlags, newIconFlags, sizeof _iconFlags);
invalidate();
}
}
bool isBrotherDead(PlayerActorID brotherID) {
return (getPlayerActorAddress(brotherID)->getActor()->isDead());
}
void StatusMsg(const char *msg, ...) { // frametime def
va_list argptr;
char buffer[128];
if (StatusLine) {
va_start(argptr, msg);
Common::vsprintf_s(buffer, msg, argptr);
va_end(argptr);
StatusLine->setLine(buffer, 500);
}
}
} // end of namespace Saga2