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

303 lines
8.2 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/>.
*
*/
#ifndef RIVEN_CARD_H
#define RIVEN_CARD_H
#include "mohawk/riven_scripts.h"
#include "mohawk/riven_sound.h"
#include "common/rect.h"
#include "common/system.h"
namespace Mohawk {
class RivenHotspot;
struct MLSTRecord;
/**
* A game view
*
* The names Card and Stack are legacy from the HyperCard engine used in
* the original mac version of Myst.
*
* Cards contain hotspots, scripts, and resource lists.
*/
class RivenCard {
public:
RivenCard(MohawkEngine_Riven *vm, uint16 id);
~RivenCard();
/**
* An image that can be drawn in this card
*/
struct Picture {
uint16 index;
uint16 id;
Common::Rect rect;
};
/** Initialization routine used to draw a card for the first time or to refresh it */
void enter(bool unkMovies);
/** Run the card's leave scripts */
void leave();
/** Run one of the card's scripts */
void runScript(uint16 scriptType);
/** Get the id of the card in the stack */
uint16 getId() const;
/** Get the card's picture with the specified index */
Picture getPicture(uint16 index) const;
/** Draw one of the card's pictures synchronously or asynchronously */
void drawPicture(uint16 index, bool queue = false);
/** Play the card's ambient sounds with the specified index */
void playSound(uint16 index, bool queue = false);
/** Get the card's sound description with the specified index */
SLSTRecord getSound(uint16 index) const;
/** Replace an ambient sound list with another one */
void overrideSound(uint16 index, uint16 withIndex);
/** Play the card's movie with the specified index */
void playMovie(uint16 index, bool queue = false);
/** Get the card's movie description with the specified index */
MLSTRecord getMovie(uint16 index) const;
/** Draw borders for all the hotspots in the card */
void drawHotspotRects();
/** Enable the zip hotspots if they match to already visited locations */
void initializeZipMode();
/** Get the hotspot containing the specified point */
RivenHotspot *getHotspotContainingPoint(const Common::Point &point) const;
/** Get the hotspot with the specified name */
RivenHotspot *getHotspotByName(const Common::String &name, bool optional = false) const;
/** Find an enabled hotspot with a name matching one of the arguments */
RivenHotspot *findEnabledHotspotByName(const char **names) const;
/** Get the hotspot with the specified BLST id */
RivenHotspot *getHotspotByBlstId(const uint16 blstId) const;
/** Get the currently hovered hotspot if any */
RivenHotspot *getCurHotspot() const;
/** Get all the hotspots in the card. To be used for debugging features only */
Common::Array<RivenHotspot *> getHotspots() const;
/** Activate a hotspot using a hotspot enable list entry */
void activateHotspotEnableRecord(uint16 index);
/** Activate a water effect list entry */
void activateWaterEffect(uint16 index);
/** Handle a mouse down event */
RivenScriptPtr onMouseDown(const Common::Point &mouse);
/** Handle a mouse up event */
RivenScriptPtr onMouseUp(const Common::Point &mouse);
/** Handle a mouse move event */
RivenScriptPtr onMouseMove(const Common::Point &mouse);
/** Handle a keyboard action */
RivenScriptPtr onKeyAction(RivenAction keyAction);
/** General frame update handler */
RivenScriptPtr onFrame();
/** Frame update handler for the mouse cursor */
RivenScriptPtr onMouseUpdate();
/** Frame update handler for mouse dragging */
RivenScriptPtr onMouseDragUpdate();
/** Write all of the card's data to standard output */
void dump() const;
private:
void loadCardResource(uint16 id);
void loadHotspots(uint16 id);
void loadCardPictureList(uint16 id);
void loadCardSoundList(uint16 id);
void loadCardMovieList(uint16 id);
void loadCardHotspotEnableList(uint16 id);
void loadCardWaterEffectList(uint16 id);
void applyPatches(uint16 id);
void applyPropertiesPatchE2E(uint32 globalId);
void applyPropertiesPatch1518D(uint32 globalId);
void applyPropertiesPatch8EB7(uint32 globalId, const Common::String &var, uint16 hotspotId);
void applyPropertiesPatch2E76(uint32 globalId);
void applyPropertiesPatch22118(uint32 globalId);
void applyPropertiesPatch2B414(uint32 globalId);
void setCurrentCardVariable();
void moveHotspot(uint16 blstId, const Common::Rect &position);
void addMenuHotspot(uint16 blstId, const Common::Rect &position, uint16 index,
uint16 externalCommandNameId, const char *externalCommandName);
RivenScriptPtr getScript(uint16 scriptType) const;
void defaultLoadScript();
void updateMouseCursor();
struct HotspotEnableRecord {
uint16 index;
uint16 enabled;
uint16 hotspotId;
};
struct WaterEffectRecord {
uint16 index;
uint16 sfxeId;
uint16 u0;
};
MohawkEngine_Riven *_vm;
// General card data
uint16 _id;
int16 _name;
uint16 _zipModePlace;
RivenScriptList _scripts;
Common::Array<RivenHotspot *> _hotspots;
RivenHotspot *_hoveredHotspot;
RivenHotspot *_pressedHotspot;
// Resource lists
Common::Array<Picture> _pictureList;
Common::Array<SLSTRecord> _soundList;
Common::Array<MLSTRecord> _movieList;
Common::Array<HotspotEnableRecord> _hotspotEnableList;
Common::Array<WaterEffectRecord> _waterEffectList;
};
struct MLSTRecord {
uint16 index;
uint16 movieID;
uint16 playbackSlot;
uint16 left;
uint16 top;
uint16 lowBoundTime;
uint16 startTime;
uint16 highBoundTime;
uint16 loop;
uint16 volume;
uint16 rate;
};
/**
* A Card Hotspot
*
* Hotspots are named rectangular areas of the view.
* Hotspots can be interactive through their scripts.
*/
class RivenHotspot {
public:
RivenHotspot(MohawkEngine_Riven *vm, Common::ReadStream *stream);
/** Get the one of the hotspot's scripts */
RivenScriptPtr getScript(uint16 scriptType) const;
/** Enable or disable the hotspot */
void enable(bool e);
/** Can the hotspot be interacted with? */
bool isEnabled() const;
/** Is the hotspot's purpose to zip to another card */
bool isZip() const;
/** Get the hotspot'a rect in Card coordinates */
Common::Rect getRect() const;
/** Does the hotspot contain the specified point? */
bool containsPoint(const Common::Point &point) const;
/** Override the hotspot's default rect */
void setRect(const Common::Rect &rect);
/** Get the default mouse cursor id to be used when hovering the hostpot */
uint16 getMouseCursor() const;
/** Get the hotspot's name from the current stack's name list */
Common::String getName() const;
/** Get the hotspot's name id */
int16 getNameId() const;
/** Get the hotspot's order in the view */
uint16 getIndex() const;
/** Get the hotspot's enable list id */
uint16 getBlstId() const;
/**
* Get the offset used to control overlap during transitions
*
* -1 means no overlap.
*/
int16 getTransitionOffset() const;
/** Write all of the hotspot's data to standard output */
void dump() const;
/** Apply patches to the hotspot's scripts to fix bugs in the original game scripts */
void applyScriptPatches(uint32 cardGlobalId);
/** Apply patches to the hotspot's properties to fix bugs in the original game scripts */
void applyPropertiesPatches(uint32 cardGlobalId);
private:
enum {
kFlagZip = 1,
kFlagEnabled = 2
};
void loadFromStream(Common::ReadStream *stream);
MohawkEngine_Riven *_vm;
uint16 _blstID;
int16 _nameResource;
Common::Rect _rect;
uint16 _u0;
uint16 _mouseCursor;
uint16 _index;
int16 _transitionOffset;
uint16 _flags;
RivenScriptList _scripts;
};
} // End of namespace Mohawk
#endif