scummvm/engines/voyeur/files.h

636 lines
14 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
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
*/
#ifndef VOYEUR_FILES_H
#define VOYEUR_FILES_H
#include "common/scummsys.h"
#include "common/file.h"
#include "common/rect.h"
#include "common/str.h"
#include "voyeur/graphics.h"
namespace Voyeur {
class VoyeurEngine;
class BoltFile;
class BoltGroup;
class BoltEntry;
class RectResource;
class PictureResource;
class ViewPortResource;
class ViewPortListResource;
class FontResource;
class CMapResource;
class VInitCycleResource;
class PtrResource;
class ControlResource;
class ThreadResource;
#define DECOMPRESS_SIZE 0x7000
class ResolveEntry {
public:
uint32 _id;
byte **_p;
ResolveEntry(uint32 id, byte **p) { _id = id; _p = p; }
};
class BoltFilesState {
public:
VoyeurEngine *_vm;
BoltFile *_curLibPtr;
BoltGroup *_curGroupPtr;
BoltEntry *_curMemberPtr;
byte *_curMemInfoPtr;
int _fromGroupFlag;
byte _xorMask;
bool _encrypt;
int _curFilePosition;
int _bufferEnd;
int _bufferBegin;
int _bytesLeft;
int _bufSize;
byte *_bufStart;
byte *_bufPos;
byte _decompressBuf[DECOMPRESS_SIZE];
int _historyIndex;
byte _historyBuffer[0x200];
int _runLength;
int _decompState;
int _runType;
int _runValue;
int _runOffset;
Common::File *_curFd;
Common::Array<ResolveEntry> _resolves;
byte *_boltPageFrame;
int _sImageShift;
bool _SVGAReset;
public:
BoltFilesState();
byte *decompress(byte *buf, int size, int mode);
void nextBlock();
void EMSGetFrameAddr(byte **pageFrame) {} // TODO: Maybe?
bool EMSAllocatePages(uint *planeSize) { return false; } // TODO: Maybe?
void EMSMapPageHandle(int planeSize, int idx1, int idx2) {} // TODO: Maybe?
};
class BoltFile {
private:
Common::Array<BoltGroup> _groups;
protected:
BoltFilesState &_state;
virtual void initResource(int resType) = 0;
void initDefault();
private:
void resolveAll();
byte *getBoltMember(uint32 id);
void termType() {} // TODO
void initMem(int id) {} // TODO
void termMem() {} // TODO
void initGro() {} // TODO
void termGro() {} // TODO
public:
Common::File _file;
public:
BoltFile(const Common::String &filename, BoltFilesState &state);
virtual ~BoltFile();
BoltGroup *getBoltGroup(uint16 id, bool process = true);
void freeBoltGroup(uint16 id, bool freeEntries = true);
void freeBoltMember(uint32 id);
byte *memberAddr(uint32 id);
byte *memberAddrOffset(uint32 id);
void resolveIt(uint32 id, byte **p);
void resolveFunction(uint32 id, GraphicMethodPtr *fn);
BoltEntry &boltEntry(uint16 id);
BoltEntry &getBoltEntryFromLong(uint32 id);
PictureResource *getPictureResource(uint32 id);
CMapResource *getCMapResource(uint32 id);
};
class BVoyBoltFile: public BoltFile {
private:
// initType method table
void sInitRect();
void sInitPic();
void vInitCMap();
void vInitCycl();
void initViewPort();
void initViewPortList();
void initFontInfo();
void initFont();
void initSoundMap();
protected:
virtual void initResource(int resType);
public:
BVoyBoltFile(BoltFilesState &state);
};
class StampBoltFile: public BoltFile {
private:
void initThread();
void initPtr();
void initControl();
protected:
virtual void initResource(int resType);
public:
StampBoltFile(BoltFilesState &state);
};
class BoltGroup {
private:
Common::SeekableReadStream *_file;
public:
byte _loaded;
bool _processed;
bool _callInitGro;
int _termGroIndex;
int _count;
int _fileOffset;
Common::Array<BoltEntry> _entries;
public:
BoltGroup(Common::SeekableReadStream *f);
virtual ~BoltGroup();
void load(uint16 groupId);
void unload();
};
class BoltEntry {
private:
Common::SeekableReadStream *_file;
public:
uint16 _id;
byte _mode;
byte _field1;
byte _initMethod;
int _fileOffset;
byte _xorMask;
int _size;
byte *_data;
// bvoy.blt resource types
RectResource *_rectResource;
PictureResource *_picResource;
ViewPortResource *_viewPortResource;
ViewPortListResource *_viewPortListResource;
FontResource *_fontResource;
FontInfoResource *_fontInfoResource;
CMapResource *_cMapResource;
VInitCycleResource *_vInitCycleResource;
// stampblt.blt resource types
PtrResource *_ptrResource;
ControlResource *_controlResource;
ThreadResource *_threadResource;
public:
BoltEntry(Common::SeekableReadStream *f, uint16 id);
virtual ~BoltEntry();
void load();
bool hasResource() const;
};
class FilesManager {
private:
int _decompressSize;
public:
BoltFilesState _boltFilesState;
BoltFile *_curLibPtr;
public:
FilesManager();
void setVm(VoyeurEngine *vm) { _boltFilesState._vm = vm; }
bool openBoltLib(const Common::String &filename, BoltFile *&boltFile);
byte *fload(const Common::String &filename, int *size = NULL);
};
class RectEntry: public Common::Rect {
public:
int _arrIndex;
int _count;
RectEntry(int x1, int y1, int x2, int y2, int arrIndex, int count);
};
class RectResource: public Common::Rect {
public:
Common::Array<RectEntry> _entries;
public:
RectResource(const byte *src, int size, bool isExtendedRects);
RectResource(int xp, int yp, int width, int height);
virtual ~RectResource() {}
};
enum DisplayFlag { DISPFLAG_1 = 1, DISPFLAG_2 = 2, DISPFLAG_4 = 4, DISPFLAG_8 = 8,
DISPFLAG_10 = 0x10, DISPFLAG_20 = 0x20, DISPFLAG_40 = 0x40, DISPFLAG_80 = 0x80,
DISPFLAG_100 = 0x100, DISPFLAG_200 = 0x200, DISPFLAG_400 = 0x400,
DISPFLAG_800 = 0x800, DISPFLAG_1000 = 0x1000, DISPFLAG_2000 = 0x2000,
DISPFLAG_4000 = 0x4000, DISPFLAG_VIEWPORT = 0x8000, DISPFLAG_CURSOR = 0x10000 };
class DisplayResource {
private:
VoyeurEngine *_vm;
public:
uint32 _flags;
public:
DisplayResource();
DisplayResource(VoyeurEngine *vm);
/**
* Fill a box of the given size at the current _drawPtr location
*/
void sFillBox(int width, int height);
/**
* Draw text at the current pen position
*/
int drawText(const Common::String &msg);
/**
* Return the width of a given text in the current font
*/
int textWidth(const Common::String &msg);
/**
* Clip the given rectangle by the currently viewable area
*/
bool clipRect(Common::Rect &rect);
};
/* bvoy.blt resource types */
enum PictureFlag { PICFLAG_2 = 2, PICFLAG_PIC_OFFSET = 8, PICFLAG_CLEAR_SCREEN = 0x10,
PICFLAG_20 = 0x20, PICFLAG_HFLIP = 0x40, PICFLAG_VFLIP = 0x80, PICFLAG_100 = 0x100,
PICFLAG_CLEAR_SCREEN00 = 0x1000
};
class PictureResource: public DisplayResource {
private:
/**
* Flip the image data horizontally
*/
void flipHorizontal(const byte *data);
/**
* Flip the image data vertically
*/
void flipVertical(const byte *data);
public:
byte _select;
byte _pick;
byte _onOff;
byte _depth;
Common::Rect _bounds;
uint32 _maskData;
uint _planeSize;
/**
* Image data for the picture
*/
byte *_imgData;
/**
* Flag to indicate whether to free the image data
*/
DisposeAfterUse::Flag _freeImgData;
public:
PictureResource(BoltFilesState &state, const byte *src);
PictureResource(int flags, int select, int pick, int onOff, int depth,
const Common::Rect &bounds, int maskData, byte *imgData, int planeSize);
PictureResource(Graphics::Surface *surface);
PictureResource();
virtual ~PictureResource();
};
typedef void (ViewPortResource::*ViewPortMethodPtr)();
class ViewPortResource: public DisplayResource {
private:
BoltFilesState &_state;
private:
void setupViewPort(PictureResource *page, Common::Rect *clipRect, ViewPortSetupPtr setupFn,
ViewPortAddPtr addFn, ViewPortRestorePtr restoreFn);
public:
ViewPortResource *_parent;
int _pageCount;
int _pageIndex;
int _lastPage;
Common::Rect _bounds;
int _field18;
PictureResource *_currentPic;
PictureResource *_activePage;
PictureResource *_pages[2];
byte *_field30;
// Rect lists and counts. Note that _rectListCount values of '-1' seem to have
// special significance, which is why I'm not making them redundant in favour
// of the arrays' .size() method
Common::Array<Common::Rect> *_rectListPtr[3];
int _rectListCount[3];
Common::Rect _clipRect;
byte *_field7A;
GraphicMethodPtr _fn1;
ViewPortSetupPtr _setupFn;
ViewPortAddPtr _addFn;
ViewPortRestorePtr _restoreFn;
Common::Rect _fontRect;
public:
ViewPortResource(BoltFilesState &state, const byte *src);
virtual ~ViewPortResource();
void setupViewPort();
void setupViewPort(PictureResource *pic, Common::Rect *clipRect = NULL);
void addSaveRect(int pageIndex, const Common::Rect &r);
void fillPic(byte onOff = 0);
void drawIfaceTime();
void drawPicPerm(PictureResource *pic, const Common::Point &pt);
};
class ViewPortPalEntry {
public:
uint16 _rEntry, _gEntry, _bEntry;
uint16 _rChange, _gChange, _bChange;
uint16 _palIndex;
public:
ViewPortPalEntry(const byte *src);
};
class ViewPortListResource {
public:
Common::Array<ViewPortPalEntry> _palette;
Common::Array<ViewPortResource *> _entries;
int _palIndex;
ViewPortListResource(BoltFilesState &state, const byte *src);
virtual ~ViewPortListResource() {}
};
class FontResource {
public:
int _minChar, _maxChar;
int field2;
int _padding;
int _fontHeight;
int _topPadding;
int *_charWidth;
byte *_charOffsets;
byte *_charImages;
FontResource(BoltFilesState &state, byte *src);
virtual ~FontResource();
};
enum FontJustify { ALIGN_LEFT = 0, ALIGN_CENTRE = 1, ALIGN_RIGHT = 2 };
class FontInfoResource {
public:
FontResource *_curFont;
byte _picFlags;
byte _picSelect;
byte _picPick;
byte _picOnOff;
byte _fontFlags;
FontJustify _justify;
int _fontSaveBack;
Common::Point _pos;
int _justifyWidth;
int _justifyHeight;
Common::Point _shadow;
int _foreColor;
int _backColor;
int _shadowColor;
public:
FontInfoResource(BoltFilesState &state, const byte *src);
FontInfoResource();
FontInfoResource(byte picFlags, byte picSelect, byte picPick, byte picOnOff, byte fontFlags,
FontJustify justify, int fontSaveBack, const Common::Point &pos, int justifyWidth,
int justifyHeight, const Common::Point &shadow, int foreColor, int backColor,
int shadowColor);
};
class CMapResource {
private:
VoyeurEngine *_vm;
public:
int _steps;
int _fadeStatus;
int _start;
int _end;
byte *_entries;
public:
CMapResource(BoltFilesState &state, const byte *src);
virtual ~CMapResource();
void startFade();
};
class VInitCycleResource {
private:
BoltFilesState &_state;
public:
int _type[4];
byte *_ptr[4];
public:
VInitCycleResource(BoltFilesState &state, const byte *src);
virtual ~VInitCycleResource() {}
void vStartCycle(int flags = 0);
void vStopCycle();
};
/* stampblt.blt resources */
class PtrResource {
public:
Common::Array<BoltEntry *> _entries;
PtrResource(BoltFilesState &state, const byte *src);
virtual ~PtrResource() {}
};
class ControlResource {
public:
int _memberIds[8];
byte *_entries[8];
byte *_ptr;
ControlResource(BoltFilesState &state, const byte *src);
virtual ~ControlResource() {}
};
class ThreadResource {
public:
static int _useCount[8];
static byte *_threadDataPtr;
static CMapResource *_cmd14Pal;
static void initUseCount();
static void unloadAllStacks(VoyeurEngine *vm);
static void init();
private:
VoyeurEngine *_vm;
Common::Point _aptPos;
private:
bool getStateInfo();
byte *getDataOffset();
void getButtonsText();
void getButtonsFlags();
void getButtonsUnused();
void performOpenCard();
const byte *getRecordOffset(const byte *p);
const byte *getNextRecord(const byte *p);
const byte *getSTAMPCard(int cardId);
int getStateFromID(uint32 id);
uint32 getSID(int sid);
void cardAction(const byte *p);
void doSTAMPCardAction();
bool goToStateID(int stackId, int id);
const byte *cardPerform(const byte *card);
bool cardPerform2(const byte *p, int cardCmdId);
void savePrevious();
void setButtonFlag(int idx, byte bits);
void clearButtonFlag(int idx, byte bits);
/**
* Frees the apartment screen data
*/
void freeTheApt();
/**
* Does any necessary animation at the start or end of showing the apartment.
*/
void doAptAnim(int mode);
/**
* Updates the mansion scroll position if ncessary, and returns true if it
* has been changed.
*/
bool checkMansionScroll();
public:
int _stateId;
int _stackId;
int _savedStateId;
int _savedStackId;
int _newStateId;
int _newStackId;
int _flags;
int _fieldA[8];
int _field2A[8];
int _stateFlags;
int _stateCount;
int _parseCount;
uint32 _field46;
byte *_threadInfoPtr;
byte _buttonFlags[64];
const byte *_field8E[64];
byte _buttonIds[64];
const byte *_buttonUnused[48];
byte *_ctlPtr;
byte *_playCommandsPtr;
public:
ThreadResource(BoltFilesState &state, const byte *src);
virtual ~ThreadResource() {}
/**
* Initialise the thread
*/
void initThreadStruct(int idx, int id);
/**
* Loads the specified stack
*/
bool loadAStack(int stackId);
/**
* Unloads the specified stack
*/
void unloadAStack(int stackId);
/**
* Go to a new state and/or stack
*/
bool goToState(int stackId, int stateId);
/**
* Initialises data for the thread based on the current state
*/
bool doState();
bool chooseSTAMPButton(int buttonId);
/**
* Parses the script commands from the currently active stack
*/
void parsePlayCommands();
/**
* Do the camera view looking at the mansion
*/
int doInterface();
/**
* Do the display of a room that has one or more evidence hotspots
* available for display
*/
void doRoom();
/**
* Shows the apartment screen
*/
int doApt();
/**
* Loads data needed for displaying the initial apartment screen
*/
void loadTheApt();
/**
* Check for whether a murder has been recorded
*/
void checkForMurder();
/**
* Check for whether incriminating evidence has been recorded
*/
void checkForIncriminate();
/**
* Synchronizes the game data
*/
void synchronize(Common::Serializer &s);
};
} // End of namespace Voyeur
#endif /* VOYEUR_FILES_H */