2011-12-14 00:23:17 +03:00
# pragma once
# include "../lib/int3.h"
2015-01-19 21:08:19 +02:00
# include "gui/Geometries.h"
2012-12-01 09:30:52 +03:00
# include "SDL.h"
2009-04-16 14:14:13 +03:00
2009-04-15 17:03:31 +03:00
/*
* mapHandler . h , part of VCMI engine
*
* Authors : listed in file AUTHORS in main folder
*
* License : GNU General Public License v2 .0 or later
* Full text of license available in license . txt file , in main folder
*
2009-04-16 14:14:13 +03:00
*/
class CGObjectInstance ;
class CGHeroInstance ;
2015-02-09 17:03:24 +02:00
class CGBoat ;
2012-11-03 22:31:16 +03:00
class CMap ;
2009-04-16 14:14:13 +03:00
class CGDefInfo ;
class CDefHandler ;
struct TerrainTile ;
struct SDL_Surface ;
2015-01-17 00:44:40 +02:00
struct SDL_Rect ;
2009-06-17 19:46:16 +03:00
class CDefEssential ;
2015-01-31 00:37:28 +02:00
class CFadeAnimation ;
2015-02-09 17:03:24 +02:00
class PlayerColor ;
2009-04-16 14:14:13 +03:00
2015-01-13 21:57:41 +02:00
enum class EWorldViewIcon
{
TOWN = 0 ,
HERO ,
ARTIFACT ,
TELEPORT ,
GATE ,
MINE_WOOD ,
MINE_MERCURY ,
MINE_STONE ,
MINE_SULFUR ,
MINE_CRYSTAL ,
MINE_GEM ,
MINE_GOLD ,
RES_WOOD ,
RES_MERCURY ,
RES_STONE ,
RES_SULFUR ,
RES_CRYSTAL ,
RES_GEM ,
RES_GOLD ,
} ;
2015-01-28 21:07:57 +02:00
enum class EMapObjectFadingType
{
NONE ,
IN ,
OUT
} ;
2015-01-31 00:37:28 +02:00
enum class EMapAnimRedrawStatus
{
OK ,
REDRAW_REQUESTED // map blitter requests quick redraw due to current animation
} ;
2015-01-28 21:07:57 +02:00
struct TerrainTileObject
{
const CGObjectInstance * obj ;
SDL_Rect rect ;
2015-01-31 00:37:28 +02:00
int fadeAnimKey ;
2015-01-28 21:07:57 +02:00
2015-01-31 00:37:28 +02:00
TerrainTileObject ( const CGObjectInstance * obj_ , SDL_Rect rect_ ) ;
~ TerrainTileObject ( ) ;
2015-01-28 21:07:57 +02:00
} ;
2009-04-16 14:14:13 +03:00
struct TerrainTile2
{
2009-12-22 23:53:50 +02:00
SDL_Surface * terbitmap ; //bitmap of terrain
2009-04-16 14:14:13 +03:00
2015-01-28 21:07:57 +02:00
std : : vector < TerrainTileObject > objects ; //pointers to objects being on this tile with rects to be easier to blit this tile on screen
2009-04-16 14:14:13 +03:00
TerrainTile2 ( ) ;
} ;
2015-01-18 19:53:40 +02:00
struct MapDrawingInfo
{
bool scaled ;
int3 & topTile ; // top-left tile in viewport [in tiles]
const std : : vector < std : : vector < std : : vector < ui8 > > > * visibilityMap ;
SDL_Rect * drawBounds ; // map rect drawing bounds on screen
CDefHandler * iconsDef ; // holds overlay icons for world view mode
float scale ; // map scale for world view mode (only if scaled == true)
bool otherheroAnim ;
ui8 anim ;
ui8 heroAnim ;
int3 movement ; // used for smooth map movement
bool puzzleMode ;
int3 grailPos ; // location of grail for puzzle mode [in tiles]
MapDrawingInfo ( int3 & topTile_ , const std : : vector < std : : vector < std : : vector < ui8 > > > * visibilityMap_ , SDL_Rect * drawBounds_ , CDefHandler * iconsDef_ = nullptr )
: scaled ( false ) ,
topTile ( topTile_ ) ,
visibilityMap ( visibilityMap_ ) ,
drawBounds ( drawBounds_ ) ,
iconsDef ( iconsDef_ ) ,
scale ( 1.0f ) ,
otherheroAnim ( false ) ,
anim ( 0u ) ,
heroAnim ( 0u ) ,
movement ( int3 ( ) ) ,
puzzleMode ( false ) ,
grailPos ( int3 ( ) )
{ }
ui8 getHeroAnim ( ) const { return otherheroAnim ? anim : heroAnim ; }
} ;
2009-04-16 14:14:13 +03:00
template < typename T > class PseudoV
{
public :
2013-12-20 16:07:58 +03:00
PseudoV ( ) : offset ( 0 ) { }
PseudoV ( std : : vector < T > & src , int rest , int before , int after , const T & fill ) : offset ( before )
2009-04-16 14:14:13 +03:00
{
2009-06-12 06:26:41 +03:00
inver . resize ( before + rest + after ) ;
for ( int i = 0 ; i < before ; i + + )
2009-04-16 14:14:13 +03:00
inver [ i ] = fill ;
for ( int i = 0 ; i < src . size ( ) ; i + + )
inver [ offset + i ] = src [ i ] ;
2009-06-12 06:26:41 +03:00
for ( int i = src . size ( ) ; i < src . size ( ) + after ; i + + )
2009-04-16 14:14:13 +03:00
inver [ offset + i ] = fill ;
}
inline T & operator [ ] ( const int & n )
{
return inver [ n + offset ] ;
}
inline const T & operator [ ] ( const int & n ) const
{
return inver [ n + offset ] ;
}
2009-06-12 06:26:41 +03:00
void resize ( int rest , int before , int after )
2009-04-16 14:14:13 +03:00
{
2009-06-12 06:26:41 +03:00
inver . resize ( before + rest + after ) ;
offset = before ;
2009-04-16 14:14:13 +03:00
}
int size ( ) const
{
return inver . size ( ) ;
}
2013-12-20 16:07:58 +03:00
private :
int offset ;
std : : vector < T > inver ;
2009-04-16 14:14:13 +03:00
} ;
class CMapHandler
{
2015-01-13 21:57:41 +02:00
enum class EMapCacheType
{
2015-01-19 21:08:19 +02:00
TERRAIN , TERRAIN_CUSTOM , OBJECTS , ROADS , RIVERS , FOW , HEROES , HERO_FLAGS , FRAME
2015-01-13 21:57:41 +02:00
} ;
/// temporarily caches rescaled sdl surfaces for map world view redrawing
class CMapCache
{
2015-01-15 18:17:26 +02:00
std : : map < EMapCacheType , std : : map < intptr_t , SDL_Surface * > > data ;
2015-01-13 21:57:41 +02:00
float worldViewCachedScale ;
public :
/// destroys all cached data (frees surfaces)
void discardWorldViewCache ( ) ;
/// updates scale and determines if currently cached data is still valid
void updateWorldViewScale ( float scale ) ;
2015-01-15 18:17:26 +02:00
void removeFromWorldViewCache ( EMapCacheType type , intptr_t key ) ;
2015-01-15 01:22:20 +02:00
/// asks for cached data; @returns cached surface or nullptr if data is not in cache
2015-01-15 18:17:26 +02:00
SDL_Surface * requestWorldViewCache ( EMapCacheType type , intptr_t key ) ;
2015-01-13 21:57:41 +02:00
/// asks for cached data; @returns cached data if found, new scaled surface otherwise
2015-01-15 18:17:26 +02:00
SDL_Surface * requestWorldViewCacheOrCreate ( EMapCacheType type , intptr_t key , SDL_Surface * fullSurface , float scale ) ;
SDL_Surface * cacheWorldViewEntry ( EMapCacheType type , intptr_t key , SDL_Surface * entry ) ;
2015-01-17 00:44:40 +02:00
intptr_t genKey ( intptr_t realPtr , ui8 mod ) ;
2015-01-13 21:57:41 +02:00
} ;
2015-02-09 17:03:24 +02:00
/// helper struct to pass around resolved bitmaps of an object; surfaces can be nullptr if object doesn't have bitmap of that type
struct AnimBitmapHolder
{
SDL_Surface * objBitmap ; // main object bitmap
SDL_Surface * flagBitmap ; // flag bitmap for the object (probably only for heroes and boats with heroes)
bool isMoving ; // indicates if the object is moving (again, heroes/boats only)
AnimBitmapHolder ( SDL_Surface * objBitmap_ = nullptr , SDL_Surface * flagBitmap_ = nullptr , bool moving = false )
: objBitmap ( objBitmap_ ) ,
flagBitmap ( flagBitmap_ ) ,
isMoving ( moving )
{ }
} ;
2015-01-13 21:57:41 +02:00
2015-01-18 19:53:40 +02:00
class CMapBlitter
2015-01-28 21:07:57 +02:00
{
2015-01-18 19:53:40 +02:00
protected :
2015-02-09 17:03:24 +02:00
static constexpr int FRAMES_PER_MOVE_ANIM_GROUP = 8 ;
2015-01-19 21:08:19 +02:00
CMapHandler * parent ; // ptr to enclosing map handler; generally for legacy reasons, probably could/should be refactored out of here
int tileSize ; // size of a tile drawn on map [in pixels]
int halfTileSizeCeil ; // half of the tile size, rounded up
int3 tileCount ; // number of tiles in current viewport
int3 topTile ; // top-left tile of the viewport
int3 initPos ; // starting drawing position [in pixels]
int3 pos ; // current position [in tiles]
int3 realPos ; // current position [in pixels]
Rect realTileRect ; // default rect based on current pos: [realPos.x, realPos.y, tileSize, tileSize]
Rect defaultTileRect ; // default rect based on 0: [0, 0, tileSize, tileSize]
const MapDrawingInfo * info ; // data for drawing passed from outside
/// general drawing method, called internally by more specialized ones
virtual void drawElement ( EMapCacheType cacheType , SDL_Surface * sourceSurf , SDL_Rect * sourceRect ,
SDL_Surface * targetSurf , SDL_Rect * destRect , bool alphaBlit = false , ui8 rotationInfo = 0u ) const = 0 ;
// first drawing pass
/// draws terrain bitmap (or custom bitmap if applicable) on current tile
virtual void drawTileTerrain ( SDL_Surface * targetSurf , const TerrainTile & tinfo , const TerrainTile2 & tile ) const ;
/// draws a river segment on current tile
virtual void drawRiver ( SDL_Surface * targetSurf , const TerrainTile & tinfo ) const ;
/// draws a road segment on current tile
virtual void drawRoad ( SDL_Surface * targetSurf , const TerrainTile & tinfo , const TerrainTile * tinfoUpper ) const ;
/// draws all objects on current tile (higher-level logic, unlike other draw*** methods)
virtual void drawObjects ( SDL_Surface * targetSurf , const TerrainTile2 & tile ) const ;
2015-02-09 17:03:24 +02:00
virtual void drawObject ( SDL_Surface * targetSurf , SDL_Surface * sourceSurf , SDL_Rect * sourceRect , bool moving ) const ;
2015-01-19 21:08:19 +02:00
virtual void drawHeroFlag ( SDL_Surface * targetSurf , SDL_Surface * sourceSurf , SDL_Rect * sourceRect , SDL_Rect * destRect , bool moving ) const ;
// second drawing pass
/// current tile: draws overlay over the map, used to draw world view icons
virtual void drawTileOverlay ( SDL_Surface * targetSurf , const TerrainTile2 & tile ) const = 0 ;
/// draws fog of war on current tile
virtual void drawFow ( SDL_Surface * targetSurf ) const ;
/// draws map border frame on current position
virtual void drawFrame ( SDL_Surface * targetSurf ) const ;
// third drawing pass
/// custom post-processing, if needed (used by puzzle view)
virtual void postProcessing ( SDL_Surface * targetSurf ) const { }
// misc methods
/// initializes frame-drawing (called at the start of every redraw)
virtual void init ( const MapDrawingInfo * drawingInfo ) = 0 ;
/// calculates clip region for map viewport
virtual SDL_Rect clip ( SDL_Surface * targetSurf ) const = 0 ;
virtual ui8 getHeroFrameNum ( ui8 dir , bool isMoving ) const ;
///returns appropriate bitmap and info if alpha blitting is necessary
virtual std : : pair < SDL_Surface * , bool > getVisBitmap ( ) const ;
virtual ui8 getPhaseShift ( const CGObjectInstance * object ) const ;
virtual bool canDrawObject ( const CGObjectInstance * obj ) const ;
virtual bool canDrawCurrentTile ( ) const ;
2015-01-28 21:07:57 +02:00
2015-02-09 17:03:24 +02:00
// internal helper methods to choose correct bitmap(s) for object; called internally by findObjectBitmap
AnimBitmapHolder findHeroBitmap ( const CGHeroInstance * hero , int anim ) const ;
AnimBitmapHolder findBoatBitmap ( const CGBoat * hero , int anim ) const ;
SDL_Surface * findFlagBitmap ( const CGHeroInstance * obj , int anim , const PlayerColor * color , int indexOffset ) const ;
SDL_Surface * findHeroFlagBitmap ( const CGHeroInstance * obj , int anim , const PlayerColor * color , int indexOffset ) const ;
SDL_Surface * findBoatFlagBitmap ( const CGBoat * obj , int anim , const PlayerColor * color , int indexOffset , ui8 dir ) const ;
SDL_Surface * findFlagBitmapInternal ( const CDefEssential * def , int anim , int indexOffset , ui8 dir , bool moving ) const ;
int findAnimIndexByGroup ( const CDefEssential * def , int groupNum ) const ;
2015-01-18 19:53:40 +02:00
public :
CMapBlitter ( CMapHandler * p ) : parent ( p ) { }
virtual ~ CMapBlitter ( ) { }
2015-01-19 21:08:19 +02:00
void blit ( SDL_Surface * targetSurf , const MapDrawingInfo * info ) ;
2015-02-09 17:03:24 +02:00
/// helper method that chooses correct bitmap(s) for given object
AnimBitmapHolder findObjectBitmap ( const CGObjectInstance * obj , int anim ) const ;
2015-01-19 21:08:19 +02:00
2015-01-18 19:53:40 +02:00
} ;
class CMapNormalBlitter : public CMapBlitter
{
protected :
2015-01-19 21:08:19 +02:00
void drawElement ( EMapCacheType cacheType , SDL_Surface * sourceSurf , SDL_Rect * sourceRect ,
SDL_Surface * targetSurf , SDL_Rect * destRect , bool alphaBlit = false , ui8 rotationInfo = 0u ) const override ;
void drawTileOverlay ( SDL_Surface * targetSurf , const TerrainTile2 & tile ) const override { }
void init ( const MapDrawingInfo * info ) override ;
SDL_Rect clip ( SDL_Surface * targetSurf ) const override ;
2015-01-18 19:53:40 +02:00
public :
CMapNormalBlitter ( CMapHandler * parent ) ;
virtual ~ CMapNormalBlitter ( ) { }
} ;
class CMapWorldViewBlitter : public CMapBlitter
{
protected :
2015-01-19 21:08:19 +02:00
void drawElement ( EMapCacheType cacheType , SDL_Surface * sourceSurf , SDL_Rect * sourceRect ,
SDL_Surface * targetSurf , SDL_Rect * destRect , bool alphaBlit = false , ui8 rotationInfo = 0u ) const override ;
void drawTileOverlay ( SDL_Surface * targetSurf , const TerrainTile2 & tile ) const override ;
void drawHeroFlag ( SDL_Surface * targetSurf , SDL_Surface * sourceSurf , SDL_Rect * sourceRect , SDL_Rect * destRect , bool moving ) const override ;
2015-02-09 17:03:24 +02:00
void drawObject ( SDL_Surface * targetSurf , SDL_Surface * sourceSurf , SDL_Rect * sourceRect , bool moving ) const override ;
2015-01-19 21:08:19 +02:00
void drawFrame ( SDL_Surface * targetSurf ) const override { }
void init ( const MapDrawingInfo * info ) override ;
SDL_Rect clip ( SDL_Surface * targetSurf ) const override ;
2015-02-09 17:03:24 +02:00
// ui8 getHeroFrameNum(ui8 dir, bool isMoving) const override { return 0u; }
2015-01-19 21:08:19 +02:00
ui8 getPhaseShift ( const CGObjectInstance * object ) const override { return 0u ; }
void drawScaledRotatedElement ( EMapCacheType type , SDL_Surface * baseSurf , SDL_Surface * targetSurf , ui8 rotation ,
float scale , SDL_Rect * dstRect , SDL_Rect * srcRect = nullptr ) const ;
void calculateWorldViewCameraPos ( ) ;
2015-01-18 19:53:40 +02:00
public :
CMapWorldViewBlitter ( CMapHandler * parent ) ;
virtual ~ CMapWorldViewBlitter ( ) { }
} ;
2015-01-19 21:08:19 +02:00
class CMapPuzzleViewBlitter : public CMapNormalBlitter
{
std : : vector < int > unblittableObjects ;
void drawObjects ( SDL_Surface * targetSurf , const TerrainTile2 & tile ) const override ;
void drawFow ( SDL_Surface * targetSurf ) const override { } // skipping FoW in puzzle view
void postProcessing ( SDL_Surface * targetSurf ) const override ;
bool canDrawObject ( const CGObjectInstance * obj ) const override ;
bool canDrawCurrentTile ( ) const override { return true ; }
public :
CMapPuzzleViewBlitter ( CMapHandler * parent ) ;
} ;
2015-01-18 19:53:40 +02:00
2015-01-13 21:57:41 +02:00
CMapCache cache ;
2015-01-18 19:53:40 +02:00
CMapBlitter * normalBlitter ;
CMapBlitter * worldViewBlitter ;
2015-01-19 21:08:19 +02:00
CMapBlitter * puzzleViewBlitter ;
2015-01-28 21:07:57 +02:00
2015-01-31 00:37:28 +02:00
std : : map < int , std : : pair < int3 , CFadeAnimation * > > fadeAnims ;
int fadeAnimCounter ;
2015-01-13 21:57:41 +02:00
2015-01-19 21:08:19 +02:00
CMapBlitter * resolveBlitter ( const MapDrawingInfo * info ) const ;
2015-01-31 00:37:28 +02:00
bool updateObjectsFade ( ) ;
2015-02-09 17:03:24 +02:00
bool startObjectFade ( TerrainTileObject & obj , bool in , int3 pos ) ;
2009-04-16 14:14:13 +03:00
public :
2009-05-07 20:20:41 +03:00
PseudoV < PseudoV < PseudoV < TerrainTile2 > > > ttiles ; //informations about map tiles
2009-06-12 06:26:41 +03:00
int3 sizes ; //map size (x = width, y = height, z = number of levels)
2012-10-26 20:51:05 +03:00
const CMap * map ;
2009-06-14 10:02:06 +03:00
// Max number of tiles that will fit in the map screen. Tiles
// can be partial on each edges.
int tilesW ;
int tilesH ;
// size of each side of the frame around the whole map, in tiles
int frameH ;
int frameW ;
// Coord in pixels of the top left corner of the top left tile to
// draw. Values range is [-31..0]. A negative value
// implies that part of the tile won't be displayed.
int offsetX ;
int offsetY ;
2009-06-12 06:26:41 +03:00
2010-08-16 16:51:31 +03:00
//std::set<int> usedHeroes;
2009-04-16 14:14:13 +03:00
std : : vector < std : : vector < SDL_Surface * > > terrainGraphics ; // [terrain id] [view type] [rotation type]
2009-06-17 19:46:16 +03:00
std : : vector < CDefEssential * > roadDefs ;
std : : vector < CDefEssential * > staticRiverDefs ;
2009-04-16 14:14:13 +03:00
2011-12-14 00:23:17 +03:00
std : : vector < std : : vector < std : : vector < ui8 > > > hideBitmap ; //specifies number of graphic that should be used to fully hide a tile
2013-02-19 01:37:22 +03:00
mutable std : : map < const CGObjectInstance * , ui8 > animationPhase ;
2009-05-07 20:20:41 +03:00
CMapHandler ( ) ; //c-tor
~ CMapHandler ( ) ; //d-tor
2009-04-16 14:14:13 +03:00
2010-02-21 17:03:30 +02:00
void getTerrainDescr ( const int3 & pos , std : : string & out , bool terName ) ; //if tername == false => empty string when tile is clear
2009-04-16 14:14:13 +03:00
CGObjectInstance * createObject ( int id , int subid , int3 pos , int owner = 254 ) ; //creates a new object with a certain id and subid
2015-01-28 21:07:57 +02:00
bool printObject ( const CGObjectInstance * obj , bool fadein = false ) ; //puts appropriate things to ttiles, so obj will be visible on map
bool hideObject ( const CGObjectInstance * obj , bool fadeout = false ) ; //removes appropriate things from ttiles, so obj will be no longer visible on map (but still will exist)
2015-01-31 19:36:57 +02:00
bool removeObject ( CGObjectInstance * obj , bool fadeout = false ) ; //removes object from each place in VCMI (I hope)
2009-04-16 14:14:13 +03:00
void init ( ) ;
void calculateBlockedPos ( ) ;
void initObjectRects ( ) ;
void borderAndTerrainBitmapInit ( ) ;
void roadsRiverTerrainInit ( ) ;
void prepareFOWDefs ( ) ;
2015-01-31 00:37:28 +02:00
EMapAnimRedrawStatus drawTerrainRectNew ( SDL_Surface * targetSurface , const MapDrawingInfo * info , bool redrawOnlyAnim = false ) ;
2009-04-16 14:14:13 +03:00
void updateWater ( ) ;
void validateRectTerr ( SDL_Rect * val , const SDL_Rect * ext ) ; //terrainRect helper
2011-12-14 00:23:17 +03:00
static ui8 getDir ( const int3 & a , const int3 & b ) ; //returns direction number in range 0 - 7 (0 is left top, clockwise) [direction: form a to b]
2015-02-09 17:03:24 +02:00
/// determines if the map is ready to handle new hero movement (not available during fading animations)
bool canStartHeroMovement ( ) ;
2009-04-16 14:14:13 +03:00
2015-01-13 21:57:41 +02:00
void discardWorldViewCache ( ) ;
2014-06-24 02:26:36 +03:00
2015-01-13 21:57:41 +02:00
static bool compareObjectBlitOrder ( const CGObjectInstance * a , const CGObjectInstance * b ) ;
2009-04-16 14:14:13 +03:00
} ;