scummvm/engines/tony/loc.h
2021-12-26 18:48:43 +01:00

572 lines
12 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/>.
*
*/
/*
* This code is based on original Tony Tough source code
*
* Copyright (c) 1997-2003 Nayma Software
*/
#ifndef TONY_LOC_H
#define TONY_LOC_H
#include "common/scummsys.h"
#include "common/mutex.h"
#include "common/file.h"
#include "tony/sound.h"
#include "tony/utils.h"
namespace Tony {
/****************************************************************************\
* Various defines
\****************************************************************************/
/**
* Valid color modes
*/
typedef enum {
CM_256,
CM_65K
} RMColorMode;
/****************************************************************************\
* Class declarations
\****************************************************************************/
/**
* Generic palette
*/
class RMPalette {
public:
byte _data[1024];
public:
void readFromStream(Common::ReadStream &ds);
};
/**
* Sound effect of an object
*/
class RMSfx {
public:
Common::String _name;
FPSfx *_fx;
bool _bPlayingLoop;
public:
RMSfx();
virtual ~RMSfx();
void play(bool bLoop = false);
void setVolume(int vol);
void pause(bool bPause);
void stop();
void readFromStream(Common::ReadStream &ds, bool bLOX = false);
};
/**
* Object pattern
*/
class RMPattern {
public:
// Type of slot
enum RMSlotType {
DUMMY1 = 0,
DUMMY2,
SPRITE,
SOUND,
COMMAND,
SPECIAL
};
// Class slot
class RMSlot {
private:
RMPoint _pos; // Child co-ordinates
public:
RMSlotType _type;
int _data;
byte _flag;
public:
RMPoint pos() {
return _pos;
}
void readFromStream(Common::ReadStream &ds, bool bLOX = false);
};
public:
Common::String _name;
private:
int _speed;
RMPoint _pos; // Parent coordinates
RMPoint _curPos; // Parent + child coordinates
int _bLoop;
int _nSlots;
int _nCurSlot;
int _nCurSprite;
RMSlot *_slots;
uint32 _nStartTime;
public:
RMPattern();
virtual ~RMPattern();
// A warning that the pattern now and the current
int init(RMSfx *sfx, bool bPlayP0 = false, byte *bFlag = NULL);
// Update the pattern, checking to see if it's time to change slot and executing
// any associated commands
int update(uint32 hEndPattern, byte &bFlag, RMSfx *sfx);
// Stop a sound effect
void stopSfx(RMSfx *sfx);
// Reads the position of the pattern
RMPoint pos();
void readFromStream(Common::ReadStream &ds, bool bLOX);
private:
void updateCoord();
};
/**
* Sprite (frame) animation of an item
*/
class RMSprite : public RMGfxTask {
public:
Common::String _name;
RMRect _rcBox;
protected:
RMGfxSourceBuffer *_buf;
public:
RMSprite();
~RMSprite() override;
void init(RMGfxSourceBuffer *buf);
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
void setPalette(byte *lpBuf);
void getSizeFromStream(Common::SeekableReadStream &ds, int *dimx, int *dimy);
void LOXGetSizeFromStream(Common::SeekableReadStream &ds, int *dimx, int *dimy);
void readFromStream(Common::SeekableReadStream &ds, bool bLOX = false);
};
/**
* Data on an item
*/
class RMItem : public RMGfxTask {
public:
Common::String _name;
protected:
int _z;
RMPoint _pos; // Coordinate ancestor
RMColorMode _cm;
RMPoint _curScroll;
byte _FX;
byte _FXparm;
virtual int getCurPattern();
private:
int _nCurPattern;
int _mpalCode;
RMPoint _hot;
RMRect _rcBox;
int _nSprites, _nSfx, _nPatterns;
byte _bPal;
RMPalette _pal;
RMSprite *_sprites;
RMSfx *_sfx;
RMPattern *_patterns;
byte _bCurFlag;
int _nCurSprite;
bool _bIsActive;
uint32 _hEndPattern;
bool _bInitCurPattern;
public:
RMPoint calculatePos();
public:
RMItem();
~RMItem() override;
// Process to make the object move on any animations.
// Returns TRUE if it should be redrawn on the next frame
bool doFrame(RMGfxTargetBuffer *bigBuf, bool bAddToList = true);
// Sets the current scrolling position
void setScrollPosition(const RMPoint &scroll);
// Overloading of check whether to remove from active list
void removeThis(CORO_PARAM, bool &result) override;
// Overloaded Draw
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
// Overloaded priority: it's based on Z ordering
int priority() override;
// Pattern number
int numPattern();
// Set anew animation pattern, changing abruptly from the current
virtual void setPattern(int nPattern, bool bPlayP0 = false);
// Set a new status
void setStatus(int nStatus);
bool isIn(const RMPoint &pt, int *size = NULL);
RMPoint getHotspot();
bool getName(Common::String &name);
int mpalCode();
// Unload
void unload();
// Wait for the end of the current pattern
void waitForEndPattern(CORO_PARAM, uint32 hCustomSkip = CORO_INVALID_PID_VALUE);
// Sets a new hotspot fro the object
void changeHotspot(const RMPoint &pt);
void setInitCurPattern(bool status);
void playSfx(int nSfx);
void readFromStream(Common::SeekableReadStream &ds, bool bLOX = false);
void pauseSound(bool bPause);
protected:
// Create a primitive that has as it's task this item
virtual RMGfxPrimitive *newItemPrimitive();
// Allocate memory for the sprites
virtual RMGfxSourceBuffer *newItemSpriteBuffer(int dimx, int dimy, bool bPreRLE);
};
#define MAXBOXES 50 // Maximum number of allowed boxes
#define MAXHOTSPOT 20 // Maximum nimber of allowed hotspots
class RMBox {
public:
struct Hotspot {
int _hotx, _hoty; // Hotspot coordinates
int _destination; // Hotspot destination
};
public:
int _left, _top, _right, _bottom; // Vertici bounding boxes
int _adj[MAXBOXES]; // List of adjacent bounding boxes
int _numHotspot; // Hotspot number
uint8 _destZ; // Z value for the bounding box
Hotspot _hotspot[MAXHOTSPOT]; // List of hotspots
bool _bActive;
bool _bReversed;
void readFromStream(Common::ReadStream &ds);
};
class RMBoxLoc {
public:
int _numbBox;
RMBox *_boxes;
void readFromStream(Common::ReadStream &ds);
public:
RMBoxLoc();
virtual ~RMBoxLoc();
void recalcAllAdj();
};
#define GAME_BOXES_SIZE 200
class RMGameBoxes {
protected:
RMBoxLoc *_allBoxes[GAME_BOXES_SIZE];
int _nLocBoxes;
public:
RMGameBoxes();
~RMGameBoxes();
void init();
void close();
// Get binding boxes for a given location
RMBoxLoc *getBoxes(int nLoc);
int getLocBoxesCount() const;
// Return the box which contains a given point
int whichBox(int nLoc, const RMPoint &pt);
// Check whether a point is inside a given box
bool isInBox(int nLoc, int nBox, const RMPoint &pt);
// Change the status of a box
void changeBoxStatus(int nLoc, int nBox, int status);
// Save state handling
int getSaveStateSize();
void saveState(byte *buf);
void loadState(byte *buf);
};
class RMCharacter : protected RMItem {
public:
enum Patterns {
PAT_STANDUP = 1,
PAT_STANDDOWN,
PAT_STANDLEFT,
PAT_STANDRIGHT,
PAT_WALKUP,
PAT_WALKDOWN,
PAT_WALKLEFT,
PAT_WALKRIGHT
};
private:
enum CharacterStatus {
STAND,
WALK
};
signed short _walkCount;
int _dx, _dy, _olddx, _olddy;
float _fx, _fy, _slope;
RMPoint _lineStart, _lineEnd, _pathEnd;
signed char _walkSpeed, _walkStatus;
char _minPath;
short _nextBox;
short _path[MAXBOXES];
short _pathLength, _pathCount;
int _curBox;
CharacterStatus _status;
int _curSpeed;
bool _bEndOfPath;
uint32 _hEndOfPath;
Common::Mutex _csMove;
int _curLocation;
bool _bRemoveFromOT;
bool _bMovingWithoutMinpath;
RMGameBoxes *_theBoxes;
RMPoint _fixedScroll;
private:
int inWhichBox(const RMPoint &pt);
bool findPath(short source, short destination);
RMPoint searching(char UP, char DOWN, char RIGHT, char LEFT, RMPoint point);
RMPoint nearestPoint(const RMPoint &punto);
void goTo(CORO_PARAM, RMPoint destcoord, bool bReversed = false);
short scanLine(const RMPoint &point);
RMPoint invScanLine(const RMPoint &point);
RMPoint nearestHotSpot(int sourcebox, int destbox);
void newBoxEntered(int nBox);
protected:
bool _bMoving;
bool _bDrawNow;
bool _bNeedToStop;
public:
RMCharacter();
~RMCharacter() override;
void linkToBoxes(RMGameBoxes *theBoxes);
void removeThis(CORO_PARAM, bool &result) override;
// Update the position of a character
void doFrame(CORO_PARAM, RMGfxTargetBuffer *bigBuf, int loc);
// Overloaded draw
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
// TRUE if you just stopped
bool endOfPath();
// Change the pattern of a character to STOP
virtual void stop(CORO_PARAM);
// Check if the character is moving
bool isMoving();
// Move the character to a certain position
void move(CORO_PARAM, RMPoint pt, bool *result = NULL);
// Place the character in a certain position WITHOUT moving
void setPosition(const RMPoint &pt, int newloc = -1);
// Wait for the end of movement
void waitForEndMovement(CORO_PARAM);
void setFixedScroll(const RMPoint &fix);
void setSpeed(int speed);
};
class RMWipe : public RMGfxTask {
private:
bool _bFading;
bool _bEndFade;
bool _bUnregister;
uint32 _hUnregistered;
int _nFadeStep;
uint32 _hEndOfFade;
bool _bMustRegister;
RMItem _wip0r;
public:
RMWipe();
~RMWipe() override;
void doFrame(RMGfxTargetBuffer &bigBuf);
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
void initFade(int type);
void closeFade();
void waitForFadeEnd(CORO_PARAM);
void unregister() override;
void removeThis(CORO_PARAM, bool &result) override;
int priority() override;
};
/**
* Location
*/
class RMLocation : public RMGfxTaskSetPrior {
public:
Common::String _name; // Name
private:
RMColorMode _cmode; // Color mode
RMGfxSourceBuffer *_buf; // Location picture
int _nItems; // Number of objects
RMItem *_items; // Objects
RMPoint _curScroll; // Current scroll position
RMPoint _fixedScroll;
RMPoint _prevScroll; // Previous scroll position
RMPoint _prevFixedScroll;
public:
// @@@@@@@@@@@@@@@@@@@@@@@
RMPoint TEMPTonyStart;
RMPoint TEMPGetTonyStart();
int TEMPNumLoc;
int TEMPGetNumLoc();
public:
RMLocation();
~RMLocation() override;
// Load variations
bool load(Common::SeekableReadStream &ds);
bool loadLOX(Common::SeekableReadStream &ds);
// Unload
void unload();
// Overloaded draw
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
// Prepare a frame by drawing the location and all it's items
void doFrame(RMGfxTargetBuffer *bigBuf);
// Return the item at a given point
RMItem *whichItemIsIn(const RMPoint &pt);
// Return the item based on it's MPAL code
RMItem *getItemFromCode(uint32 dwCode);
// Set the current scroll position
void setScrollPosition(const RMPoint &scroll);
// Sets an additinal offset for scrolling
void setFixedScroll(const RMPoint &scroll);
// Update the scrolling coordinates to display the specified point
void updateScrolling(const RMPoint &ptShowThis);
// Read the current scroll position
RMPoint scrollPosition();
// Pause sound
void pauseSound(bool bPause);
};
/**
* MPAL message, composed of more ASCIIZ
*/
class RMMessage {
private:
char *_lpMessage;
char *_lpPeriods[256];
int _nPeriods;
private:
void parseMessage();
public:
RMMessage();
RMMessage(uint32 dwId);
virtual ~RMMessage();
void load(uint32 dwId);
bool isValid();
int numPeriods();
char *period(int num);
char *operator[](int num);
};
} // End of namespace Tony
#endif /* TONY_H */