/* 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/screen.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 StateResource; 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; int _bufferEnd; int _bufferBegin; int _bytesLeft; int _bufSize; byte *_bufStart; byte *_bufPos; byte _decompressBuf[DECOMPRESS_SIZE]; int _historyIndex; byte _historyBuffer[0x200]; int _runLength; bool _decompState; int _runType; int _runValue; int _runOffset; Common::File *_curFd; Common::Array _resolves; byte *_boltPageFrame; public: BoltFilesState(VoyeurEngine *vm); byte *decompress(byte *buf, int size, int mode); void nextBlock(); }; class BoltFile { private: Common::Array _groups; protected: BoltFilesState &_state; virtual void initResource(int resType) = 0; void initDefault(); private: void resolveAll(); byte *getBoltMember(uint32 id); public: Common::File _file; BoltFile(const Common::String &filename, BoltFilesState &state); virtual ~BoltFile(); BoltGroup *getBoltGroup(uint16 id); void freeBoltGroup(uint16 id); void freeBoltMember(uint32 id); byte *memberAddr(uint32 id); byte *memberAddrOffset(uint32 id); void resolveIt(uint32 id, byte **p); void resolveFunction(uint32 id, ScreenMethodPtr *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 initState(); void initPtr(); void initControl(); protected: virtual void initResource(int resType); public: StampBoltFile(BoltFilesState &state); }; class BoltGroup { private: Common::SeekableReadStream *_file; public: bool _loaded; bool _processed; int _count; int _fileOffset; Common::Array _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 _initMethod; int _fileOffset; 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; StateResource *_stateResource; ThreadResource *_threadResource; public: BoltEntry(Common::SeekableReadStream *f, uint16 id); virtual ~BoltEntry(); void load(); bool hasResource() const; }; class FilesManager { public: BoltFilesState *_boltFilesState; BoltFile *_curLibPtr; public: FilesManager(VoyeurEngine *vm); ~FilesManager(); bool openBoltLib(const Common::String &filename, BoltFile *&boltFile); byte *fload(const Common::String &filename, int *size); }; 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 _entries; public: RectResource(const byte *src, int size, bool isExtendedRects); RectResource(int xp, int yp, int width, int height); virtual ~RectResource() {} }; /* 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 }; 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, DISPFLAG_NONE = 0}; 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); }; 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; Common::Rect _bounds; uint32 _maskData; uint _planeSize; byte _keyColor; /** * 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, 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 *clippingRect, ViewPortSetupPtr setupFn, ViewPortAddPtr addFn, ViewPortRestorePtr restoreFn); public: ViewPortResource *_parent; ViewPortSetupPtr _setupFn; int _pageCount; ViewPortAddPtr _addFn; int _pageIndex; ViewPortRestorePtr _restoreFn; int _lastPage; ScreenMethodPtr _fn1; Common::Rect _bounds; PictureResource *_currentPic; PictureResource *_activePage; PictureResource *_pages[2]; // 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 favor // of the arrays' .size() method Common::Array *_rectListPtr[3]; int _rectListCount[3]; Common::Rect _clipRect; Common::Rect _fontRect; public: ViewPortResource(BoltFilesState &state, const byte *src); virtual ~ViewPortResource(); void setupViewPort(); void setupViewPort(PictureResource *pic, Common::Rect *clippingRect = NULL); void addSaveRect(int pageIndex, const Common::Rect &r); void fillPic(byte onOff); 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 _palette; Common::Array _entries; int _palIndex; ViewPortListResource(BoltFilesState &state, const byte *src); virtual ~ViewPortListResource() {} }; class FontResource { public: int _minChar, _maxChar; int _fontDepth; 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_CENTER = 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(); void vStopCycle(); }; /* stampblt.blt resources */ class PtrResource { public: Common::Array _entries; PtrResource(BoltFilesState &state, const byte *src); virtual ~PtrResource() {} }; class ControlResource { public: int _memberIds[8]; byte *_entries[8]; int _stateId; StateResource *_state; ControlResource(BoltFilesState &state, const byte *src); virtual ~ControlResource() {} }; /** * Stores data about the intended victim */ class StateResource { public: int _vals[4]; int &_victimIndex; int &_victimEvidenceIndex; int &_victimMurderIndex; StateResource(BoltFilesState &state, const byte *src); virtual ~StateResource() {} /** * Synchronizes the game data */ void synchronize(Common::Serializer &s); }; class ThreadResource { public: static int _useCount[8]; 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 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 _stateFlags; int _stateCount; int _parseCount; uint32 _nextStateId; byte *_threadInfoPtr; byte _buttonFlags[64]; byte _buttonIds[64]; byte *_ctlPtr; byte *_playCommandsPtr; /** * Loads the specified stack */ bool loadAStack(int stackId); /** * Unloads the specified stack */ void unloadAStack(int stackId); /** * Initializes data for the thread based on the current state */ bool doState(); public: ThreadResource(BoltFilesState &state, const byte *src); virtual ~ThreadResource() {} /** * Initialize the thread */ void initThreadStruct(int idx, int id); /** * Go to a new state and/or stack */ bool goToState(int stackId, int stateId); 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(); /** * Synchronizes the game data */ void synchronize(Common::Serializer &s); }; } // End of namespace Voyeur #endif /* VOYEUR_FILES_H */