scummvm/engines/saga2/dispnode.cpp
2022-10-29 23:46:00 +02:00

1019 lines
29 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/blitters.h"
#include "saga2/detection.h"
#include "saga2/spelshow.h"
#include "saga2/player.h"
#include "saga2/sensor.h"
#include "saga2/mouseimg.h"
namespace Saga2 {
uint8 bubbleColorTable[] = { 1, 0, 0, 0 };
DisplayNode *DisplayNodeList::_head;
bool centerActorIndicatorEnabled;
/* ===================================================================== *
Imports
* ===================================================================== */
extern WorldMapData *mapList;
extern StaticPoint16 fineScroll;
extern SpriteSet *objectSprites, // object sprites
*spellSprites; // spell effect sprites
ActorAppearance *tempAppearance; // test structure
/* ===================================================================== *
Test spell crap
* ===================================================================== */
bool InCombatPauseKludge();
//void updateSpellPos( int32 delTime );
//-----------------------------------------------------------------------
// build the list of stuff to draw (like guns)
uint8 identityColors[256] = {
0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15,
16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31,
32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47,
48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63,
64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79,
80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 91, 92, 93, 94, 95,
96, 97, 98, 99, 100, 101, 102, 103, 104, 105, 106, 107, 108, 109, 110, 111,
112, 113, 114, 115, 116, 117, 118, 119, 120, 121, 122, 123, 124, 125, 126, 127,
128, 129, 130, 131, 132, 133, 134, 135, 136, 137, 138, 139, 140, 141, 142, 143,
144, 145, 146, 147, 148, 149, 150, 151, 152, 153, 154, 155, 156, 157, 158, 159,
160, 161, 162, 163, 164, 165, 166, 167, 168, 169, 170, 171, 172, 173, 174, 175,
176, 177, 178, 179, 180, 181, 182, 183, 184, 185, 186, 187, 188, 189, 190, 191,
192, 193, 194, 195, 196, 197, 198, 199, 200, 201, 202, 203, 204, 205, 206, 207,
208, 209, 210, 211, 212, 213, 214, 215, 216, 217, 218, 219, 220, 221, 222, 223,
224, 225, 226, 227, 228, 229, 230, 231, 232, 233, 234, 235, 236, 237, 238, 239,
240, 241, 242, 243, 244, 245, 246, 247, 248, 249, 250, 251, 252, 253, 254, 255
};
//-----------------------------------------------------------------------
// build the list of stuff to draw (like guns)
void buildDisplayList() {
g_vm->_mainDisplayList->buildObjects(true);
g_vm->_activeSpells->buildList();
}
//-----------------------------------------------------------------------
// Update all objects which have no motion task
void updateObjectAppearances(int32 deltaTime) {
g_vm->_mainDisplayList->updateOStates(deltaTime);
#ifdef WEWANTSPELLSTOSTOPINCOMBAT
if (!InCombatPauseKludge())
#endif
g_vm->_activeSpells->updateStates(deltaTime);
}
//-----------------------------------------------------------------------
// Draw all sprites on the display list
void drawDisplayList() {
g_vm->_mainDisplayList->draw();
}
void DisplayNodeList::init(uint16 s) {
for (int i = 0; i < s; i++) {
_displayList[i]._efx = nullptr;
_displayList[i]._nextDisplayed = nullptr;
_displayList[i]._object = nullptr;
_displayList[i]._type = kNodeTypeObject;
}
}
//-----------------------------------------------------------------------
// DisplayNode stuff
DisplayNode::DisplayNode() {
_nextDisplayed = nullptr;
_sortDepth = 0;
_object = nullptr;
_flags = 0; // various flags
_type = kNodeTypeObject;
_efx = nullptr;
}
TilePoint DisplayNode::SpellPos() {
if (_efx)
return _efx->_current;
return Nowhere;
}
inline void DisplayNode::updateEffect(const int32 deltaTime) {
if (_efx) _efx->updateEffect(deltaTime);
}
//-----------------------------------------------------------------------
// Update router
void DisplayNodeList::updateOStates(const int32 deltaTime) {
if (_count)
for (uint16 i = 0; i < _count; i++)
_displayList[i].updateObject(deltaTime);
}
void DisplayNodeList::updateEStates(const int32 deltaTime) {
if (_count)
for (uint16 i = 0; i < _count; i++)
_displayList[i].updateEffect(deltaTime);
}
//-----------------------------------------------------------------------
// Draw router
void DisplayNodeList::draw() {
DisplayNode *dn;
SpriteSet *objectSet,
*spellSet;
objectSet = objectSprites;
if (objectSet == nullptr)
error("Object sprites have been dumped!\n");
if (g_vm->getGameId() == GID_FTA2) {
spellSet = spellSprites;
if (spellSet == nullptr)
error("Spell sprites have been dumped!\n");
}
for (dn = DisplayNodeList::_head; dn; dn = dn->_nextDisplayed) {
if (dn->_type == kNodeTypeEffect)
dn->drawEffect();
else
dn->drawObject();
}
}
//-----------------------------------------------------------------------
// This routine searches through the map and finds the 64
// objects or actors which are closest to the center view point.
void DisplayNodeList::buildObjects(bool fromScratch) {
GameObject *sortList[kMaxDisplayed + 1];
int16 distList[kMaxDisplayed + 1];
int16 sortCount = 0;
int16 i;
int16 viewSize = kTileRectHeight;
// Distance at which characters should be loaded.
int16 loadDist = viewSize + viewSize / 2;
// Run through list generated from previous incarnation,
// and put to bed all actors which are too far away from the
// view region.
for (i = 0; i < _count; i++) {
DisplayNode *dn = &_displayList[i];
GameObject *obj = dn->_object;
TilePoint objLoc = obj->getLocation();
int16 dist;
// Compute distance from object to screen center.
dist = ABS(viewCenter.u - objLoc.u)
+ ABS(viewCenter.v - objLoc.v);
// Determine if the object is beyond the screen threshold
if ((dist >= loadDist
|| obj->IDParent() != currentWorld->thisID())) {
// Mark this object as being off-screen
obj->setOnScreen(false);
// If it's an actor
if (isActor(obj)) {
Actor *a = (Actor *)obj;
// Release the actor appearance if loaded
if (a->_appearance != nullptr) {
ReleaseActorAppearance(a->_appearance);
a->_appearance = nullptr;
}
}
}
}
if (currentWorld == nullptr) return;
DispRegionObjectIterator iter(currentWorld, viewCenter, loadDist);
GameObject *obj = nullptr;
ObjectID id;
int16 dist = 0;
Actor *centerActor = getCenterActor();
if (fromScratch)
// Reset the list...
DisplayNodeList::_head = nullptr;
for (id = iter.first(&obj, &dist);
id != Nothing;
id = iter.next(&obj, &dist)) {
// Of object is anywhere near screen center,
// then insert object into array, sorted by
// distance.
// Also, don't add object to display list if it's
// invisible.
if (!(obj->isInvisible())) {
// Special processing for actors to "wake up"
if (isActor(id)) {
Actor *a = (Actor *)obj;
// If actor is newly entered to the arena
// (appearance == NULL), then load the
// actor's appearance.
if (a->_appearance == nullptr) {
a->_appearance =
LoadActorAppearance(a->_appearanceID, kSprStandBank);
}
}
// An insertion sort which has been clamped
// to a limited number of items.
for (i = sortCount; i > 0;) {
if (dist >= distList[i - 1]) break;
i--;
distList[i + 1] = distList[i];
sortList[i + 1] = sortList[i];
}
if (i < kMaxDisplayed) {
distList[i] = dist;
sortList[i] = obj;
if (sortCount < kMaxDisplayed) sortCount++;
}
}
}
// Build display nodes for each of the objects.
_count = sortCount;
for (i = 0; i < sortCount; i++) {
DisplayNode *dn = &_displayList[i];
GameObject *ob = sortList[i];
DisplayNode **search;
TilePoint oLoc = ob->getLocation();
dn->_nextDisplayed = nullptr;
dn->_object = ob;
dn->_type = kNodeTypeObject;
dn->_flags = 0;
if (centerActorIndicatorEnabled
&& isActor(dn->_object)
&& ((Actor *)dn->_object) == centerActor)
dn->_flags |= DisplayNode::kDisplayIndicator;
// Various test data
// dn->spriteFrame = 0;
// Convert object coordinates to screen coords
TileToScreenCoords(oLoc, dn->_screenCoords);
// REM: At this point we could reject some more off-screen
// objects.
// Set the sort depth for this object
dn->_sortDepth = dn->_screenCoords.y + oLoc.z / 2;
// Find where we belong on the sorted list
for (search = &DisplayNodeList::_head;
*search;
search = &(*search)->_nextDisplayed) {
if ((*search)->_sortDepth >= dn->_sortDepth) break;
}
// Insert into the sorted list
dn->_nextDisplayed = *search;
*search = dn;
}
}
//-----------------------------------------------------------------------
// Update normal objects
void DisplayNode::updateObject(const int32 deltaTime) {
GameObject *obj = _object;
if (obj->isMoving()) return;
if (isActor(obj)) {
Actor *a = (Actor *)obj;
a->updateAppearance(deltaTime);
}
}
//-----------------------------------------------------------------------
// Draw sprites for normal objects
void DisplayNode::drawObject() {
const int maxSpriteWidth = (g_vm->getGameId() == GID_FTA2) ? 32 : 320;
const int maxSpriteHeight = (g_vm->getGameId() == GID_FTA2) ? 120 : 320;
const int maxSpriteBaseLine = (g_vm->getGameId() == GID_FTA2) ? 16 : 50;
ColorTable mainColors, // colors for object
leftColors, // colors for left-hand object
rightColors; // colors for right-hand object
SpriteComponent scList[3],
*sc;
int16 bodyIndex, // drawing order of body
leftIndex, // drawing order of left
rightIndex, // drawing order of right
partCount; // number of sprite parts
bool ghostIt = false;
GameObject *obj = _object;
ProtoObj *proto = obj->proto();
Point16 drawPos;
SpriteSet *ss;
Sprite *bodySprite;
ActorAppearance *aa = nullptr;
SpriteSet *sprPtr = nullptr;
TilePoint objCoords = obj->getLocation(),
tCoords,
mCoords;
MetaTile *mt;
RipTable *rt;
tCoords.u = (objCoords.u >> kTileUVShift) & kPlatMask;
tCoords.v = (objCoords.v >> kTileUVShift) & kPlatMask;
mCoords.u = objCoords.u >> (kTileUVShift + kPlatShift);
mCoords.v = objCoords.v >> (kTileUVShift + kPlatShift);
mCoords.z = 0;
// Do not display objects that are on a ripped roof
if ((mt = mapList[g_vm->_currentMapNum].lookupMeta(mCoords)) != nullptr) {
if ((rt = mt->ripTable(g_vm->_currentMapNum)) != nullptr) {
if (objCoords.z >= rt->zTable[tCoords.u][tCoords.v]) {
// Disable hit-test on the object's box
_hitBox.width = -1;
_hitBox.height = -1;
obj->setOnScreen(false);
obj->setObscured(false);
return;
}
}
}
TileToScreenCoords(objCoords, _screenCoords);
drawPos.x = _screenCoords.x + fineScroll.x;
drawPos.y = _screenCoords.y + fineScroll.y;
// If it's an object, then the drawing is fairly straight
// forward.
if (isObject(obj)) {
ObjectSpriteInfo sprInfo;
// Reject any sprites which fall off the edge of the screen.
if (drawPos.x < -32
|| drawPos.x > kTileRectX + kTileRectWidth + 32
|| drawPos.y < -32
|| drawPos.y > kTileRectY + kTileRectHeight + 100) {
// Disable hit-test on the object's box
_hitBox.width = -1;
_hitBox.height = -1;
// Mark as being off screen
obj->setOnScreen(false);
obj->setObscured(false);
return;
}
if (!obj->isOnScreen()) {
SenseInfo info;
obj->setOnScreen(true);
if (getCenterActor()->canSenseSpecificObject(info, kMaxSenseRange, obj->thisID()))
obj->setSightedByCenter(true);
else {
obj->setSightedByCenter(false);
obj->setObscured(false);
}
obj->_data.sightCtr = 5;
} else {
if (--obj->_data.sightCtr == 0) {
SenseInfo info;
if (getCenterActor()->canSenseSpecificObject(info, kMaxSenseRange, obj->thisID()))
obj->setSightedByCenter(true);
else {
obj->setSightedByCenter(false);
obj->setObscured(false);
}
obj->_data.sightCtr = 5;
}
}
// Figure out which sprite to show
sprInfo = proto->getSprite(obj, ProtoObj::kObjOnGround);
// Build the color translation table for the object
obj->getColorTranslation(mainColors);
// Fill in the SpriteComponent structure
sc = &scList[0];
sc->sp = sprInfo.sp;
sc->offset.x = scList->offset.y = 0;
sc->colorTable = mainColors;
sc->flipped = sprInfo.flipped;
partCount = 1;
bodyIndex = 0;
} else {
Actor *a = (Actor *)obj;
ActorAnimation *anim;
ActorPose *pose;
int16 poseFlags;
if (!a->isDead() && objCoords.z < -proto->height - 8) {
// The actor is under water so display the bubbles sprite
drawPos.y += objCoords.z;
objCoords.z = 0;
// Disable hit-test on the object's box
_hitBox.width = -1;
_hitBox.height = -1;
// Reject any sprites which fall off the edge of the screen.
if (drawPos.x < -maxSpriteWidth
|| drawPos.x > kTileRectX + kTileRectWidth + maxSpriteWidth
|| drawPos.y < -maxSpriteBaseLine
|| drawPos.y > kTileRectY + kTileRectHeight + maxSpriteHeight) {
// Mark as being off screen
a->setOnScreen(false);
a->setObscured(false);
return;
}
buildColorTable(
mainColors,
bubbleColorTable,
ARRAYSIZE(bubbleColorTable));
if (a->_kludgeCount < 0 || ++a->_kludgeCount >= kBubbleSpriteCount)
a->_kludgeCount = 0;
sc = &scList[0];
sc->sp = spellSprites->sprite(
kBaseBubbleSpriteIndex + a->_kludgeCount);
sc->offset.x = scList->offset.y = 0;
sc->colorTable = mainColors;
sc->flipped = false;
partCount = 1;
bodyIndex = 0;
} else {
// Reject any sprites which fall off the edge of the screen.
if (drawPos.x < -maxSpriteWidth
|| drawPos.x > kTileRectX + kTileRectWidth + maxSpriteWidth
|| drawPos.y < -maxSpriteBaseLine
|| drawPos.y > kTileRectY + kTileRectHeight + maxSpriteHeight) {
// Disable hit-test on the object's box
_hitBox.width = -1;
_hitBox.height = -1;
// Mark as being off screen
a->setOnScreen(false);
a->setObscured(false);
return;
}
if (a->hasEffect(kActorInvisible)) {
if (!isPlayerActor(a)
&& !(getCenterActor()->hasEffect(kActorSeeInvis))) {
_hitBox.width = -1;
_hitBox.height = -1;
return;
}
ghostIt = true;
}
if (!a->isOnScreen()) {
SenseInfo info;
a->setOnScreen(true);
if (getCenterActor()->canSenseSpecificActor(info, kMaxSenseRange, a))
a->setSightedByCenter(true);
else {
a->setSightedByCenter(false);
a->setObscured(false);
}
a->_data.sightCtr = 5;
} else {
if (--a->_data.sightCtr == 0) {
SenseInfo info;
if (getCenterActor()->canSenseSpecificActor(info, kMaxSenseRange, a))
a->setSightedByCenter(true);
else {
a->setSightedByCenter(false);
a->setObscured(false);
}
a->_data.sightCtr = 5;
}
}
aa = a->_appearance;
if (aa == nullptr)
return;
// Fetch the animation series, and determine which
// pose in the series is the current one.
anim = aa->animation(a->_currentAnimation);
pose = aa->pose(anim, a->_currentFacing, a->_currentPose);
if (anim == nullptr)
return;
assert(anim->start[0] < 10000);
assert(anim->start[1] < 10000);
assert(anim->start[2] < 10000);
assert(pose->rightObjectOffset.x < 1000);
assert(pose->rightObjectOffset.x > -1000);
assert(pose->rightObjectOffset.y < 1000);
assert(pose->rightObjectOffset.y > -1000);
assert(pose->leftObjectOffset.x < 1000);
assert(pose->leftObjectOffset.x > -1000);
assert(pose->leftObjectOffset.y < 1000);
assert(pose->leftObjectOffset.y > -1000);
// washHandle( aa->spriteBanks[pose->actorFrameBank] );
// If the new sprite is loaded, then we can go
// ahead and show it. If it's not, then we can
// pause for a frame or two until it is loaded.
// However, if the previous frame isn't loaded
// either, then we need to go ahead and force
// the new frame to finish loaded (handled by
// lockResource())
if (aa->isBankLoaded(pose->actorFrameBank)
|| !aa->isBankLoaded(a->_poseInfo.actorFrameBank)) {
ActorPose pTemp = *pose;
// Initiate a load of the sprite bank needed.
/* if (!RHandleLoading(
(RHANDLE)(aa->spriteBanks[pose->actorFrameBank]) ))
{
aa->loadSpriteBanks( (1<<pose->actorFrameBank) );
} */
aa->requestBank(pose->actorFrameBank);
// Indicate that animation is OK.
a->_animationFlags &= ~kAnimateNotLoaded;
// Set up which bank and frame to use.
a->_poseInfo = pTemp;
} else {
// Indicate that animation isn't loaded
a->_animationFlags |= kAnimateNotLoaded;
// Initiate a load of the sprite bank needed.
/* if (!RHandleLoading(
(RHANDLE)(aa->spriteBanks[pose->actorFrameBank]) ))
{
aa->loadSpriteBanks( (1<<pose->actorFrameBank) );
}
*/
aa->requestBank(pose->actorFrameBank);
}
// For actors, start by assuming that the actor has
// nothing in either hand.
bodyIndex = 0;
rightIndex = leftIndex = -2;
partCount = 1;
poseFlags = a->_poseInfo.flags;
a->getColorTranslation(mainColors);
// Do various tests to see what the actor is
// carrying in each hand, and what drawing
// order should be used for these objects.
if (a->_leftHandObject != Nothing) {
partCount++;
if (poseFlags & ActorPose::kLeftObjectInFront) {
leftIndex = 1;
} else {
leftIndex = 0;
bodyIndex = 1;
}
}
if (a->_rightHandObject != Nothing) {
partCount++;
if (poseFlags & ActorPose::kRightObjectInFront) {
if (leftIndex == 1
&& poseFlags & ActorPose::kLeftOverRight) {
leftIndex = 2;
rightIndex = 1;
} else {
rightIndex = partCount - 1;
}
} else {
if (leftIndex == 0
&& poseFlags & ActorPose::kLeftOverRight) {
rightIndex = 0;
leftIndex = 1;
bodyIndex = 2;
} else {
rightIndex = 0;
bodyIndex++;
if (leftIndex != -2) leftIndex++;
}
}
}
// REM: Locking bug...
// ss = (SpriteSet *)RLockHandle( aa->sprites );
sprPtr = aa->_spriteBanks[a->_poseInfo.actorFrameBank];
ss = sprPtr;
if (ss == nullptr)
return;
// Fill in the SpriteComponent structure for body
sc = &scList[bodyIndex];
assert(a->_poseInfo.actorFrameIndex < ss->count);
sc->sp = ss->sprite(a->_poseInfo.actorFrameIndex);
sc->offset.x = sc->offset.y = 0;
// Color remapping info
sc->colorTable = mainColors;
// sc->colorTable = aa->schemeList ? mainColors : identityColors;
sc->flipped = (poseFlags & ActorPose::kActorFlipped);
assert(sc->sp != nullptr);
assert(sc->sp->size.x > 0);
assert(sc->sp->size.y > 0);
assert(sc->sp->size.x < 255);
assert(sc->sp->size.y < 255);
// If we were carrying something in the left hand,
// then fill in the component structure for it.
if (leftIndex >= 0) {
GameObject *ob = GameObject::objectAddress(a->_leftHandObject);
ProtoObj *prot = ob->proto();
ob->getColorTranslation(leftColors);
sc = &scList[leftIndex];
sc->sp = prot->getOrientedSprite(
ob,
a->_poseInfo.leftObjectIndex);
assert(sc->sp != nullptr);
sc->offset = a->_poseInfo.leftObjectOffset;
assert(sc->offset.x < 1000);
assert(sc->offset.x > -1000);
assert(sc->offset.y < 1000);
assert(sc->offset.y > -1000);
sc->colorTable = leftColors;
sc->flipped = (poseFlags & ActorPose::kLeftObjectFlipped);
}
// If we were carrying something in the right hand,
// then fill in the component structure for it.
if (rightIndex >= 0) {
GameObject *ob = GameObject::objectAddress(a->_rightHandObject);
ProtoObj *prot = ob->proto();
ob->getColorTranslation(rightColors);
sc = &scList[rightIndex];
sc->sp = prot->getOrientedSprite(
ob,
a->_poseInfo.rightObjectIndex);
assert(sc->sp != nullptr);
assert(sc->sp->size.x > 0);
assert(sc->sp->size.y > 0);
assert(sc->sp->size.x < 255);
assert(sc->sp->size.y < 255);
sc->offset = a->_poseInfo.rightObjectOffset;
assert(sc->offset.x < 1000);
assert(sc->offset.x > -1000);
assert(sc->offset.y < 1000);
assert(sc->offset.y > -1000);
sc->colorTable = rightColors;
sc->flipped = (poseFlags & ActorPose::kRightObjectFlipped);
}
}
}
if (!ghostIt && obj->isGhosted())
ghostIt = true;
int16 effectFlags = 0;
bool obscured;
if (ghostIt) effectFlags |= kSprFXGhosted;
if (obj->isSightedByCenter() && objRoofRipped(obj))
effectFlags |= kSprFXGhostIfObscured;
effectFlags |= kSprFXTerrainMask;
DrawCompositeMaskedSprite(
g_vm->_backPort,
scList,
partCount,
drawPos,
objCoords,
effectFlags,
&obscured);
if (effectFlags & kSprFXGhostIfObscured)
obj->setObscured(obscured);
// Record the extent box that the sprite was drawn
// at, in order to facilitate mouse picking functions
// later on in the event loop.
bodySprite = scList[bodyIndex].sp;
_hitBox.x = drawPos.x
+ (scList[bodyIndex].flipped
? -bodySprite->size.x - bodySprite->offset.x
: bodySprite->offset.x)
- fineScroll.x;
_hitBox.y = drawPos.y + bodySprite->offset.y - fineScroll.y;
_hitBox.width = bodySprite->size.x;
_hitBox.height = bodySprite->size.y;
if (_flags & kDisplayIndicator) {
Point16 indicatorCoords;
gPixelMap &indicator = *mouseCursors[kMouseCenterActorIndicatorImage];
indicatorCoords.x = _hitBox.x + fineScroll.x + (_hitBox.width - indicator._size.x) / 2;
indicatorCoords.y = _hitBox.y + fineScroll.y - indicator._size.y - 2;
TBlit(g_vm->_backPort._map, &indicator, indicatorCoords.x, indicatorCoords.y);
}
}
//-----------------------------------------------------------------------
// Do mouse hit-test on objects
ObjectID pickObject(const StaticPoint32 &mouse, StaticTilePoint &objPos) {
DisplayNode *dn;
ObjectID result = Nothing;
int32 dist = maxint32;
SpriteSet *objectSet;
objectSet = objectSprites;
if (objectSet == nullptr)
error("Object sprites have been dumped!");
for (dn = DisplayNodeList::_head; dn; dn = dn->_nextDisplayed) {
if (dn->_type == kNodeTypeObject) {
GameObject *obj = dn->_object;
if (obj->parent() == currentWorld && dn->_hitBox.ptInside(mouse.x, mouse.y)) {
TilePoint loc = obj->getLocation();
int32 newDist = loc.u + loc.v;
if (newDist < dist) {
Point16 testPoint;
SpriteSet *ss;
Sprite *spr;
ActorAppearance *aa = nullptr;
SpriteSet *sprPtr = nullptr;
bool flipped = true;
testPoint.x = mouse.x - dn->_hitBox.x;
testPoint.y = mouse.y - dn->_hitBox.y;
// If it's an object, then the test is fairly straight
// forward.
if (isObject(obj)) {
ObjectSpriteInfo sprInfo;
sprInfo = obj->proto()->getSprite(obj, ProtoObj::kObjOnGround);
spr = sprInfo.sp;
flipped = sprInfo.flipped;
} else {
Actor *a = (Actor *)obj;
aa = a->_appearance;
if (aa == nullptr) continue;
sprPtr = aa->_spriteBanks[a->_poseInfo.actorFrameBank];
ss = sprPtr;
if (ss == nullptr)
continue;
spr = ss->sprite(a->_poseInfo.actorFrameIndex);
flipped =
(a->_poseInfo.flags & ActorPose::kActorFlipped) ? 1 : 0;
}
if (GetSpritePixel(spr, flipped, testPoint)) {
dist = newDist;
result = obj->thisID();
objPos.set(loc.u, loc.v, loc.z);
objPos.z += MAX(-spr->offset.y - testPoint.y, 0);
} else if (result == Nothing) { // If no object found yet
Point16 testPoint2;
int16 minX, maxX;
// Try checking a wider area for mouse hit
testPoint2.y = testPoint.y;
minX = MAX(0, testPoint.x - 6);
maxX = MIN(dn->_hitBox.width - 1, testPoint.x + 6);
// scan a horizontal strip of the character for a hit.
// If we find a hit, go ahead and set result anyway
// If we later find a real hit, then it will overwrite
// the results of this one.
for (testPoint2.x = minX; testPoint2.x <= maxX; testPoint2.x++) {
if (GetSpritePixel(spr, flipped, testPoint2)) {
result = obj->thisID();
objPos.set(loc.u, loc.v, loc.z);
objPos.z += MAX(-spr->offset.y - testPoint.y, 0);
break;
}
}
}
}
}
}
}
return result;
}
//-----------------------------------------------------------------------
// Adds spell effects into the dispplay list
//
// NOTE : all spell effects are currently placed behind any real stuff
// they can also easily be placed in front
void DisplayNodeList::buildEffects(bool) {
if (_count) {
for (int i = 0; i < _count; i++) {
DisplayNode *dn = DisplayNodeList::_head;
if (_displayList[i]._efx->isHidden() || _displayList[i]._efx->isDead())
continue;
// make sure it knows it's not a real object
_displayList[i]._type = kNodeTypeEffect;
_displayList[i]._sortDepth = _displayList[i]._efx->_screenCoords.y + _displayList[i]._efx->_current.z / 2;
if (dn) {
int32 sd = _displayList[i]._sortDepth;
while (dn->_nextDisplayed && dn->_nextDisplayed->_sortDepth <= sd)
dn = dn->_nextDisplayed;
}
if (dn == DisplayNodeList::_head) {
_displayList[i]._nextDisplayed = DisplayNodeList::_head;
DisplayNodeList::_head = &_displayList[i];
} else {
_displayList[i]._nextDisplayed = dn->_nextDisplayed;
dn->_nextDisplayed = &_displayList[i];
}
}
}
}
bool DisplayNodeList::dissipated() {
if (_count) {
for (int i = 0; i < _count; i++) {
if (_displayList[i]._efx && !_displayList[i]._efx->isDead())
return false;
}
}
return true;
}
//-----------------------------------------------------------------------
// Draw sprites for spell effects
//
// NOTE : all spell effects currently use the center actor for their
// sprites.
void DisplayNode::drawEffect() {
if (_efx) _efx->drawEffect();
}
void Effectron::drawEffect() {
ColorTable eColors; // colors for object
bool obscured = false;
Point16 drawPos;
TilePoint objCoords = SpellPos();
SpriteComponent scList[3],
*sc;
if (isHidden() || isDead())
return;
drawPos.x = _screenCoords.x + fineScroll.x;
drawPos.y = _screenCoords.y + fineScroll.y;
// Reject any sprites which fall off the edge of the screen.
if (drawPos.x < -32
|| drawPos.x > kTileRectX + kTileRectWidth + 32
|| drawPos.y < -32
|| drawPos.y > kTileRectY + kTileRectHeight + 100) {
// Disable hit-test on the object's box
_hitBox.width = -1;
_hitBox.height = -1;
return;
}
TileToScreenCoords(objCoords, _screenCoords);
sc = &scList[0];
//sc->sp = (*spellSprites)->sprite( spriteID() );
sc->sp = spellSprites->sprite(spriteID()); //tempSpellSpriteIDs[rand()%39] );
sc->offset.x = scList->offset.y = 0;
(*g_vm->_sdpList)[_parent->_spell]->getColorTranslation(eColors, this);
sc->colorTable = eColors;
sc->flipped = false;
obscured = (visiblePixelsInSprite(sc->sp,
sc->flipped,
sc->colorTable,
drawPos,
_current,
0) <= 5);
DrawCompositeMaskedSprite(
g_vm->_backPort,
scList,
1,
drawPos,
objCoords,
((obscured) && //objectFlags & GameObject::kObjectObscured ) &&
0
? kSprFXGhosted : kSprFXTerrainMask));
}
/* ===================================================================== *
Misc. functions
* ===================================================================== */
//-----------------------------------------------------------------------
// Enable or disable the center actor indicator
void setCenterActorIndicator(bool enabled) {
centerActorIndicatorEnabled = enabled;
}
} // end of namespace Saga2