/* 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 PRINCE_PRINCE_H #define PRINCE_PRINCE_H #include "common/random.h" #include "common/system.h" #include "common/debug.h" #include "common/debug-channels.h" #include "common/textconsole.h" #include "common/rect.h" #include "common/events.h" #include "common/endian.h" #include "common/savefile.h" #include "common/serializer.h" #include "image/bmp.h" #include "gui/debugger.h" #include "engines/engine.h" #include "engines/util.h" #include "audio/mixer.h" #include "video/flic_decoder.h" #include "prince/mob.h" #include "prince/object.h" #include "prince/pscr.h" namespace Prince { enum PrinceGameType { kPrinceDataUNK, kPrinceDataDE, kPrinceDataPL }; struct SavegameHeader; class PrinceEngine; struct PrinceGameDescription; class GraphicsMan; class Script; class Interpreter; class InterpreterFlags; class Debugger; class MusicPlayer; class VariaTxt; class Cursor; class MhwanhDecoder; class Font; class Hero; class Animation; class Room; class Pscr; enum { GF_TRANSLATED = 1 << 0, GF_EXTRACTED = 1 << 1, GF_NOVOICES = 1 << 2 }; struct SavegameHeader { uint8 version; Common::String saveName; Graphics::Surface *thumbnail; int saveYear, saveMonth, saveDay; int saveHour, saveMinutes; }; #define kSavegameStrSize 14 #define kSavegameStr "SCUMMVM_PRINCE" struct Text { const char *_str; uint16 _x, _y; uint16 _time; uint32 _color; Text() : _str(nullptr), _x(0), _y(0), _time(0), _color(255){ } }; struct AnimListItem { uint16 _type; // type of animation - for background anims RND of frame uint16 _fileNumber; uint16 _startPhase; // first phase number uint16 _endPhase; uint16 _loopPhase; int16 _x; int16 _y; uint16 _loopType; uint16 _nextAnim; // number of animation to do for loop = 3 uint16 _flags; // byte 0 - draw masks, byte 1 - draw in front of mask, byte 2 - load but turn off drawing bool loadFromStream(Common::SeekableReadStream &stream); }; struct BAS { int32 _type; // type of sequence int32 _data; // additional data int32 _anims; // number of animations int32 _current; // actual number of animation int32 _counter; // time counter for animation int32 _currRelative; //actual relative number for animation int32 _data2; // additional data for measurements }; const int kStructSizeBAS = 28; struct BASA { int16 _num; // animation number int16 _start; // initial frame int16 _end; // final frame //int16 _pad; // fulfilment to 8 bytes }; const int kStructSizeBASA = 8; // background and normal animation struct Anim { BASA _basaData; int32 _addr; //animation adress int16 _usage; int16 _state; // state of animation: 0 - turning on, 1 - turning off int16 _flags; int16 _frame; // number of phase to show int16 _lastFrame; // last phase int16 _loopFrame; // first frame of loop int16 _showFrame; // actual visible frame of animation int16 _loopType; // type of loop (0 - last frame; 1 - normal loop (begin from _loopFrame); 2 - no loop; 3 - load new animation) int16 _nextAnim; // number of next animation to load after actual int16 _x; int16 _y; int32 _currFrame; int16 _currX; int16 _currY; int16 _currW; int16 _currH; int16 _packFlag; int32 _currShadowFrame; int16 _packShadowFlag; int32 _shadowBack; int16 _relX; int16 _relY; Animation *_animData; Animation *_shadowData; enum AnimOffsets { kAnimState = 10, kAnimFrame = 14, kAnimLastFrame = 16, kAnimX = 26 }; int16 getAnimData(Anim::AnimOffsets offset) { switch (offset) { case kAnimState: return _state; case kAnimFrame: return _frame + 1; // fix for location 30 - man with a dog animation case kAnimX: return _x; default: error("getAnimData() - Wrong offset type: %d", (int) offset); } } void setAnimData(Anim::AnimOffsets offset, int16 value) { if (offset == kAnimX) { _x = value; } else { error("setAnimData() - Wrong offset: %d, value: %d", (int) offset, value); } } }; struct BackgroundAnim { BAS _seq; Common::Array backAnims; }; enum AnimType { kBackgroundAnimation, kNormalAnimation }; // Nak (PL - Nakladka) struct Mask { uint16 _state; // visible / invisible int16 _flags; // turning on / turning off of an mask int16 _x1; int16 _y1; int16 _x2; int16 _y2; int16 _z; int16 _number; // number of mask for background recreating int16 _width; int16 _height; byte *_data; int16 getX() const { return READ_LE_UINT16(_data); } int16 getY() const { return READ_LE_UINT16(_data + 2); } int16 getWidth() const { return READ_LE_UINT16(_data + 4); } int16 getHeight() const { return READ_LE_UINT16(_data + 6); } byte *getMask() const { return (byte *)(_data + 8); } }; struct InvItem { int _x; int _y; Graphics::Surface *_surface; Graphics::Surface *getSurface() const { return _surface; } }; struct DrawNode { int posX; int posY; int posZ; int32 width; int32 height; int32 scaleValue; Graphics::Surface *s; Graphics::Surface *originalRoomSurface; void *data; void (*drawFunction)(Graphics::Surface *, DrawNode *); }; struct DebugChannel { enum Type { kScript, kEngine }; }; class PrinceEngine : public Engine { protected: Common::Error run(); public: PrinceEngine(OSystem *syst, const PrinceGameDescription *gameDesc); virtual ~PrinceEngine(); virtual bool hasFeature(EngineFeature f) const; virtual void pauseEngineIntern(bool pause); virtual bool canSaveGameStateCurrently(); virtual bool canLoadGameStateCurrently(); virtual Common::Error saveGameState(int slot, const Common::String &desc); virtual Common::Error loadGameState(int slot); void playVideo(Common::String videoFilename); WARN_UNUSED_RESULT static bool readSavegameHeader(Common::InSaveFile *in, SavegameHeader &header, bool skipThumbnail = true); Common::String generateSaveName(int slot); void writeSavegameHeader(Common::OutSaveFile *out, SavegameHeader &header); void syncGame(Common::SeekableReadStream *readStream, Common::WriteStream *writeStream); bool loadGame(int slotNumber); void resetGame(); int32 _creditsDataSize; byte *_creditsData; void scrollCredits(); int getGameType() const; const char *getGameId() const; uint32 getFeatures() const; Common::Language getLanguage() const; const PrinceGameDescription *_gameDescription; Video::FlicDecoder _flicPlayer; const Graphics::Surface *_flcFrameSurface; VariaTxt *_variaTxt; uint32 _talkTxtSize; byte *_talkTxt; uint32 _mobTranslationSize; byte *_mobTranslationData; bool loadLocation(uint16 locationNr); bool loadAnim(uint16 animNr, bool loop); bool loadVoice(uint32 textSlot, uint32 sampleSlot, const Common::String &name); bool loadSample(uint32 sampleSlot, const Common::String &name); bool loadZoom(byte *zoomBitmap, uint32 dataSize, const char *resourceName); bool loadShadow(byte *shadowBitmap, uint32 dataSize, const char *resourceName1, const char *resourceName2); bool loadTrans(byte *transTable, const char *resourceName); bool loadMobPriority(const char *resourceName); void loadMobTranslationTexts(); void setMobTranslationTexts(); bool loadMusic(int musNumber); void stopMusic(); void playSample(uint16 sampleId, uint16 loopType); void stopSample(uint16 sampleId); void stopAllSamples(); void freeSample(uint16 sampleId); void freeAllSamples(); void setVoice(uint16 slot, uint32 sampleSlot, uint16 flag); virtual GUI::Debugger *getDebugger(); void changeCursor(uint16 curId); void printAt(uint32 slot, uint8 color, char *s, uint16 x, uint16 y); int calcTextLines(const char *s); int calcTextTime(int numberOfLines); void correctStringDEU(char *s); static const uint8 kMaxTexts = 32; Text _textSlots[kMaxTexts]; Hero *_mainHero; Hero *_secondHero; enum HeroId { kMainHero, kSecondHero }; int _mouseFlag; uint32 _currentTime; uint16 _locationNr; uint16 _sceneWidth; int32 _picWindowX; int32 _picWindowY; Image::BitmapDecoder *_roomBmp; MhwanhDecoder *_suitcaseBmp; Room *_room; Script *_script; InterpreterFlags *_flags; Interpreter *_interpreter; GraphicsMan *_graph; uint8 _currentMidi; byte *_zoomBitmap; byte *_shadowBitmap; byte *_transTable; int16 _scaleValue; // scale for hero or special shadow animation int16 _lightX; // for hero shadow int16 _lightY; int32 _shadScaleValue; int32 _shadLineLen; byte *_shadowLine; void setShadowScale(int32 shadowScale); static void plotShadowLinePoint(int x, int y, int color, void *data); static const int16 kFPS = 15; static const int32 kIntMax = 2147483647; static const int16 kMaxPicWidth = 1280; static const int16 kMaxPicHeight = 480; static const int16 kZoomStep = 4; static const int32 kZoomBitmapLen = kMaxPicHeight / kZoomStep * kMaxPicWidth / kZoomStep; static const int32 kShadowBitmapSize = kMaxPicWidth * kMaxPicHeight / 8; static const int16 kShadowLineArraySize = 2 * 1280 * 4; static const int16 kZoomBitmapWidth = kMaxPicWidth / kZoomStep; static const int16 kZoomBitmapHeight = kMaxPicHeight / kZoomStep; static const int16 kNormalWidth = 640; static const int16 kNormalHeight = 480; static const uint32 kTransTableSize = 256 * 256; static const int kMaxNormAnims = 64; static const int kMaxBackAnims = 64; static const int kMaxObjects = 64; static const int kMaxMobs = 64; Common::Array _drawNodeList; Common::Array _animList; Common::Array _backAnimList; Common::Array _normAnimList; Common::Array _mobList; Common::Array _mobPriorityList; Common::Array _maskList; Common::Array _objList; uint16 *_objSlot; void freeNormAnim(int slot); void freeAllNormAnims(); void removeSingleBackAnim(int slot); Common::RandomSource _randomSource; void checkMasks(int x1, int y1, int sprWidth, int sprHeight, int z); void insertMasks(Graphics::Surface *originalRoomSurface); void showMask(int maskNr, Graphics::Surface *originalRoomSurface); void clsMasks(); void grabMap(); int _selectedMob; // number of selected Mob / inventory item int _selectedItem; // number of item on mouse cursor int _selectedMode; int _currentPointerNumber; static const int16 kMaxInv = 90; // max amount of inventory items in whole game static const uint16 kMaxItems = 30; // size of inventory uint32 _invTxtSize; byte *_invTxt; Graphics::Surface *_optionsPic; Graphics::Surface *_optionsPicInInventory; bool _optionsFlag; int _optionEnabled; int _optionsMob; int _optionsX; int _optionsY; int _optionsWidth; int _optionsHeight; int _invOptionsWidth; int _invOptionsHeight; int _optionsStep; int _invOptionsStep; int _optionsNumber; int _invOptionsNumber; int _optionsColor1; // color for non-selected options int _optionsColor2; // color for selected option bool _showInventoryFlag; int _invExamY; bool _inventoryBackgroundRemember; int _invLineX; int _invLineY; int _invLine; // number of items in one line int _invLines; // number of lines with inventory items int _invLineW; int _invLineH; int _maxInvW; int _maxInvH; int _invLineSkipX; int _invLineSkipY; int _invX1; int _invY1; int _invWidth; int _invHeight; bool _invCurInside; int _mst_shadow; int _mst_shadow2; // blinking after adding new item int _candleCounter; // special counter for candle inventory item int _invMaxCount; // time to turn inventory on int _invCounter; // turning on counter void inventoryFlagChange(bool inventoryState); bool loadAllInv(); void rememberScreenInv(); void prepareInventoryToView(); void drawInvItems(); void displayInventory(); void addInv(int heroId, int item, bool addItemQuiet); void remInv(int heroId, int item); void clearInv(int heroId); void swapInv(int heroId); void addInvObj(); void makeInvCursor(int itemNr); void enableOptions(bool checkType); void checkOptions(); void checkInvOptions(); void openInventoryCheck(); void leftMouseButton(); void rightMouseButton(); void inventoryLeftMouseButton(); void inventoryRightMouseButton(); void dialogLeftMouseButton(byte *string, int dialogSelected); uint32 _dialogDatSize; byte *_dialogDat; byte *_dialogData; // on, off flags for lines of dialog text byte *_dialogBoxAddr[32]; // adresses of dialog windows byte *_dialogOptAddr[32]; // adresses of dialog options int _dialogOptLines[4 * 32]; // numbers of initial dialog lines byte *_dialogText; int _dialogLines; bool _dialogFlag; int _dialogWidth; int _dialogHeight; int _dialogLineSpace; int _dialogColor1; // color for non-selected options int _dialogColor2; // color for selected option Graphics::Surface *_dialogImage; void createDialogBox(int dialogBoxNr); void dialogRun(); void talkHero(int slot); void doTalkAnim(int animNumber, int slot, AnimType animType); static const uint8 zoomInStep = 8; void initZoomIn(int slot); void initZoomOut(int slot); void doZoomIn(int slot); void doZoomOut(int slot); void freeZoomObject(int slot); static const uint8 kFadeStep = 4; void blackPalette(); void setPalette(const byte *palette); int getMob(Common::Array &mobList, bool usePriorityList, int posX, int posY); // 'Throw a rock' mini-game: static const int16 kCurveLen = 17; static const int kCelStep = 4; int16 *_curveData; int _curvPos; void makeCurve(); void getCurve(); void mouseWeirdo(); static const uint16 kPowerBarPosX = 288; static const uint16 kPowerBarPosY = 430; static const uint8 kPowerBarWidth = 64; static const uint8 kPowerBarHeight = 16; static const uint8 kPowerBarBackgroundColor = 0; static const uint16 kPowerBarGreenPosY = 434; static const uint8 kPowerBarGreenColor1 = 202; static const uint8 kPowerBarGreenColor2 = 235; static const uint8 kPowerBarGreenHeight = 8; void showPower(); // Pathfinding static const int16 kPathGridStep = 2; static const uint32 kPathBitmapLen = (kMaxPicHeight / kPathGridStep * kMaxPicWidth / kPathGridStep) / 8; static const int32 kTracePts = 8000; static const int32 kPBW = kMaxPicWidth / 16; // PathBitmapWidth static const int kMinDistance = 2500; byte *_roomPathBitmap; // PL - Sala byte *_roomPathBitmapTemp; // PL - SSala byte *_coordsBufEnd; byte *_coordsBuf; // optimal path byte *_coords; // last path point adress from coordsBuf byte *_coordsBuf2; byte *_coords2; byte *_coordsBuf3; byte *_coords3; int _traceLineLen; bool _traceLineFirstPointFlag; // if plotTraceLine after first point bool _tracePointFirstPointFlag; // if plotTracePoint after first point byte *_directionTable; int _shanLen; byte *_checkBitmapTemp; byte *_checkBitmap; int _checkMask; int _checkX; int _checkY; byte *_rembBitmapTemp; byte *_rembBitmap; int _rembMask; int _rembX; int _rembY; int _fpX; int _fpY; int drawLine(int x0, int y0, int x1, int y1, int (*plotProc)(int, int, void *), void *data); bool loadPath(const char *resourceName); byte *makePath(int heroId, int currX, int currY, int destX, int destY); void findPoint(int x, int y); int getPixelAddr(byte *pathBitmap, int x, int y); static int plotTraceLine(int x, int y, void *data); void specialPlotInside(int x, int y); bool tracePath(int x1, int y1, int x2, int y2); Direction makeDirection(int x1, int y1, int x2, int y2); void specialPlot(int x, int y); void specialPlot2(int x, int y); void allocCoords2(); void freeCoords2(); void freeCoords3(); static int plotTracePoint(int x, int y, void *data); void specialPlotInside2(int x, int y); void approxPath(); void freeDirectionTable(); void scanDirections(); int scanDirectionsFindNext(byte *coords, int xDiff, int yDiff); void moveShandria(); void walkTo(); void moveRunHero(int heroId, int x, int y, int dir, bool runHeroFlag); int leftDownDir(); int leftDir(); int leftUpDir(); int rightDownDir(); int rightDir(); int rightUpDir(); int upLeftDir(); int upDir(); int upRightDir(); int downLeftDir(); int downDir(); int downRightDir(); int cpe(); int checkLeftDownDir(); int checkLeftDir(); int checkDownDir(); int checkUpDir(); int checkRightDir(); int checkLeftUpDir(); int checkRightDownDir(); int checkRightUpDir(); private: bool playNextFLCFrame(); void keyHandler(Common::Event event); int checkMob(Graphics::Surface *screen, Common::Array &mobList, bool usePriorityList); void drawScreen(); void showTexts(Graphics::Surface *screen); void init(); void showLogo(); void showAnim(Anim &anim); void showNormAnims(); void setBackAnim(Anim &backAnim); void showBackAnims(); void clearBackAnimList(); bool spriteCheck(int sprWidth, int sprHeight, int destX, int destY); void showSprite(Graphics::Surface *spriteSurface, int destX, int destY, int destZ); void showSpriteShadow(Graphics::Surface *shadowSurface, int destX, int destY, int destZ); void showObjects(); void showParallax(); static bool compareDrawNodes(DrawNode d1, DrawNode d2); void runDrawNodes(); void makeShadowTable(int brightness); void pausePrinceEngine(int fps = kFPS); uint32 getTextWidth(const char *s); void debugEngine(const char *s, ...); uint8 _cursorNr; Common::RandomSource *_rnd; Cursor *_cursor1; Graphics::Surface *_cursor2; Cursor *_cursor3; Debugger *_debugger; Font *_font; MusicPlayer *_midiPlayer; static const int kMaxSamples = 60; Audio::RewindableAudioStream *_audioStream[kMaxSamples]; Audio::SoundHandle _soundHandle[kMaxSamples]; Common::Array _pscrList; Common::Array _allInvList; Common::Array _invMobList; bool _flicLooped; void mainLoop(); }; } // End of namespace Prince #endif