mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-13 13:10:53 +00:00
aed02365ec
svn-id: r47541
1048 lines
36 KiB
C++
1048 lines
36 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.
|
|
*
|
|
* $URL$
|
|
* $Id$
|
|
*
|
|
*/
|
|
|
|
#ifndef COMMON_SYSTEM_H
|
|
#define COMMON_SYSTEM_H
|
|
|
|
#include "common/scummsys.h"
|
|
#include "common/noncopyable.h"
|
|
#include "common/rect.h"
|
|
|
|
#include "graphics/pixelformat.h"
|
|
|
|
namespace Audio {
|
|
class Mixer;
|
|
}
|
|
|
|
namespace Graphics {
|
|
struct Surface;
|
|
}
|
|
|
|
namespace Common {
|
|
struct Event;
|
|
class EventManager;
|
|
class SaveFileManager;
|
|
class SearchSet;
|
|
class TimerManager;
|
|
class SeekableReadStream;
|
|
class WriteStream;
|
|
class HardwareKeySet;
|
|
}
|
|
|
|
class FilesystemFactory;
|
|
|
|
/**
|
|
* A structure describing time and date. This is a clone of struct tm
|
|
* from time.h. We roll our own since not all systems provide time.h.
|
|
* We also do not imitate all files of struct tm, only those we
|
|
* actually need.
|
|
*
|
|
* @note For now, the members are named exactly as in struct tm to ease
|
|
* the transition.
|
|
*/
|
|
struct TimeDate {
|
|
int tm_sec; ///< seconds (0 - 60)
|
|
int tm_min; ///< minutes (0 - 59)
|
|
int tm_hour; ///< hours (0 - 23)
|
|
int tm_mday; ///< day of month (1 - 31)
|
|
int tm_mon; ///< month of year (0 - 11)
|
|
int tm_year; ///< year - 1900
|
|
};
|
|
|
|
/**
|
|
* Interface for ScummVM backends. If you want to port ScummVM to a system
|
|
* which is not currently covered by any of our backends, this is the place
|
|
* to start. ScummVM will create an instance of a subclass of this interface
|
|
* and use it to interact with the system.
|
|
*
|
|
* In particular, a backend provides a video surface for ScummVM to draw in;
|
|
* methods to create timers, to handle user input events,
|
|
* control audio CD playback, and sound output.
|
|
*/
|
|
class OSystem : Common::NonCopyable {
|
|
protected:
|
|
OSystem();
|
|
virtual ~OSystem();
|
|
|
|
public:
|
|
|
|
/**
|
|
* The following method is called once, from main.cpp, after all
|
|
* config data (including command line params etc.) are fully loaded.
|
|
*
|
|
* @note Subclasses should always invoke the implementation of their
|
|
* parent class. They should do so near the end of their own
|
|
* implementation.
|
|
*/
|
|
virtual void initBackend() { }
|
|
|
|
/**
|
|
* Allows the backend to perform engine specific init.
|
|
* Called just before the engine is run.
|
|
*/
|
|
virtual void engineInit() { }
|
|
|
|
/**
|
|
* Allows the backend to perform engine specific de-init.
|
|
* Called after the engine finishes.
|
|
*/
|
|
virtual void engineDone() { }
|
|
|
|
/** @name Feature flags */
|
|
//@{
|
|
|
|
/**
|
|
* A feature in this context means an ability of the backend which can be
|
|
* either on or off. Examples include:
|
|
* - fullscreen mode
|
|
* - aspect ration correction
|
|
* - a virtual keyboard for text entry (on PDAs)
|
|
*/
|
|
enum Feature {
|
|
/**
|
|
* If your backend supports both a windowed and a fullscreen mode,
|
|
* then this feature flag can be used to switch between the two.
|
|
*/
|
|
kFeatureFullscreenMode,
|
|
|
|
/**
|
|
* Control aspect ratio correction. Aspect ratio correction is used to
|
|
* correct games running at 320x200 (i.e with an aspect ratio of 8:5),
|
|
* but which on their original hardware were displayed with the
|
|
* standard 4:3 ratio (that is, the original graphics used non-square
|
|
* pixels). When the backend support this, then games running at
|
|
* 320x200 pixels should be scaled up to 320x240 pixels. For all other
|
|
* resolutions, ignore this feature flag.
|
|
* @note You can find utility functions in common/scaler.h which can
|
|
* be used to implement aspect ratio correction. In particular,
|
|
* stretch200To240() can stretch a rect, including (very fast)
|
|
* interpolation, and works in-place.
|
|
*/
|
|
kFeatureAspectRatioCorrection,
|
|
|
|
/**
|
|
* Determine whether a virtual keyboard is too be shown or not.
|
|
* This would mostly be implemented by backends for hand held devices,
|
|
* like PocketPC, Palms, Symbian phones like the P800, Zaurus, etc.
|
|
*/
|
|
kFeatureVirtualKeyboard,
|
|
|
|
/**
|
|
* This flag is a bit more obscure: it gives a hint to the backend that
|
|
* the frontend code is very inefficient in doing screen updates. So
|
|
* the frontend might do a lot of fullscreen blits even though only a
|
|
* tiny portion of the actual screen data changed. In that case, it
|
|
* might pay off for the backend to compute which parts actually changed,
|
|
* and then only mark those as dirty.
|
|
* Implementing this is purely optional, and no harm should arise
|
|
* when not doing so (except for decreased speed in said frontends).
|
|
*/
|
|
kFeatureAutoComputeDirtyRects,
|
|
|
|
/**
|
|
* This flag determines whether or not the cursor can have its own palette.
|
|
* It is currently used only by some Macintosh versions of Humongous
|
|
* Entertainment games. If the backend doesn't implement this feature then
|
|
* the engine switches to b/w versions of cursors.
|
|
* The GUI also relies on this feature for mouse cursors.
|
|
*
|
|
* To enable the cursor palette call "disableCursorPalette" with false.
|
|
* @see disableCursorPalette
|
|
*/
|
|
kFeatureCursorHasPalette,
|
|
|
|
/**
|
|
* Set to true if the overlay pixel format has an alpha channel.
|
|
* This should only be set if it offers at least 3-4 bits of accuracy,
|
|
* as opposed to a single alpha bit.
|
|
*/
|
|
kFeatureOverlaySupportsAlpha,
|
|
|
|
/**
|
|
* Set to true to iconify the window.
|
|
*/
|
|
kFeatureIconifyWindow,
|
|
|
|
/**
|
|
* This feature, set to true, is a hint toward the backend to disable all
|
|
* key filtering/mapping, in cases where it would be beneficial to do so.
|
|
* As an example case, this is used in the agi engine's predictive dialog.
|
|
* When the dialog is displayed this feature is set so that backends with
|
|
* phone-like keypad temporarily unmap all user actions which leads to
|
|
* comfortable word entry. Conversely, when the dialog exits the feature
|
|
* is set to false.
|
|
* TODO: Fingolfin suggests that the way the feature is used can be
|
|
* generalized in this sense: Have a keyboard mapping feature, which the
|
|
* engine queries for to assign keys to actions ("Here's my default key
|
|
* map for these actions, what do you want them set to?").
|
|
*/
|
|
kFeatureDisableKeyFiltering
|
|
};
|
|
|
|
/**
|
|
* Determine whether the backend supports the specified feature.
|
|
*/
|
|
virtual bool hasFeature(Feature f) { return false; }
|
|
|
|
/**
|
|
* En-/disable the specified feature. For example, this may be used to
|
|
* enable fullscreen mode, or to deactivate aspect correction, etc.
|
|
*/
|
|
virtual void setFeatureState(Feature f, bool enable) {}
|
|
|
|
/**
|
|
* Query the state of the specified feature. For example, test whether
|
|
* fullscreen mode is active or not.
|
|
*/
|
|
virtual bool getFeatureState(Feature f) { return false; }
|
|
|
|
//@}
|
|
|
|
|
|
|
|
/**
|
|
* @name Graphics
|
|
*
|
|
* The way graphics work in the class OSystem are meant to make
|
|
* it possible for game frontends to implement all they need in
|
|
* an efficient manner. The downside of this is that it may be
|
|
* rather complicated for backend authors to fully understand and
|
|
* implement the semantics of the OSystem interface.
|
|
*
|
|
*
|
|
* The graphics visible to the user in the end are actually
|
|
* composed in three layers: the game graphics, the overlay
|
|
* graphics, and the mouse.
|
|
*
|
|
* First, there are the game graphics. They are always 8bpp, and
|
|
* the methods in this section deal with them exclusively. In
|
|
* particular, the size of the game graphics is defined by a call
|
|
* to initSize(), and copyRectToScreen() blits 8bpp data into the
|
|
* game layer. Let W and H denote the width and height of the
|
|
* game graphics.
|
|
*
|
|
* Before the user sees these graphics, the backend may apply some
|
|
* transformations to it; for example, the may be scaled to better
|
|
* fit on the visible screen; or aspect ratio correction may be
|
|
* performed (see kFeatureAspectRatioCorrection). As a result of
|
|
* this, a pixel of the game graphics may occupy a region bigger
|
|
* than a single pixel on the screen. We define p_w and p_h to be
|
|
* the width resp. height of a game pixel on the screen.
|
|
*
|
|
* In addition, there is a vertical "shake offset" (as defined by
|
|
* setShakePos) which is used in some games to provide a shaking
|
|
* effect. Note that shaking is applied to all three layers, i.e.
|
|
* also to the overlay and the mouse. We denote the shake offset
|
|
* by S.
|
|
*
|
|
* Putting this together, a pixel (x,y) of the game graphics is
|
|
* transformed to a rectangle of height p_h and width p_w
|
|
* appearing at position (p_w * x, p_hw * (y + S)) on the real
|
|
* screen (in addition, a backend may choose to offset
|
|
* everything, e.g. to center the graphics on the screen).
|
|
*
|
|
*
|
|
* The next layer is the overlay. It is composed over the game
|
|
* graphics. By default, it has exactly the same size and
|
|
* resolution as the game graphics. However, client code can
|
|
* specify an overlay scale (as an additional parameter to
|
|
* initSize()). This is meant to increase the resolution of the
|
|
* overlay while keeping its size the same as that of the game
|
|
* graphics. For example, if the overlay scale is 2, and the game
|
|
* graphics have a resolution of 320x200; then the overlay shall
|
|
* have a resolution of 640x400, but it still has the same
|
|
* physical size as the game graphics.
|
|
* The overlay usually uses 16bpp, but on some ports, only 8bpp
|
|
* are availble, so that is supported, too, via a compile time
|
|
* switch (see also the OverlayColor typedef in scummsys.h).
|
|
*
|
|
*
|
|
* Finally, there is the mouse layer. This layer doesn't have to
|
|
* actually exist within the backend -- it all depends on how a
|
|
* backend chooses to implement mouse cursors, but in the default
|
|
* SDL backend, it really is a separate layer. The mouse is
|
|
* always in 8bpp but can have a palette of its own, if the
|
|
* backend supports it. The scale of the mouse cursor is called
|
|
* 'cursorTargetScale'. This is meant as a hint to the backend.
|
|
* For example, let us assume the overlay is not visible, and the
|
|
* game graphics are displayed using a 2x scaler. If a mouse
|
|
* cursor with a cursorTargetScale of 1 is set, then it should be
|
|
* scaled by factor 2x, too, just like the game graphics. But if
|
|
* it has a cursorTargetScale of 2, then it shouldn't be scaled
|
|
* again by the game graphics scaler.
|
|
*/
|
|
//@{
|
|
|
|
/**
|
|
* Description of a graphics mode.
|
|
*/
|
|
struct GraphicsMode {
|
|
/**
|
|
* The 'name' of the graphics mode. This name is matched when selecting
|
|
* a mode via the command line, or via the config file.
|
|
* Examples: "1x", "advmame2x", "hq3x"
|
|
*/
|
|
const char *name;
|
|
/**
|
|
* Human readable description of the scaler.
|
|
* Examples: "Normal (no scaling)", "AdvMAME2x", "HQ3x"
|
|
*/
|
|
const char *description;
|
|
/**
|
|
* ID of the graphics mode. How to use this is completely up to the
|
|
* backend. This value will be passed to the setGraphicsMode(int)
|
|
* method by client code.
|
|
*/
|
|
int id;
|
|
};
|
|
|
|
/**
|
|
* Retrieve a list of all graphics modes supported by this backend.
|
|
* This can be both video modes as well as graphic filters/scalers;
|
|
* it is completely up to the backend maintainer to decide what is
|
|
* appropriate here and what not.
|
|
* The list is terminated by an all-zero entry.
|
|
* @return a list of supported graphics modes
|
|
*/
|
|
virtual const GraphicsMode *getSupportedGraphicsModes() const = 0;
|
|
|
|
/**
|
|
* Return the ID of the 'default' graphics mode. What exactly this means
|
|
* is up to the backend. This mode is set by the client code when no user
|
|
* overrides are present (i.e. if no custom graphics mode is selected via
|
|
* the command line or a config file).
|
|
*
|
|
* @return the ID of the 'default' graphics mode
|
|
*/
|
|
virtual int getDefaultGraphicsMode() const = 0;
|
|
|
|
/**
|
|
* Switch to the specified graphics mode. If switching to the new mode
|
|
* failed, this method returns false.
|
|
*
|
|
* @param mode the ID of the new graphics mode
|
|
* @return true if the switch was successful, false otherwise
|
|
*/
|
|
virtual bool setGraphicsMode(int mode) = 0;
|
|
|
|
/**
|
|
* Switch to the graphics mode with the given name. If 'name' is unknown,
|
|
* or if switching to the new mode failed, this method returns false.
|
|
*
|
|
* @param name the name of the new graphics mode
|
|
* @return true if the switch was successful, false otherwise
|
|
* @note This is implemented via the setGraphicsMode(int) method, as well
|
|
* as getSupportedGraphicsModes() and getDefaultGraphicsMode().
|
|
* In particular, backends do not have to overload this!
|
|
*/
|
|
bool setGraphicsMode(const char *name);
|
|
|
|
/**
|
|
* Determine which graphics mode is currently active.
|
|
* @return the active graphics mode
|
|
*/
|
|
virtual int getGraphicsMode() const = 0;
|
|
|
|
#ifdef USE_RGB_COLOR
|
|
/**
|
|
* Determine the pixel format currently in use for screen rendering.
|
|
* @return the active screen pixel format.
|
|
* @see Graphics::PixelFormat
|
|
*/
|
|
virtual Graphics::PixelFormat getScreenFormat() const = 0;
|
|
|
|
/**
|
|
* Returns a list of all pixel formats supported by the backend.
|
|
* The first item in the list must be directly supported by hardware,
|
|
* and provide the largest color space of those formats with direct
|
|
* hardware support. It is also strongly recommended that remaining
|
|
* formats should be placed in order of descending preference for the
|
|
* backend to use.
|
|
*
|
|
* EG: a backend that supports 32-bit ABGR and 16-bit 555 BGR in hardware
|
|
* and provides conversion from equivalent RGB(A) modes should order its list
|
|
* 1) Graphics::PixelFormat(4, 0, 0, 0, 0, 0, 8, 16, 24)
|
|
* 2) Graphics::PixelFormat(2, 3, 3, 3, 8, 0, 5, 10, 0)
|
|
* 3) Graphics::PixelFormat(4, 0, 0, 0, 0, 24, 16, 8, 0)
|
|
* 4) Graphics::PixelFormat(2, 3, 3, 3, 8, 10, 5, 0, 0)
|
|
* 5) Graphics::PixelFormat::createFormatCLUT8()
|
|
*
|
|
* @see Graphics::PixelFormat
|
|
*
|
|
* @note Backends supporting RGB color should accept game data in RGB color
|
|
* order, even if hardware uses BGR or some other color order.
|
|
*/
|
|
virtual Common::List<Graphics::PixelFormat> getSupportedFormats() = 0;
|
|
#else
|
|
inline Graphics::PixelFormat getScreenFormat() const {
|
|
return Graphics::PixelFormat::createFormatCLUT8();
|
|
};
|
|
|
|
inline Common::List<Graphics::PixelFormat> getSupportedFormats() const {
|
|
Common::List<Graphics::PixelFormat> list;
|
|
list.push_back(Graphics::PixelFormat::createFormatCLUT8());
|
|
return list;
|
|
};
|
|
#endif
|
|
|
|
/**
|
|
* Set the size and color format of the virtual screen. Typical sizes include:
|
|
* - 320x200 (e.g. for most SCUMM games, and Simon)
|
|
* - 320x240 (e.g. for FM-TOWN SCUMM games)
|
|
* - 640x480 (e.g. for Curse of Monkey Island)
|
|
*
|
|
* This is the resolution for which the client code generates data;
|
|
* this is not necessarily equal to the actual display size. For example,
|
|
* a backend may magnify the graphics to fit on screen (see also the
|
|
* GraphicsMode); stretch the data to perform aspect ratio correction;
|
|
* or shrink it to fit on small screens (in cell phones).
|
|
*
|
|
* Typical formats include:
|
|
* CLUT8 (e.g. 256 color, for most games)
|
|
* RGB555 (e.g. 16-bit color, for later SCUMM HE games)
|
|
* RGB565 (e.g. 16-bit color, for Urban Runner)
|
|
*
|
|
* This is the pixel format for which the client code generates data;
|
|
* this is not necessarily equal to the hardware pixel format. For example,
|
|
* a backend may perform color lookup of 8-bit graphics before pushing
|
|
* a screen to hardware, or correct the ARGB color order.
|
|
*
|
|
* @param width the new virtual screen width
|
|
* @param height the new virtual screen height
|
|
* @param format the new virtual screen pixel format
|
|
*/
|
|
virtual void initSize(uint width, uint height, const Graphics::PixelFormat *format = NULL) = 0;
|
|
|
|
/**
|
|
* Return an int value which is changed whenever any screen
|
|
* parameters (like the resolution) change. That is, whenever a
|
|
* EVENT_SCREEN_CHANGED would be sent. You can track this value
|
|
* in your code to detect screen changes in case you do not have
|
|
* full control over the event loop(s) being used (like the GUI
|
|
* code).
|
|
*
|
|
* @return an integer which can be used to track screen changes
|
|
*
|
|
* @note Backends which generate EVENT_SCREEN_CHANGED events MUST
|
|
* overload this method appropriately.
|
|
*/
|
|
virtual int getScreenChangeID() const { return 0; }
|
|
|
|
/**
|
|
* Begin a new GFX transaction, which is a sequence of GFX mode changes.
|
|
* The idea behind GFX transactions is to make it possible to activate
|
|
* several different GFX changes at once as a "batch" operation. For
|
|
* example, assume we are running in 320x200 with a 2x scaler (thus using
|
|
* 640x400 pixels in total). Now, we want to switch to 640x400 with the 1x
|
|
* scaler. Without transactions, we have to choose whether we want to first
|
|
* switch the scaler mode, or first to 640x400 mode. In either case,
|
|
* depending on the backend implementation, some ugliness may result.
|
|
* E.g. the window might briefly switch to 320x200 or 1280x800.
|
|
* Using transactions, this can be avoided.
|
|
*
|
|
* @note Transaction support is optional, and the default implementations
|
|
* of the relevant methods simply do nothing.
|
|
* @see endGFXTransaction
|
|
*/
|
|
virtual void beginGFXTransaction() {}
|
|
|
|
/**
|
|
* This type is able to save the different errors which can happen while
|
|
* changing GFX config values inside GFX transactions.
|
|
*
|
|
* endGFXTransaction returns a ORed combination of the '*Failed' values
|
|
* if any problem occures, on success 0.
|
|
*
|
|
* @see endGFXTransaction
|
|
*/
|
|
enum TransactionError {
|
|
kTransactionSuccess = 0, /**< Everything fine (use EQUAL check for this one!) */
|
|
kTransactionAspectRatioFailed = (1 << 0), /**< Failed switching aspect ratio correction mode */
|
|
kTransactionFullscreenFailed = (1 << 1), /**< Failed switching fullscreen mode */
|
|
kTransactionModeSwitchFailed = (1 << 2), /**< Failed switching the GFX graphics mode (setGraphicsMode) */
|
|
kTransactionSizeChangeFailed = (1 << 3), /**< Failed switching the screen dimensions (initSize) */
|
|
kTransactionFormatNotSupported = (1 << 4) /**< Failed setting the color format */
|
|
};
|
|
|
|
/**
|
|
* End (and thereby commit) the current GFX transaction.
|
|
* @see beginGFXTransaction
|
|
* @see kTransactionError
|
|
* @return returns a ORed combination of TransactionError values or 0 on success
|
|
*/
|
|
virtual TransactionError endGFXTransaction() { return kTransactionSuccess; }
|
|
|
|
|
|
/**
|
|
* Returns the currently set virtual screen height.
|
|
* @see initSize
|
|
* @return the currently set virtual screen height
|
|
*/
|
|
virtual int16 getHeight() = 0;
|
|
|
|
/**
|
|
* Returns the currently set virtual screen width.
|
|
* @see initSize
|
|
* @return the currently set virtual screen width
|
|
*/
|
|
virtual int16 getWidth() = 0;
|
|
|
|
/**
|
|
* Replace the specified range of the palette with new colors.
|
|
* The palette entries from 'start' till (start+num-1) will be replaced - so
|
|
* a full palette update is accomplished via start=0, num=256.
|
|
*
|
|
* The palette data is specified in interleaved RGBA format. That is, the
|
|
* first byte of the memory block 'colors' points at is the red component
|
|
* of the first new color; the second byte the green component of the first
|
|
* new color; the third byte the blue component, the last byte to the alpha
|
|
* (transparency) value. Then the second color starts, and so on. So memory
|
|
* looks like this: R1-G1-B1-A1-R2-G2-B2-A2-R3-...
|
|
*
|
|
* @param colors the new palette data, in interleaved RGBA format
|
|
* @param start the first palette entry to be updated
|
|
* @param num the number of palette entries to be updated
|
|
*
|
|
* @note It is an error if start+num exceeds 256, behaviour is undefined
|
|
* in that case (the backend may ignore it silently or assert).
|
|
* @note The alpha value is not actually used, and future revisions of this
|
|
* API are probably going to remove it.
|
|
*/
|
|
virtual void setPalette(const byte *colors, uint start, uint num) = 0;
|
|
|
|
/**
|
|
* Grabs a specified part of the currently active palette.
|
|
* The format is the same as for setPalette.
|
|
*
|
|
* @see setPalette
|
|
* @param colors the palette data, in interleaved RGBA format
|
|
* @param start the first platte entry to be read
|
|
* @param num the number of palette entries to be read
|
|
*/
|
|
virtual void grabPalette(byte *colors, uint start, uint num) = 0;
|
|
|
|
/**
|
|
* Blit a bitmap to the virtual screen.
|
|
* The real screen will not immediately be updated to reflect the changes.
|
|
* Client code has to to call updateScreen to ensure any changes are
|
|
* visible to the user. This can be used to optimize drawing and reduce
|
|
* flicker.
|
|
* The graphics data uses 8 bits per pixel, using the palette specified
|
|
* via setPalette.
|
|
*
|
|
* @param buf the buffer containing the graphics data source
|
|
* @param pitch the pitch of the buffer (number of bytes in a scanline)
|
|
* @param x the x coordinate of the destination rectangle
|
|
* @param y the y coordinate of the destination rectangle
|
|
* @param w the width of the destination rectangle
|
|
* @param h the height of the destination rectangle
|
|
*
|
|
* @note The specified destination rectangle must be completly contained
|
|
* in the visible screen space, and must be non-empty. If not, a
|
|
* backend may or may not perform clipping, trigger an assert or
|
|
* silently corrupt memory.
|
|
*
|
|
* @see updateScreen
|
|
*/
|
|
virtual void copyRectToScreen(const byte *buf, int pitch, int x, int y, int w, int h) = 0;
|
|
|
|
/**
|
|
* Lock the active screen framebuffer and return a Graphics::Surface
|
|
* representing it. The caller can then perform arbitrary graphics
|
|
* transformations on the framebuffer (blitting, scrolling, etc.).
|
|
* Must be followed by matching call to unlockScreen(). Calling code
|
|
* should make sure to only lock the framebuffer for the briefest
|
|
* periods of time possible, as the whole system is potentially stalled
|
|
* while the lock is active.
|
|
* Returns 0 if an error occurred. Otherwise an 8bit surface is returned.
|
|
*
|
|
* The returned surface must *not* be deleted by the client code.
|
|
*/
|
|
virtual Graphics::Surface *lockScreen() = 0;
|
|
|
|
/**
|
|
* Unlock the screen framebuffer, and mark it as dirty (i.e. during the
|
|
* next updateScreen() call, the whole screen will be updated.
|
|
*/
|
|
virtual void unlockScreen() = 0;
|
|
|
|
/**
|
|
* Fills the screen with a given color value.
|
|
*
|
|
* @note We are using uint32 here even though currently
|
|
* we only support 8bpp indexed mode. Thus the value should
|
|
* be always inside [0, 255] for now.
|
|
*/
|
|
virtual void fillScreen(uint32 col) = 0;
|
|
|
|
/**
|
|
* Flush the whole screen, that is render the current content of the screen
|
|
* framebuffer to the display.
|
|
*
|
|
* This method could be called very often by engines. Backends are hence
|
|
* supposed to only perform any redrawing if it is necessary, and otherwise
|
|
* return immediately. For a discussion of the subject, see
|
|
* <http://www.nabble.com/ATTN-porters%3A-updateScreen%28%29-OSystem-method-tt3960261.html#a3960261>
|
|
*/
|
|
virtual void updateScreen() = 0;
|
|
|
|
/**
|
|
* Set current shake position, a feature needed for some SCUMM screen
|
|
* effects. The effect causes the displayed graphics to be shifted upwards
|
|
* by the specified (always positive) offset. The area at the bottom of the
|
|
* screen which is moved into view by this is filled with black. This does
|
|
* not cause any graphic data to be lost - that is, to restore the original
|
|
* view, the game engine only has to call this method again with offset
|
|
* equal to zero. No calls to copyRectToScreen are necessary.
|
|
* @param shakeOffset the shake offset
|
|
*
|
|
* @note This is currently used in the SCUMM, QUEEN and KYRA engines.
|
|
*/
|
|
virtual void setShakePos(int shakeOffset) = 0;
|
|
|
|
/**
|
|
* Sets the area of the screen that has the focus. For example, when a character
|
|
* is speaking, they will have the focus. Allows for pan-and-scan style views
|
|
* where the backend could follow the speaking character or area of interest on
|
|
* the screen.
|
|
*
|
|
* The backend is responsible for clipping the rectangle and deciding how best to
|
|
* zoom the screen to show any shape and size rectangle the engine provides.
|
|
*
|
|
* @param rect A rectangle on the screen to be focused on
|
|
* @see clearFocusRectangle
|
|
*/
|
|
virtual void setFocusRectangle(const Common::Rect& rect) {}
|
|
|
|
/**
|
|
* Clears the focus set by a call to setFocusRectangle(). This allows the engine
|
|
* to clear the focus during times when no particular area of the screen has the
|
|
* focus.
|
|
* @see setFocusRectangle
|
|
*/
|
|
virtual void clearFocusRectangle() {}
|
|
|
|
//@}
|
|
|
|
|
|
|
|
/**
|
|
* @name Overlay
|
|
* In order to be able to display dialogs atop the game graphics, backends
|
|
* must provide an overlay mode.
|
|
*
|
|
* While the game graphics are always 8 bpp, the overlay can be 8 or 16 bpp.
|
|
* Depending on which it is, OverlayColor is 8 or 16 bit.
|
|
*
|
|
* For 'coolness' we usually want to have an overlay which is blended over
|
|
* the game graphics. On backends which support alpha blending, this is
|
|
* no issue; but on other systems (in particular those which only support
|
|
* 8bpp), this needs some trickery.
|
|
*
|
|
* Essentially, we fake (alpha) blending on these systems by copying the
|
|
* current game graphics into the overlay buffer when activating the overlay,
|
|
* then manually compose whatever graphics we want to show in the overlay.
|
|
* This works because we assume the game to be "paused" whenever an overlay
|
|
* is active.
|
|
*/
|
|
//@{
|
|
|
|
/** Activate the overlay mode. */
|
|
virtual void showOverlay() = 0;
|
|
|
|
/** Deactivate the overlay mode. */
|
|
virtual void hideOverlay() = 0;
|
|
|
|
/**
|
|
* Returns the pixel format description of the overlay.
|
|
* @see Graphics::PixelFormat
|
|
*/
|
|
virtual Graphics::PixelFormat getOverlayFormat() const = 0;
|
|
|
|
/**
|
|
* Reset the overlay.
|
|
*
|
|
* After calling this method while the overlay mode is active, the user
|
|
* should be seeing only the game graphics. How this is achieved depends
|
|
* on how the backend implements the overlay. Either it sets all pixels of
|
|
* the overlay to be transparent (when alpha blending is used).
|
|
*
|
|
* Or, in case of fake alpha blending, it might just put a copy of the
|
|
* current game graphics screen into the overlay.
|
|
*/
|
|
virtual void clearOverlay() = 0;
|
|
|
|
/**
|
|
* Copy the content of the overlay into a buffer provided by the caller.
|
|
* This is only used to implement fake alpha blending.
|
|
*/
|
|
virtual void grabOverlay(OverlayColor *buf, int pitch) = 0;
|
|
|
|
/**
|
|
* Blit a graphics buffer to the overlay.
|
|
* In a sense, this is the reverse of grabOverlay.
|
|
*
|
|
* @note The pitch parameter actually contains the 'pixel pitch', i.e.,
|
|
* the number of pixels per scanline, and not as usual the number of bytes
|
|
* per scanline.
|
|
*
|
|
* @todo Change 'pitch' to be byte and not pixel based
|
|
*
|
|
* @param buf the buffer containing the graphics data source
|
|
* @param pitch the pixel pitch of the buffer (number of pixels in a scanline)
|
|
* @param x the x coordinate of the destination rectangle
|
|
* @param y the y coordinate of the destination rectangle
|
|
* @param w the width of the destination rectangle
|
|
* @param h the height of the destination rectangle
|
|
*
|
|
* @see copyRectToScreen
|
|
* @see grabOverlay
|
|
*/
|
|
virtual void copyRectToOverlay(const OverlayColor *buf, int pitch, int x, int y, int w, int h) = 0;
|
|
|
|
/**
|
|
* Return the height of the overlay.
|
|
* @see getHeight
|
|
*/
|
|
virtual int16 getOverlayHeight() = 0;
|
|
|
|
/**
|
|
* Return the width of the overlay.
|
|
* @see getWidth
|
|
*/
|
|
virtual int16 getOverlayWidth() = 0;
|
|
|
|
//@}
|
|
|
|
|
|
|
|
/** @name Mouse
|
|
* This is the lower level implementation as provided by the
|
|
* backends. The engines should use the Graphics::CursorManager
|
|
* class instead of using it directly.
|
|
*/
|
|
//@{
|
|
|
|
/**
|
|
* Show or hide the mouse cursor.
|
|
*
|
|
* Currently the backend is not required to immediately draw the
|
|
* mouse cursor on showMouse(true).
|
|
*
|
|
* TODO: We might want to reconsider this fact,
|
|
* check Graphics::CursorManager::showMouse for some details about
|
|
* this.
|
|
*
|
|
* @see Graphics::CursorManager::showMouse
|
|
*/
|
|
virtual bool showMouse(bool visible) = 0;
|
|
|
|
/**
|
|
* Move ("warp") the mouse cursor to the specified position in virtual
|
|
* screen coordinates.
|
|
* @param x the new x position of the mouse
|
|
* @param y the new y position of the mouse
|
|
*/
|
|
virtual void warpMouse(int x, int y) = 0;
|
|
|
|
/**
|
|
* Set the bitmap used for drawing the cursor.
|
|
*
|
|
* @param buf the pixmap data to be used
|
|
* @param w width of the mouse cursor
|
|
* @param h height of the mouse cursor
|
|
* @param hotspotX horizontal offset from the left side to the hotspot
|
|
* @param hotspotY vertical offset from the top side to the hotspot
|
|
* @param keycolor transparency color value. This should not exceed the maximum color value of the specified format.
|
|
* In case it does the behavior is undefined. The backend might just error out or simply ignore the
|
|
* value. (The SDL backend will just assert to prevent abuse of this).
|
|
* @param cursorTargetScale scale factor which cursor is designed for
|
|
* @param format pointer to the pixel format which cursor graphic uses (0 means CLUT8)
|
|
*/
|
|
virtual void setMouseCursor(const byte *buf, uint w, uint h, int hotspotX, int hotspotY, uint32 keycolor, int cursorTargetScale = 1, const Graphics::PixelFormat *format = NULL) = 0;
|
|
|
|
/**
|
|
* Replace the specified range of cursor the palette with new colors.
|
|
* The palette entries from 'start' till (start+num-1) will be replaced - so
|
|
* a full palette update is accomplished via start=0, num=256.
|
|
*
|
|
* Backends which implement it should have kFeatureCursorHasPalette flag set
|
|
*
|
|
* @see setPalette
|
|
* @see kFeatureCursorHasPalette
|
|
*/
|
|
virtual void setCursorPalette(const byte *colors, uint start, uint num) {}
|
|
|
|
/**
|
|
* Disable or enable cursor palette.
|
|
*
|
|
* Backends which implement it should have kFeatureCursorHasPalette flag set
|
|
*
|
|
* @param disable True to disable, false to enable.
|
|
*
|
|
* @see setPalette
|
|
* @see kFeatureCursorHasPalette
|
|
*/
|
|
virtual void disableCursorPalette(bool disable) {}
|
|
|
|
//@}
|
|
|
|
|
|
|
|
/** @name Events and Time */
|
|
//@{
|
|
|
|
/** Get the number of milliseconds since the program was started. */
|
|
virtual uint32 getMillis() = 0;
|
|
|
|
/** Delay/sleep for the specified amount of milliseconds. */
|
|
virtual void delayMillis(uint msecs) = 0;
|
|
|
|
/**
|
|
* Get the current time and date, in the local timezone.
|
|
* Corresponds on many systems to the combination of time()
|
|
* and localtime().
|
|
*/
|
|
virtual void getTimeAndDate(TimeDate &t) const = 0;
|
|
|
|
/**
|
|
* Return the timer manager singleton. For more information, refer
|
|
* to the TimerManager documentation.
|
|
*/
|
|
virtual Common::TimerManager *getTimerManager() = 0;
|
|
|
|
/**
|
|
* Return the event manager singleton. For more information, refer
|
|
* to the EventManager documentation.
|
|
*/
|
|
virtual Common::EventManager *getEventManager() = 0;
|
|
|
|
/**
|
|
* Register hardware keys with keymapper
|
|
*
|
|
* @return HardwareKeySet with all keys and recommended mappings
|
|
*
|
|
* See keymapper documentation for further reference.
|
|
*/
|
|
virtual Common::HardwareKeySet *getHardwareKeySet() { return 0; }
|
|
|
|
//@}
|
|
|
|
|
|
|
|
/**
|
|
* @name Mutex handling
|
|
* Historically, the OSystem API used to have a method which allowed
|
|
* creating threads. Hence mutex support was needed for thread syncing.
|
|
* To ease portability, though, we decided to remove the threading API.
|
|
* Instead, we now use timers (see setTimerCallback() and Common::Timer).
|
|
* But since those may be implemented using threads (and in fact, that's
|
|
* how our primary backend, the SDL one, does it on many systems), we
|
|
* still have to do mutex syncing in our timer callbacks.
|
|
* In addition, the sound mixer uses a mutex in case the backend runs it
|
|
* from a dedicated thread (as e.g. the SDL backend does).
|
|
*
|
|
* Hence backends which do not use threads to implement the timers simply
|
|
* can use dummy implementations for these methods.
|
|
*/
|
|
//@{
|
|
|
|
typedef struct OpaqueMutex *MutexRef;
|
|
|
|
/**
|
|
* Create a new mutex.
|
|
* @return the newly created mutex, or 0 if an error occured.
|
|
*/
|
|
virtual MutexRef createMutex() = 0;
|
|
|
|
/**
|
|
* Lock the given mutex.
|
|
*
|
|
* @note ScummVM code assumes that the mutex implementation supports
|
|
* recursive locking. That is, a thread may lock a mutex twice w/o
|
|
* deadlocking. In case of a multilock, the mutex has to be unlocked
|
|
* as many times as it was locked befored it really becomes unlocked.
|
|
*
|
|
* @param mutex the mutex to lock.
|
|
*/
|
|
virtual void lockMutex(MutexRef mutex) = 0;
|
|
|
|
/**
|
|
* Unlock the given mutex.
|
|
* @param mutex the mutex to unlock.
|
|
*/
|
|
virtual void unlockMutex(MutexRef mutex) = 0;
|
|
|
|
/**
|
|
* Delete the given mutex. Make sure the mutex is unlocked before you delete it.
|
|
* If you delete a locked mutex, the behavior is undefined, in particular, your
|
|
* program may crash.
|
|
* @param mutex the mutex to delete.
|
|
*/
|
|
virtual void deleteMutex(MutexRef mutex) = 0;
|
|
|
|
//@}
|
|
|
|
|
|
|
|
/** @name Sound */
|
|
//@{
|
|
|
|
/**
|
|
* Return the audio mixer. For more information, refer to the
|
|
* Audio::Mixer documentation.
|
|
*/
|
|
virtual Audio::Mixer *getMixer() = 0;
|
|
|
|
//@}
|
|
|
|
|
|
|
|
/**
|
|
* @name Audio CD
|
|
* The methods in this group deal with Audio CD playback.
|
|
* The default implementation simply does nothing.
|
|
* This is the lower level implementation as provided by the
|
|
* backends. The engines should use the Audio::AudioCDManager
|
|
* class instead of using it directly.
|
|
*/
|
|
//@{
|
|
|
|
/**
|
|
* Initialise the specified CD drive for audio playback.
|
|
* @return true if the CD drive was inited succesfully
|
|
*/
|
|
virtual bool openCD(int drive);
|
|
|
|
/**
|
|
* Poll CD status.
|
|
* @return true if CD audio is playing
|
|
*/
|
|
virtual bool pollCD();
|
|
|
|
/**
|
|
* Start audio CD playback.
|
|
* @param track the track to play.
|
|
* @param num_loops how often playback should be repeated (-1 = infinitely often).
|
|
* @param start_frame the frame at which playback should start (75 frames = 1 second).
|
|
* @param duration the number of frames to play.
|
|
*/
|
|
virtual void playCD(int track, int num_loops, int start_frame, int duration) {}
|
|
|
|
/**
|
|
* Stop audio CD playback.
|
|
*/
|
|
virtual void stopCD() {}
|
|
|
|
/**
|
|
* Update cdrom audio status.
|
|
*/
|
|
virtual void updateCD() {}
|
|
|
|
//@}
|
|
|
|
|
|
|
|
/** @name Miscellaneous */
|
|
//@{
|
|
/** Quit (exit) the application. */
|
|
virtual void quit() = 0;
|
|
|
|
/**
|
|
* Set a window caption or any other comparable status display to the
|
|
* given value. The caption must be a pure ISO LATIN 1 string. Passing a
|
|
* string with a different encoding may lead to unexpected behavior,
|
|
* even crashes.
|
|
*
|
|
* @param caption the window caption to use, as an ISO LATIN 1 string
|
|
*/
|
|
virtual void setWindowCaption(const char *caption) {}
|
|
|
|
/**
|
|
* Display a message in an 'on screen display'. That is, display it in a
|
|
* fashion where it is visible on or near the screen (e.g. in a transparent
|
|
* rectangle over the regular screen content; or in a message box beneath
|
|
* it; etc.).
|
|
*
|
|
* Currently, only pure ASCII messages can be expected to show correctly.
|
|
*
|
|
* @note There is a default implementation which uses a TimedMessageDialog
|
|
* to display the message. Hence implementing this is optional.
|
|
*
|
|
* @param msg the message to display on screen
|
|
*/
|
|
virtual void displayMessageOnOSD(const char *msg) = 0;
|
|
|
|
/**
|
|
* Return the SaveFileManager, used to store and load savestates
|
|
* and other modifiable persistent game data. For more information,
|
|
* refer to the SaveFileManager documentation.
|
|
*/
|
|
virtual Common::SaveFileManager *getSavefileManager() = 0;
|
|
|
|
/**
|
|
* Returns the FilesystemFactory object, depending on the current architecture.
|
|
*
|
|
* @return the FSNode factory for the current architecture
|
|
*/
|
|
virtual FilesystemFactory *getFilesystemFactory() = 0;
|
|
|
|
/**
|
|
* Add system specific Common::Archive objects to the given SearchSet.
|
|
* E.g. on Unix the dir corresponding to DATA_PATH (if set), or on
|
|
* Mac OS X the 'Resource' dir in the app bundle.
|
|
*
|
|
* @todo Come up with a better name. This one sucks.
|
|
*
|
|
* @param s the SearchSet to which the system specific dirs, if any, are added
|
|
* @param priority the priority with which those dirs are added
|
|
*/
|
|
virtual void addSysArchivesToSearchSet(Common::SearchSet &s, int priority = 0) {}
|
|
|
|
/**
|
|
* Open the default config file for reading, by returning a suitable
|
|
* ReadStream instance. It is the callers responsiblity to delete
|
|
* the stream after use.
|
|
*/
|
|
virtual Common::SeekableReadStream *createConfigReadStream() = 0;
|
|
|
|
/**
|
|
* Open the default config file for writing, by returning a suitable
|
|
* WriteStream instance. It is the callers responsiblity to delete
|
|
* the stream after use.
|
|
*
|
|
* May return 0 to indicate that writing to config file is not possible.
|
|
*/
|
|
virtual Common::WriteStream *createConfigWriteStream() = 0;
|
|
|
|
//@}
|
|
};
|
|
|
|
|
|
/** The global OSystem instance. Initialised in main(). */
|
|
extern OSystem *g_system;
|
|
|
|
#endif
|