mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-10 20:01:25 +00:00
1126 lines
30 KiB
C++
1126 lines
30 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 2
|
|
* 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
|
|
* aint32 with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
*
|
|
* Based on the original sources
|
|
* Faery Tale II -- The Halls of the Dead
|
|
* (c) 1993-1996 The Wyrmkeep Entertainment Co.
|
|
*/
|
|
|
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL // FIXME: Remove
|
|
|
|
#include "saga2/std.h"
|
|
#include "saga2/intrface.h"
|
|
#include "saga2/contain.h"
|
|
#include "saga2/task.h"
|
|
#include "saga2/motion.h"
|
|
#include "saga2/transit.h"
|
|
#include "saga2/localize.h"
|
|
#include "saga2/savefile.h"
|
|
|
|
namespace Saga2 {
|
|
|
|
extern bool allPlayerActorsDead;
|
|
|
|
extern ObjectID viewCenterObject; // ID of object which the
|
|
// camera tracks
|
|
|
|
extern ReadyContainerView *TrioCviews[kNumViews];
|
|
extern ReadyContainerView *indivCviewTop, *indivCviewBot;
|
|
extern ContainerNode *indivReadyNode;
|
|
|
|
void updateMainDisplay(void);
|
|
|
|
TilePoint selectNearbySite(
|
|
ObjectID worldID,
|
|
const TilePoint &startingCoords,
|
|
int32 minDist,
|
|
int32 maxDist,
|
|
bool offScreenOnly);
|
|
|
|
bool checkPath(
|
|
ObjectID worldID,
|
|
uint8 height,
|
|
const TilePoint &startingPt,
|
|
const TilePoint &destPt);
|
|
|
|
/* ===================================================================== *
|
|
Globals
|
|
* ===================================================================== */
|
|
|
|
static PlayerActorID centerActor; // Index of the current center
|
|
// actor
|
|
|
|
bool brotherBandingEnabled;
|
|
|
|
// Master list of all playerActor structures
|
|
PlayerActor playerList[playerActors] = {
|
|
PlayerActor(ActorBaseID + 0), // Julian
|
|
PlayerActor(ActorBaseID + 1), // Philip
|
|
PlayerActor(ActorBaseID + 2), // Kevin
|
|
};
|
|
|
|
|
|
/* ===================================================================== *
|
|
Methods
|
|
* ===================================================================== */
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Resolve the banding state of this actor
|
|
|
|
void PlayerActor::resolveBanding(void) {
|
|
Actor *follower = getActor();
|
|
Actor *centerActor = getCenterActor();
|
|
|
|
// if already following, tell the actor to cease and desist
|
|
if (follower->leader) {
|
|
follower->disband();
|
|
}
|
|
|
|
// do not allow actor to follow it's self
|
|
if (brotherBandingEnabled
|
|
&& isBanded()
|
|
&& follower != centerActor) {
|
|
// create a new follow assignment
|
|
|
|
follower->bandWith(centerActor);
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Re-evaluate the portrait type for this player actor
|
|
|
|
void PlayerActor::recalcPortraitType(void) {
|
|
PortraitType pType;
|
|
Actor *a = getActor();
|
|
ActorAttributes &stats = getBaseStats();
|
|
|
|
if (a->isDead())
|
|
pType = dead;
|
|
else if (a->enchantmentFlags & (1 << actorAsleep))
|
|
pType = asleep;
|
|
else if (stats.vitality >= a->effectiveStats.vitality * 3)
|
|
pType = wounded;
|
|
else if (a->enchantmentFlags & ((1 << actorDiseased) | (1 << actorPoisoned)))
|
|
pType = sick;
|
|
else if (stats.vitality * 2 > a->effectiveStats.vitality * 3)
|
|
pType = ouch;
|
|
else if (a->enchantmentFlags & ((1 << actorParalyzed) | (1 << actorFear) | (1 << actorBlind)))
|
|
pType = confused;
|
|
else if (isAggressive())
|
|
pType = angry;
|
|
else pType = normal;
|
|
|
|
if (pType != portraitType)
|
|
updateBrotherPortrait(getPlayerActorID(this), portraitType = pType);
|
|
}
|
|
|
|
|
|
void PlayerActor::recoveryUpdate(void) { // change name to recovery update
|
|
manaUpdate();
|
|
AttribUpdate();
|
|
}
|
|
|
|
|
|
|
|
void PlayerActor::AttribUpdate(void) {
|
|
// get the actor pointer for this character
|
|
Actor *actor = getActor();
|
|
|
|
// get the effective stats for this player actor
|
|
ActorAttributes *effStats = &actor->effectiveStats;
|
|
|
|
for (int16 i = 0; i < numSkills; i++) {
|
|
// go through each skill and update as needed
|
|
stdAttribUpdate(effStats->skill(i),
|
|
baseStats.skill(i),
|
|
i);
|
|
}
|
|
}
|
|
|
|
|
|
void PlayerActor::stdAttribUpdate(uint8 &stat, uint8 baseStat, int16 index) {
|
|
// first find out if this actor is wounded
|
|
if (stat < baseStat) {
|
|
// whole vitality number goes here
|
|
int16 recover;
|
|
int16 fractionRecover;
|
|
|
|
// get the whole number first
|
|
recover = attribPointsPerUpdate / attribPointsPerValue;
|
|
|
|
// get the fraction
|
|
fractionRecover = attribPointsPerUpdate % attribPointsPerValue;
|
|
|
|
// if there is an overrun
|
|
if (attribRecPools[index] + fractionRecover > attribPointsPerValue) {
|
|
// add the overrun to the whole number
|
|
recover++;
|
|
attribRecPools[index] = (attribRecPools[index] + fractionRecover) - attribPointsPerValue;
|
|
} else {
|
|
attribRecPools[index] += fractionRecover;
|
|
}
|
|
|
|
|
|
if (stat + recover >= baseStat) {
|
|
stat = baseStat;
|
|
} else {
|
|
stat += recover;
|
|
}
|
|
}
|
|
}
|
|
|
|
void PlayerActor::manaUpdate(void) {
|
|
const int numManas = 6;
|
|
const int minMana = 0;
|
|
|
|
// get the actor pointer for this character
|
|
Actor *actor = getActor();
|
|
|
|
// get indirections for each of the effective mana types
|
|
int16 *effectiveMana[numManas] = { &actor->effectiveStats.redMana,
|
|
&actor->effectiveStats.orangeMana,
|
|
&actor->effectiveStats.yellowMana,
|
|
&actor->effectiveStats.greenMana,
|
|
&actor->effectiveStats.blueMana,
|
|
&actor->effectiveStats.violetMana
|
|
};
|
|
|
|
// get indirections for each of the base mana types
|
|
int16 *baseMana[numManas] = { &baseStats.redMana,
|
|
&baseStats.orangeMana,
|
|
&baseStats.yellowMana,
|
|
&baseStats.greenMana,
|
|
&baseStats.blueMana,
|
|
&baseStats.violetMana
|
|
};
|
|
|
|
uint16 diff;
|
|
|
|
// do each mana type
|
|
for (int16 i = 0; i < numManas; i++) {
|
|
int levelBump;
|
|
int recRate;
|
|
|
|
// if baseMana has gone to zero, force it to 1
|
|
if (*baseMana[i] <= 0) *baseMana[i] = 1;
|
|
|
|
// Make mana harder to increase as it goes up.
|
|
if (*baseMana[i] >= 100) levelBump = 40;
|
|
else if (*baseMana[i] >= 40) levelBump = 20;
|
|
else levelBump = 10;
|
|
|
|
// is their current mana less then their maximum
|
|
if (*effectiveMana[i] < *baseMana[i]) {
|
|
diff = *effectiveMana[i];
|
|
|
|
recRate = 1;
|
|
if (*baseMana[i] >= 120) recRate = 3;
|
|
else if (*baseMana[i] >= 80) recRate = 2;
|
|
else if (*baseMana[i] >= 40) {
|
|
// This effectively causes recRate to be 1.5, i.e.
|
|
// hald of the time its 1 and the other half its 2.
|
|
if (*effectiveMana[i] % 3 == 0) recRate = 2;
|
|
}
|
|
|
|
// recover mana at specified rate
|
|
*effectiveMana[i] = clamp(minMana,
|
|
*effectiveMana[i] += recRate,
|
|
*baseMana[i]);
|
|
|
|
// get the difference between the manas
|
|
diff = *effectiveMana[i] - diff;
|
|
|
|
|
|
// find out if we're recovering from below one third
|
|
if (*effectiveMana[i] < *baseMana[i] / 3) {
|
|
// add the diff
|
|
// Deleted at request of Client.
|
|
// manaMemory[i] -= diff;
|
|
} else {
|
|
manaMemory[i] += diff;
|
|
}
|
|
|
|
|
|
// if we bumped passed the ( +/- ) levelBump mark
|
|
// decrement the base mana
|
|
*baseMana[i] += (manaMemory[i] / levelBump);
|
|
|
|
// get the fraction back to memory
|
|
manaMemory[i] = manaMemory[i] % levelBump;
|
|
|
|
//WriteStatusF( 4, " mana: %d", *effectiveMana[i] );
|
|
}
|
|
}
|
|
}
|
|
|
|
// get the skill advancement number ( could be zero )
|
|
void PlayerActor::skillAdvance(SkillProto *proto,
|
|
uint8 points,
|
|
uint8 useMult) { // useMult defaulted to 1
|
|
// get the skill level for the skill passed ( i.e. 1-100 )
|
|
uint8 skillLevel = getSkillLevel(proto, true); // true, use base stats
|
|
|
|
// get the stat index in question
|
|
uint8 stat = getStatIndex(proto);
|
|
|
|
// get the percentile chance of advancing
|
|
uint8 advanceChance = ActorAttributes::skillBasePercent - skillLevel;
|
|
|
|
// call the main body of code
|
|
skillAdvance(stat, advanceChance, points, useMult);
|
|
}
|
|
|
|
// get the skill advancement number ( could be zero )
|
|
void PlayerActor::skillAdvance(ActorSkillID stat,
|
|
uint8 points,
|
|
uint8 useMult) { // useMult defaulted to 1
|
|
// get the skill level for the skill passed ( i.e. 1-100 )
|
|
uint8 skillLevel = clamp(0, baseStats.skill(stat), ActorAttributes::skillMaxLevel);
|
|
|
|
// get the percentile chance of advancing
|
|
uint8 advanceChance = ActorAttributes::skillBasePercent - skillLevel;
|
|
|
|
// call the main body of code
|
|
skillAdvance(stat, advanceChance, points, useMult);
|
|
|
|
//WriteStatusF( 1, "adchan: %d, skillev: %d ", advanceChance, skillLevel );
|
|
}
|
|
|
|
|
|
void PlayerActor::skillAdvance(uint8 stat,
|
|
uint8 advanceChance,
|
|
uint8 points,
|
|
uint8 useMult) {
|
|
// roll percentile dice
|
|
if (rand() % 100 < advanceChance) {
|
|
uint8 increase;
|
|
int16 oldValue = baseStats.skill(stat) / ActorAttributes::skillFracPointsPerLevel;
|
|
|
|
// success, now apply the multiplyer
|
|
attribMemPools[stat] += points * useMult;
|
|
|
|
// get the amout of whole increase points
|
|
increase = attribMemPools[stat] / ActorAttributes::skillFracPointsPerLevel;
|
|
|
|
// now set the pool with the fraction
|
|
attribMemPools[stat] =
|
|
attribMemPools[stat]
|
|
- increase
|
|
* ActorAttributes::skillFracPointsPerLevel;
|
|
|
|
// now apply changes to the baseStats
|
|
baseStats.skill(stat) = clamp(
|
|
0,
|
|
baseStats.skill(stat) += increase,
|
|
ActorAttributes::skillMaxLevel);
|
|
|
|
if (baseStats.skill(stat) / ActorAttributes::skillFracPointsPerLevel != oldValue) {
|
|
static char *skillNames[] = {
|
|
ARCHERY_SKILL,
|
|
SWORD_SKILL,
|
|
SHIELD_SKILL,
|
|
BLUDGEON_SKILL,
|
|
DEAD_SKILL,
|
|
SPELL_SKILL,
|
|
DEAD2_SKILL,
|
|
AGILITY_SKILL,
|
|
BRAWN_SKILL
|
|
};
|
|
|
|
StatusMsg(SKILL_STATUS, getActor()->objName(), skillNames[stat]);
|
|
}
|
|
//WriteStatusF( 6, "frac: %d inc: %d, base: %d", attribMemPools[stat], increase, baseStats.allSkills[stat] );
|
|
}
|
|
}
|
|
|
|
void PlayerActor::vitalityAdvance(uint8 points) {
|
|
char buffer[64];
|
|
|
|
while (points-- > 0) {
|
|
if (rand() % ActorAttributes::vitalityLimit > baseStats.vitality) {
|
|
if (++vitalityMemory >= vitalityLevelBump) {
|
|
vitalityMemory -= vitalityLevelBump;
|
|
baseStats.vitality++;
|
|
StatusMsg(VITALITY_STATUS, getActor()->objName());
|
|
}
|
|
}
|
|
}
|
|
|
|
assert(baseStats.vitality < ActorAttributes::vitalityLimit);
|
|
}
|
|
|
|
// this function will return a value of 0 - 4 to indicate
|
|
// relative proficiency in requested skill
|
|
int8 PlayerActor::getSkillLevel(SkillProto *skill, bool base) { // basestats defaulted to false
|
|
// get the id for this skill
|
|
SpellID skillID = skill->getSpellID();
|
|
|
|
// index
|
|
uint16 stat;
|
|
|
|
// get the current stats for this player actor
|
|
ActorAttributes *effStats = getEffStats();
|
|
|
|
// check to see if this is a special case
|
|
if (skillID == skillVitality) {
|
|
return effStats->vitality / ActorAttributes::skillFracPointsPerLevel;
|
|
} else if (skillID == skillCartography) {
|
|
// cartography has no levels
|
|
return 0;
|
|
}
|
|
|
|
// get the index
|
|
stat = getStatIndex(skill);
|
|
|
|
|
|
// stat stored as skillLevel *
|
|
// skillFracPointsPerLevel +
|
|
// skillCurrentFracPoints
|
|
if (base) {
|
|
return clamp(0,
|
|
baseStats.skill(stat) / ActorAttributes::skillFracPointsPerLevel,
|
|
ActorAttributes::skillLevels - 1);
|
|
} else {
|
|
return clamp(0,
|
|
effStats->skill(stat) / ActorAttributes::skillFracPointsPerLevel,
|
|
ActorAttributes::skillLevels - 1);
|
|
}
|
|
}
|
|
|
|
uint8 PlayerActor::getStatIndex(SkillProto *proto) {
|
|
// get the id for this skill
|
|
SpellID skillID = proto->getSpellID();
|
|
uint16 stat;
|
|
|
|
// get the current stats for this player actor
|
|
ActorAttributes *effStats = getEffStats();
|
|
|
|
// now map the id gotten from spellid to the
|
|
// attributeskilll enum for the allSkills array
|
|
switch (skillID) {
|
|
case skillPickpocket:
|
|
stat = skillIDPilfer;
|
|
break;
|
|
|
|
case skillSeeHidden:
|
|
stat = skillIDSpotHidden;
|
|
break;
|
|
|
|
case skillLockPick:
|
|
stat = skillIDLockpick;
|
|
break;
|
|
|
|
case skillFirstAid:
|
|
stat = skillIDFirstAid;
|
|
break;
|
|
|
|
case skillArchery:
|
|
stat = skillIDArchery;
|
|
break;
|
|
|
|
case skillSwordcraft:
|
|
stat = skillIDSwordcraft;
|
|
break;
|
|
|
|
case skillShieldcraft:
|
|
stat = skillIDShieldcraft;
|
|
break;
|
|
|
|
case skillBludgeon:
|
|
stat = skillIDBludgeon;
|
|
break;
|
|
|
|
case skillThrowing:
|
|
stat = skillIDThrowing;
|
|
break;
|
|
|
|
case skillSpellcraft:
|
|
stat = skillIDSpellcraft;
|
|
break;
|
|
|
|
case skillStealth:
|
|
stat = skillIDStealth;
|
|
break;
|
|
|
|
case skillAgility:
|
|
stat = skillIDAgility;
|
|
break;
|
|
|
|
case skillBrawn:
|
|
stat = skillIDBrawn;
|
|
break;
|
|
|
|
default:
|
|
|
|
error("Invalid case detected: Player.cpp-getSkillLevel() = %d", skillID);
|
|
}
|
|
|
|
// make sure we have a good index
|
|
if (stat >= numSkills) {
|
|
error("Invalid array index detected: Player.cpp-getSkillLevel()");
|
|
}
|
|
|
|
// return the index
|
|
return stat;
|
|
}
|
|
|
|
ActorAttributes *PlayerActor::getEffStats(void) {
|
|
// get the actor pointer for this character
|
|
Actor *actor = getActor();
|
|
|
|
// get the effective stats for this player actor
|
|
ActorAttributes *effStats = &actor->effectiveStats;
|
|
|
|
// valid?
|
|
assert(effStats);
|
|
|
|
// return current stats for this player actor
|
|
return effStats;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Notify the user of attack if necessary
|
|
|
|
void PlayerActor::handleAttacked(void) {
|
|
if (!notifiedOfAttack) {
|
|
StatusMsg(ATTACK_STATUS, getActor()->objName());
|
|
notifiedOfAttack = true;
|
|
}
|
|
}
|
|
|
|
/* ===================================================================== *
|
|
Functions
|
|
* ===================================================================== */
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Return a pointer to a PlayerActor given it's ID
|
|
|
|
PlayerActor *getPlayerActorAddress(PlayerActorID id) {
|
|
assert(id >= 0 && id < elementsof(playerList));
|
|
|
|
return &playerList[id];
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Return a PlayerActor ID given it's address
|
|
|
|
PlayerActorID getPlayerActorID(PlayerActor *p) {
|
|
return p - playerList;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Return a pointer the center actor's Actor structure
|
|
|
|
Actor *getCenterActor(void) {
|
|
return playerList[centerActor].getActor();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Return the center actor's object ID
|
|
|
|
ObjectID getCenterActorID(void) {
|
|
return playerList[centerActor].getActorID();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Return the center actor's player actor ID
|
|
|
|
PlayerActorID getCenterActorPlayerID(void) {
|
|
return centerActor;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Set a new center actor based upon a PlayerActor ID
|
|
|
|
void setCenterActor(PlayerActorID newCenter) {
|
|
extern void setEnchantmentDisplay(void);
|
|
|
|
assert(newCenter < playerActors);
|
|
|
|
Actor *a = playerList[newCenter].getActor();
|
|
PlayerActorIterator iter;
|
|
PlayerActor *player;
|
|
|
|
// If this actor is dead return immediately
|
|
if (a->isDead()) return;
|
|
|
|
// Take previous center actor out of fight stance
|
|
getCenterActor()->setFightStance(false);
|
|
|
|
// get rid of any following assignments the center actor might have
|
|
if (a->leader) {
|
|
a->disband();
|
|
}
|
|
|
|
centerActor = newCenter;
|
|
viewCenterObject = playerList[centerActor].getActorID();
|
|
|
|
indivReadyNode->changeOwner(newCenter);
|
|
globalContainerList.setPlayerNum(newCenter);
|
|
setEnchantmentDisplay();
|
|
|
|
if (a->curTask != NULL) {
|
|
a->curTask->abortTask();
|
|
delete a->curTask;
|
|
a->curTask = NULL;
|
|
}
|
|
|
|
// Set the new centers fight stance based upon his aggression state
|
|
a->setFightStance(playerList[newCenter].isAggressive());
|
|
|
|
// band actors to new center if banding button set
|
|
for (player = iter.first(); player != NULL; player = iter.next()) {
|
|
player->resolveBanding();
|
|
}
|
|
|
|
// clear the last center actor's button state
|
|
updateBrotherRadioButtons(newCenter);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Set a new center actor based upon an Actor address
|
|
|
|
void setCenterActor(Actor *newCenter) {
|
|
assert(newCenter->disposition >= dispositionPlayer);
|
|
setCenterActor(newCenter->disposition - dispositionPlayer);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Set a new center actor based upon a PlayerActor address
|
|
|
|
void setCenterActor(PlayerActor *newCenter) {
|
|
assert(newCenter >= playerList && newCenter < &playerList[playerActors]);
|
|
setCenterActor(newCenter - playerList);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Return the coordinates of the current center actor
|
|
|
|
TilePoint centerActorCoords(void) {
|
|
Actor *a;
|
|
|
|
a = playerList[centerActor].getActor();
|
|
return a->getLocation();
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Set or clear a player's aggressive state
|
|
|
|
void setAggression(PlayerActorID player, bool aggression) {
|
|
assert(player >= 0 && player < playerActors);
|
|
|
|
Actor *a = playerList[player].getActor();
|
|
|
|
if (a->isDead()) return;
|
|
|
|
if (aggression)
|
|
playerList[player].setAggression();
|
|
else
|
|
playerList[player].clearAggression();
|
|
|
|
if (player == centerActor)
|
|
a->setFightStance(aggression);
|
|
|
|
a->evaluateNeeds();
|
|
|
|
updateBrotherAggressionButton(player, aggression);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Determine if player actor is in an aggressive state
|
|
|
|
bool isAggressive(PlayerActorID player) {
|
|
assert(player >= 0 && player < playerActors);
|
|
return playerList[player].isAggressive();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Adjust the player actors aggression setting based upon their
|
|
// proximity to enemies
|
|
|
|
void autoAdjustAggression(void) {
|
|
PlayerActorID i;
|
|
|
|
// Iterate through all player actors
|
|
for (i = 0; i < playerActors; i++) {
|
|
if (i == centerActor || isBanded(i)) {
|
|
bool enemiesPresent = false;
|
|
Actor *actor = playerList[i].getActor();
|
|
|
|
if (actor->getStats()->vitality >= minAutoAggressionVitality) {
|
|
GameObject *obj;
|
|
ActiveRegion *activeReg = getActiveRegion(i);
|
|
TileRegion region = activeReg->getRegion();
|
|
GameWorld *world = activeReg->getWorld();
|
|
|
|
RegionalObjectIterator iter(world, region.min, region.max);
|
|
|
|
// Iterate through the objects in this player actor's
|
|
// active region to determine if their are enemy actor's
|
|
// in the vicinity.
|
|
for (iter.first(&obj); obj != NULL; iter.next(&obj)) {
|
|
Actor *a;
|
|
|
|
if (!isActor(obj)) continue;
|
|
|
|
a = (Actor *)obj;
|
|
|
|
if (a->disposition == dispositionEnemy) {
|
|
enemiesPresent = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Set this player actor's aggression
|
|
setAggression(i, enemiesPresent);
|
|
}
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Set a player actor's banding
|
|
|
|
void setBanded(PlayerActorID player, bool banded) {
|
|
assert(player >= 0 && player < playerActors);
|
|
|
|
if (playerList[player].getActor()->isDead()) return;
|
|
|
|
if (banded)
|
|
playerList[player].setBanded();
|
|
else
|
|
playerList[player].clearBanded();
|
|
|
|
playerList[player].resolveBanding();
|
|
|
|
updateBrotherBandingButton(player, banded);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Determine if a player actor is banded
|
|
|
|
bool isBanded(PlayerActorID player) {
|
|
assert(player >= 0 && player < playerActors);
|
|
return playerList[player].isBanded();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Globally enable or disable brother banding
|
|
|
|
void setBrotherBanding(bool enabled) {
|
|
if (brotherBandingEnabled != enabled) {
|
|
brotherBandingEnabled = enabled;
|
|
|
|
if (areActorsInitialized()) {
|
|
PlayerActorIterator iter;
|
|
PlayerActor *player;
|
|
|
|
// Update the state of the banding
|
|
for (player = iter.first(); player != NULL; player = iter.next()) {
|
|
player->resolveBanding();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Set the portrait type for this actor based on the current state.
|
|
|
|
void recalcPortraitType(PlayerActorID brotherID) {
|
|
PlayerActor *pa = getPlayerActorAddress(brotherID);
|
|
|
|
pa->recalcPortraitType();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Returns an integer value representing this player actor's portrait
|
|
// state
|
|
|
|
int16 getPortraitType(PlayerActorID id) {
|
|
PlayerActor *pa = getPlayerActorAddress(id);
|
|
|
|
return pa->getPortraitType();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Given an actor, returns the corresponding player Actor ID
|
|
|
|
bool actorToPlayerID(Actor *a, PlayerActorID &result) {
|
|
if (a->disposition >= dispositionPlayer) {
|
|
result = a->disposition - dispositionPlayer;
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool actorIDToPlayerID(ObjectID id, PlayerActorID &result) {
|
|
if (!isActor(id)) return false;
|
|
|
|
Actor *a = (Actor *)GameObject::objectAddress(id);
|
|
|
|
if (a->disposition >= dispositionPlayer) {
|
|
result = a->disposition - dispositionPlayer;
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void handlePlayerActorDeath(PlayerActorID id) {
|
|
assert(id >= 0 && id < playerActors);
|
|
|
|
if (getCenterActor()->isDead()) {
|
|
PlayerActor *newCenter;
|
|
LivingPlayerActorIterator iter;
|
|
|
|
if ((newCenter = iter.first()) != NULL)
|
|
setCenterActor(getPlayerActorID(newCenter));
|
|
else
|
|
allPlayerActorsDead = true;
|
|
}
|
|
|
|
PlayerActor *player = &playerList[id];
|
|
|
|
player->clearAggression();
|
|
player->clearBanded();
|
|
updateBrotherAggressionButton(id, false);
|
|
updateBrotherBandingButton(id, false);
|
|
|
|
StatusMsg(DEATH_STATUS, player->getActor()->objName());
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Transport the center actor and the banded brothers who have a path
|
|
// to the center actor
|
|
|
|
void transportCenterBand(const Location &loc) {
|
|
assert(isWorld(loc.context));
|
|
|
|
fadeDown();
|
|
|
|
Actor *center = getCenterActor();
|
|
TilePoint centerLoc = center->getLocation();
|
|
ObjectID centerWorldID = center->world()->thisID();
|
|
PlayerActor *player;
|
|
LivingPlayerActorIterator iter;
|
|
|
|
center->move(loc);
|
|
if (center->moveTask != NULL) center->moveTask->finishWalk();
|
|
|
|
for (player = iter.first();
|
|
player != NULL;
|
|
player = iter.next()) {
|
|
Actor *a = player->getActor();
|
|
|
|
if (a != center
|
|
&& player->isBanded()
|
|
&& a->world()->thisID() == centerWorldID
|
|
&& checkPath(
|
|
centerWorldID,
|
|
a->proto()->height,
|
|
a->getLocation(),
|
|
centerLoc)) {
|
|
TilePoint dest;
|
|
|
|
dest = selectNearbySite(
|
|
loc.context,
|
|
loc,
|
|
1,
|
|
3,
|
|
false);
|
|
|
|
if (dest != Nowhere) {
|
|
a->move(Location(dest, loc.context));
|
|
if (a->moveTask != NULL) a->moveTask->finishWalk();
|
|
player->resolveBanding();
|
|
}
|
|
}
|
|
}
|
|
|
|
updateMainDisplay();
|
|
|
|
fadeUp();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
|
|
void handlePlayerActorAttacked(PlayerActorID id) {
|
|
PlayerActor *pa = getPlayerActorAddress(id);
|
|
|
|
pa->handleAttacked();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
|
|
void handleEndOfCombat(void) {
|
|
PlayerActorID i;
|
|
|
|
// Iterate through all player actors
|
|
for (i = 0; i < playerActors; i++)
|
|
playerList[i].resetAttackNotification();
|
|
}
|
|
|
|
/* ======================================================================= *
|
|
PlayerActor list management functions
|
|
* ======================================================================= */
|
|
|
|
|
|
|
|
// This structure is used in archiving the player actor list
|
|
struct PlayerActorArchive {
|
|
int16 portraitType;
|
|
uint16 flags;
|
|
ActorAttributes baseStats;
|
|
int16 manaMemory[numManas];
|
|
uint8 attribRecPools[numSkills];
|
|
uint8 attribMemPools[numSkills];
|
|
uint8 vitalityMemory;
|
|
bool notifiedOfAttack;
|
|
};
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Initialize the player list
|
|
|
|
void initPlayerActors(void) {
|
|
PlayerActorID i;
|
|
|
|
for (i = 0; i < playerActors; i++) {
|
|
PlayerActor *p = &playerList[i];
|
|
Actor *a = p->getActor();
|
|
ActorProto *proto = (ActorProto *)a->proto();
|
|
|
|
// Set the portrait type
|
|
p->portraitType = normal;
|
|
|
|
// Clear all flags
|
|
p->flags = 0;
|
|
// Copy the base stats from the actor's prototype
|
|
memcpy(&p->baseStats, &proto->baseStats, sizeof(p->baseStats));
|
|
|
|
// Clear out the accumulation arrays
|
|
memset(&p->manaMemory, 0, sizeof(p->manaMemory));
|
|
memset(&p->attribRecPools, 0, sizeof(p->attribRecPools));
|
|
memset(&p->attribMemPools, 0, sizeof(p->attribMemPools));
|
|
|
|
// Clear the vitalityMemory
|
|
p->vitalityMemory = 0;
|
|
|
|
// Clear the attack notification flag
|
|
p->notifiedOfAttack = false;
|
|
|
|
// Set the actor's disposition field to reflect that that
|
|
// actor is a player actor
|
|
a->disposition = dispositionPlayer + i;
|
|
|
|
// Turn on banding for player actors
|
|
setBanded(i, true);
|
|
}
|
|
|
|
readyContainerSetup();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Save the player list data to a save file
|
|
|
|
void savePlayerActors(SaveFileConstructor &saveGame) {
|
|
int16 i;
|
|
PlayerActorArchive archiveBuffer[playerActors];
|
|
|
|
for (i = 0; i < playerActors; i++) {
|
|
PlayerActor *p = &playerList[i];
|
|
PlayerActorArchive *a = &archiveBuffer[i];
|
|
|
|
// Store the portrait type
|
|
a->portraitType = p->portraitType;
|
|
|
|
// Store the flags
|
|
a->flags = p->flags;
|
|
|
|
// Store the base stats
|
|
memcpy(&a->baseStats, &p->baseStats, sizeof(a->baseStats));
|
|
|
|
// Store accumulation arrays
|
|
memcpy(
|
|
&a->manaMemory,
|
|
&p->manaMemory,
|
|
sizeof(a->manaMemory));
|
|
memcpy(
|
|
&a->attribRecPools,
|
|
&p->attribRecPools,
|
|
sizeof(a->attribRecPools));
|
|
memcpy(
|
|
&a->attribMemPools,
|
|
&p->attribMemPools,
|
|
sizeof(a->attribMemPools));
|
|
|
|
// Store the vitality memory
|
|
a->vitalityMemory = p->vitalityMemory;
|
|
|
|
// Store the attack notification flag
|
|
a->notifiedOfAttack = p->notifiedOfAttack;
|
|
}
|
|
|
|
// Write the player actor chunk
|
|
saveGame.writeChunk(
|
|
MakeID('P', 'L', 'Y', 'R'),
|
|
archiveBuffer,
|
|
sizeof(archiveBuffer));
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Load the player list data from a save file
|
|
|
|
void loadPlayerActors(SaveFileReader &saveGame) {
|
|
int16 i;
|
|
PlayerActorArchive archiveBuffer[playerActors];
|
|
|
|
saveGame.read(archiveBuffer, sizeof(archiveBuffer));
|
|
|
|
for (i = 0; i < playerActors; i++) {
|
|
PlayerActor *p = &playerList[i];
|
|
PlayerActorArchive *a = &archiveBuffer[i];
|
|
|
|
// Restore the portrait type
|
|
p->portraitType = a->portraitType;
|
|
|
|
// Restore the flags
|
|
p->flags = a->flags;
|
|
|
|
// Restore the base stats
|
|
memcpy(&p->baseStats, &a->baseStats, sizeof(p->baseStats));
|
|
|
|
// Restore the accumulation arrays
|
|
memcpy(
|
|
&p->manaMemory,
|
|
&a->manaMemory,
|
|
sizeof(p->manaMemory));
|
|
memcpy(
|
|
&p->attribRecPools,
|
|
&a->attribRecPools,
|
|
sizeof(p->attribRecPools));
|
|
memcpy(
|
|
&p->attribMemPools,
|
|
&a->attribMemPools,
|
|
sizeof(p->attribMemPools));
|
|
|
|
// Restore the vitality memory
|
|
p->vitalityMemory = a->vitalityMemory;
|
|
|
|
// Restore the attack notification flag
|
|
p->notifiedOfAttack = a->notifiedOfAttack;
|
|
}
|
|
|
|
readyContainerSetup();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Cleanup the player actor list
|
|
|
|
void cleanupPlayerActors(void) {
|
|
cleanupReadyContainers();
|
|
}
|
|
|
|
/* ======================================================================= *
|
|
CenterActor management function prototypes
|
|
* ======================================================================= */
|
|
|
|
// This structure is used in archiving the center actor ID and the view
|
|
// object ID
|
|
struct CenterActorArchive {
|
|
PlayerActorID centerActor;
|
|
ObjectID viewCenterObject;
|
|
};
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Initialize the center actor ID and view object ID
|
|
|
|
void initCenterActor(void) {
|
|
centerActor = FTA_JULIAN;
|
|
viewCenterObject = playerList[centerActor].getActorID();
|
|
|
|
// clear the last center actor's button state
|
|
updateBrotherRadioButtons(FTA_JULIAN);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Save the center actor ID and the view object ID to a save file
|
|
|
|
void saveCenterActor(SaveFileConstructor &saveGame) {
|
|
CenterActorArchive a;
|
|
|
|
// Store the center actor and view object
|
|
a.centerActor = centerActor;
|
|
a.viewCenterObject = viewCenterObject;
|
|
|
|
saveGame.writeChunk(MakeID('C', 'N', 'T', 'R'), &a, sizeof(a));
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Load the center actor ID and the view object ID from the save file
|
|
|
|
void loadCenterActor(SaveFileReader &saveGame) {
|
|
CenterActorArchive a;
|
|
|
|
saveGame.read(&a, sizeof(a));
|
|
|
|
// Restore the center actor and view object
|
|
centerActor = a.centerActor;
|
|
viewCenterObject = a.viewCenterObject;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Iterates through all player actors
|
|
|
|
PlayerActor *PlayerActorIterator::first(void) {
|
|
index = 0;
|
|
return &playerList[index++];
|
|
}
|
|
|
|
PlayerActor *PlayerActorIterator::next(void) {
|
|
return (index < playerActors) ? &playerList[index++] : NULL;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Iterates through all player actors that are not dead.
|
|
|
|
PlayerActor *LivingPlayerActorIterator::first(void) {
|
|
index = 0;
|
|
return LivingPlayerActorIterator::next();
|
|
}
|
|
|
|
PlayerActor *LivingPlayerActorIterator::next(void) {
|
|
Actor *a = playerList[index].getActor();
|
|
|
|
while (a == NULL || a->isDead()) {
|
|
if (++index >= playerActors) break;
|
|
a = playerList[index].getActor();
|
|
}
|
|
|
|
return (index < playerActors) ? &playerList[index++] : NULL;
|
|
}
|
|
|
|
} // end of namespace Saga2
|