/* 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 .
*
*/
#ifndef SCUMM_ACTOR_H
#define SCUMM_ACTOR_H
#include "common/scummsys.h"
#include "common/serializer.h"
#include "scumm/scumm.h"
namespace Scumm {
enum {
V12_X_MULTIPLIER = 8,
V12_Y_MULTIPLIER = 2,
V12_X_SHIFT = 3,
V12_Y_SHIFT = 1
};
enum MoveFlags {
MF_NEW_LEG = 1,
MF_IN_LEG = 2,
MF_TURN = 4,
MF_LAST_LEG = 8,
MF_FROZEN = 0x80
};
struct CostumeData {
byte active[16];
uint16 animCounter;
byte soundCounter;
byte soundPos;
uint16 stopped;
uint16 curpos[16];
uint16 start[16];
uint16 end[16];
uint16 frame[16];
/* HE specific */
uint16 heJumpOffsetTable[16];
uint16 heJumpCountTable[16];
uint32 heCondMaskTable[16];
void reset() {
stopped = 0;
for (int i = 0; i < 16; i++) {
active[i] = 0;
curpos[i] = start[i] = end[i] = frame[i] = 0xFFFF;
}
}
};
struct AdjustBoxResult { /* Result type of AdjustBox functions */
int16 x, y;
byte box;
};
enum {
kOldInvalidBox = 255, // For small header games
kNewInavlidBox = 0
};
class Actor : public Common::Serializable {
public:
static byte kInvalidBox;
protected:
ScummEngine *_vm;
/** The position of the actor inside the virtual screen. */
Common::Point _pos;
public:
int _top, _bottom;
uint _width;
byte _number;
uint16 _costume;
byte _room;
public:
byte _talkColor;
int _talkFrequency;
byte _talkPan;
byte _talkVolume;
uint16 _boxscale;
byte _scalex, _scaley;
byte _charset;
byte _moving;
bool _ignoreBoxes;
byte _forceClip;
byte _initFrame;
byte _walkFrame;
byte _standFrame;
byte _talkStartFrame;
byte _talkStopFrame;
bool _needRedraw, _needBgReset, _visible;
byte _shadowMode;
bool _flip;
byte _frame;
byte _walkbox;
int16 _talkPosX, _talkPosY;
uint16 _talkScript, _walkScript;
bool _ignoreTurns;
bool _drawToBackBuf;
int32 _layer;
uint16 _sound[32];
CostumeData _cost;
/* HE specific */
int _heOffsX, _heOffsY;
bool _heSkipLimbs;
uint32 _heCondMask;
uint32 _hePaletteNum;
uint32 _heXmapNum;
protected:
struct ActorWalkData {
Common::Point dest; // Final destination point
byte destbox; // Final destination box
int16 destdir; // Final destination, direction to face at
Common::Point cur; // Last position
byte curbox; // Last box
Common::Point next; // Next position on our way to the destination, i.e. our intermediate destination
Common::Point point3;
int32 deltaXFactor, deltaYFactor;
uint16 xfrac, yfrac;
uint16 xAdd, yAdd;
void reset() {
dest.x = dest.y = 0;
destbox = 0;
destdir = 0;
cur.x = cur.y = 0;
curbox = 0;
next.x = next.y = 0;
point3.x = point3.y = 0;
deltaXFactor = 0;
deltaYFactor = 0;
xfrac = 0;
yfrac = 0;
xAdd = 0;
yAdd = 0;
}
};
uint16 _palette[256];
int _elevation;
uint16 _facing;
uint16 _targetFacing;
uint _speedx, _speedy;
byte _animProgress, _animSpeed;
bool _costumeNeedsInit;
ActorWalkData _walkdata;
int16 _animVariable[27];
public:
Actor(ScummEngine *scumm, int id);
~Actor() override {}
//protected:
virtual void hideActor();
void showActor();
virtual void initActor(int mode);
void putActor() {
putActor(_pos.x, _pos.y, _room);
}
void putActor(int room) {
putActor(_pos.x, _pos.y, room);
}
void putActor(int x, int y) {
putActor(x, y, _room);
}
void putActor(int x, int y, int room);
void setActorWalkSpeed(uint newSpeedX, uint newSpeedY);
protected:
int calcMovementFactor(const Common::Point& next);
int actorWalkStep();
int remapDirection(int dir, bool is_walking);
virtual void setupActorScale();
void setBox(int box);
int updateActorDirection(bool is_walking);
public:
void adjustActorPos();
virtual AdjustBoxResult adjustXYToBeInBox(int dstX, int dstY);
virtual void setDirection(int direction);
void faceToObject(int obj);
void turnToDirection(int newdir);
virtual void walkActor();
void drawActorCostume(bool hitTestMode = false);
virtual void prepareDrawActorCostume(BaseCostumeRenderer *bcr);
virtual void animateCostume();
virtual void setActorCostume(int c);
void animateLimb(int limb, int f);
bool actorHitTest(int x, int y);
const byte *getActorName();
void startWalkActor(int x, int y, int dir);
void stopActorMoving();
protected:
void startWalkAnim(int cmd, int angle);
public:
void runActorTalkScript(int f);
virtual void startAnimActor(int frame);
void remapActorPalette(int r_fact, int g_fact, int b_fact, int threshold);
void remapActorPaletteColor(int slot, int color);
void animateActor(int anim);
bool isInCurrentRoom() const {
return _room == _vm->_currentRoom;
}
Common::Point getPos() const {
Common::Point p(_pos);
if (_vm->_game.version <= 2) {
p.x *= V12_X_MULTIPLIER;
p.y *= V12_Y_MULTIPLIER;
}
return p;
}
const Common::Point& getRealPos() const {
return _pos;
}
int getRoom() const {
return _room;
}
int getFacing() const {
return _facing;
}
void setFacing(int newFacing) {
_facing = newFacing;
}
int getAnimVar(byte var) const;
void setAnimVar(byte var, int value);
void setAnimSpeed(byte newAnimSpeed) {
_animSpeed = newAnimSpeed;
_animProgress = 0;
}
int getAnimSpeed() const {
return _animSpeed;
}
int getAnimProgress() const {
return _animProgress;
}
int getElevation() const {
return _elevation;
}
void setElevation(int newElevation) {
if (_elevation != newElevation) {
_elevation = newElevation;
_needRedraw = true;
}
}
void setPalette(int idx, int val) {
_palette[idx] = val;
_needRedraw = true;
}
void setScale(int sx, int sy) {
if (sx != -1)
_scalex = sx;
if (sy != -1)
_scaley = sy;
_needRedraw = true;
}
void classChanged(int cls, bool value);
void saveLoadWithSerializer(Common::Serializer &ser) override;
protected:
bool isInClass(int cls);
virtual bool isPlayer();
bool findPathTowards(byte box, byte box2, byte box3, Common::Point &foundPath);
};
class Actor_v3 : public Actor {
public:
Actor_v3(ScummEngine *scumm, int id) : Actor(scumm, id), _stepX(1), _stepThreshold(0), _facingXYratio(scumm->_game.version == 3 ? 3 : 1) {}
void initActor(int mode) override;
void walkActor() override;
void saveLoadWithSerializer(Common::Serializer &ser) override;
protected:
int calcMovementFactor(const Common::Point& next);
int actorWalkStep();
void setupActorScale() override;
void findPathTowardsOld(byte box, byte box2, byte box3, Common::Point &p2, Common::Point &p3);
private:
uint _stepX, _stepThreshold;
const int _facingXYratio;
};
class Actor_v2 : public Actor_v3 {
public:
Actor_v2(ScummEngine *scumm, int id) : Actor_v3(scumm, id) {}
void initActor(int mode) override;
void walkActor() override;
AdjustBoxResult adjustXYToBeInBox(int dstX, int dstY) override;
protected:
bool isPlayer() override;
void prepareDrawActorCostume(BaseCostumeRenderer *bcr) override;
};
enum ActorV0MiscFlags {
kActorMiscFlagStrong = 0x01, // Kid is strong (Hunk-O-Matic used)
kActorMiscFlagGTFriend = 0x02, // Kid is green tentacle's friend (recording contract)
kActorMiscFlagWatchedTV = 0x04, // Kid knows publisher's address (watched TV)
kActorMiscFlagEdsEnemy = 0x08, // Kid is not Weird Ed's friend
kActorMiscFlag_10 = 0x10, // ???
kActorMiscFlag_20 = 0x20, // ???
kActorMiscFlagFreeze = 0x40, // Stop moving
kActorMiscFlagHide = 0x80 // Kid is invisible (dead or in radiation suit)
};
class Actor_v0 : public Actor_v2 {
public:
Common::Point _CurrentWalkTo, _NewWalkTo;
Common::Array _walkboxHistory;
byte _walkboxQueue[0x10];
byte _walkboxQueueIndex;
byte _costCommandNew;
byte _costCommand;
byte _miscflags;
byte _speaking;
byte _walkCountModulo;
bool _newWalkBoxEntered;
byte _walkDirX;
byte _walkDirY;
byte _walkYCountGreaterThanXCount;
byte _walkXCount;
byte _walkXCountInc;
byte _walkYCount;
byte _walkYCountInc;
byte _walkMaxXYCountInc;
Common::Point _tmp_Pos;
Common::Point _tmp_NewPos;
byte _tmp_WalkBox;
bool _tmp_NewWalkBoxEntered;
int8 _animFrameRepeat;
int8 _limbFrameRepeatNew[8];
int8 _limbFrameRepeat[8];
bool _limb_flipped[8];
private:
bool walkBoxQueueAdd(int box);
bool walkBoxQueueFind(int box);
void walkboxQueueReverse();
public:
Actor_v0(ScummEngine *scumm, int id) : Actor_v2(scumm, id) {}
void initActor(int mode) override;
void animateActor(int anim);
void animateCostume() override;
void limbFrameCheck(int limb);
void directionUpdate();
void speakCheck();
void setDirection(int direction) override;
void startAnimActor(int f) override;
bool calcWalkDistances();
void walkActor() override;
void actorSetWalkTo();
byte actorWalkXCalculate();
byte actorWalkYCalculate();
byte updateWalkbox();
void walkBoxQueueReset();
bool walkBoxQueuePrepare();
AdjustBoxResult adjustXYToBeInBox(int dstX, int dstY) override;
AdjustBoxResult adjustPosInBorderWalkbox(AdjustBoxResult box);
void setActorToTempPosition();
void setActorToOriginalPosition();
void saveLoadWithSerializer(Common::Serializer &ser) override;
};
} // End of namespace Scumm
#endif