mirror of
https://github.com/vcmi/vcmi.git
synced 2025-01-04 00:15:53 +02:00
628 lines
20 KiB
C++
628 lines
20 KiB
C++
/*
|
|
* MapRenderer.cpp, 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
|
|
*
|
|
*/
|
|
|
|
#include "StdInc.h"
|
|
#include "MapRenderer.h"
|
|
|
|
#include "MapRendererContext.h"
|
|
#include "mapHandler.h"
|
|
|
|
#include "../CGameInfo.h"
|
|
#include "../render/CAnimation.h"
|
|
#include "../render/Canvas.h"
|
|
#include "../render/IImage.h"
|
|
|
|
#include "../../CCallback.h"
|
|
|
|
#include "../../lib/RiverHandler.h"
|
|
#include "../../lib/RoadHandler.h"
|
|
#include "../../lib/TerrainHandler.h"
|
|
#include "../../lib/mapObjects/CGHeroInstance.h"
|
|
#include "../../lib/mapping/CMap.h"
|
|
#include "../../lib/CPathfinder.h"
|
|
|
|
struct NeighborTilesInfo
|
|
{
|
|
//567
|
|
//3 4
|
|
//012
|
|
std::bitset<8> d;
|
|
|
|
NeighborTilesInfo(const IMapRendererContext & context, const int3 & pos)
|
|
{
|
|
auto checkTile = [&](int dx, int dy)
|
|
{
|
|
return context.isVisible(pos + int3(dx, dy, 0));
|
|
};
|
|
|
|
// sides
|
|
d[1] = checkTile(0, +1);
|
|
d[3] = checkTile(-1, 0);
|
|
d[4] = checkTile(+1, 0);
|
|
d[6] = checkTile(0, -1);
|
|
|
|
// corners - select visible image if either corner or adjacent sides are visible
|
|
d[0] = d[1] || d[3] || checkTile(-1, +1);
|
|
d[2] = d[1] || d[4] || checkTile(+1, +1);
|
|
d[5] = d[3] || d[6] || checkTile(-1, -1);
|
|
d[7] = d[4] || d[6] || checkTile(+1, -1);
|
|
}
|
|
|
|
bool areAllHidden() const
|
|
{
|
|
return d.none();
|
|
}
|
|
|
|
int getBitmapID() const
|
|
{
|
|
//NOTE: some images have unused in VCMI pair (same blockmap but a bit different look)
|
|
// 0-1, 2-3, 4-5, 11-13, 12-14
|
|
static const int visBitmaps[256] = {
|
|
-1, 34, 4, 4, 22, 23, 4, 4, 36, 36, 38, 38, 47, 47, 38, 38, //16
|
|
3, 25, 12, 12, 3, 25, 12, 12, 9, 9, 6, 6, 9, 9, 6, 6, //32
|
|
35, 39, 48, 48, 41, 43, 48, 48, 36, 36, 38, 38, 47, 47, 38, 38, //48
|
|
26, 49, 28, 28, 26, 49, 28, 28, 9, 9, 6, 6, 9, 9, 6, 6, //64
|
|
0, 45, 29, 29, 24, 33, 29, 29, 37, 37, 7, 7, 50, 50, 7, 7, //80
|
|
13, 27, 44, 44, 13, 27, 44, 44, 8, 8, 10, 10, 8, 8, 10, 10, //96
|
|
0, 45, 29, 29, 24, 33, 29, 29, 37, 37, 7, 7, 50, 50, 7, 7, //112
|
|
13, 27, 44, 44, 13, 27, 44, 44, 8, 8, 10, 10, 8, 8, 10, 10, //128
|
|
15, 17, 30, 30, 16, 19, 30, 30, 46, 46, 40, 40, 32, 32, 40, 40, //144
|
|
2, 25, 12, 12, 2, 25, 12, 12, 9, 9, 6, 6, 9, 9, 6, 6, //160
|
|
18, 42, 31, 31, 20, 21, 31, 31, 46, 46, 40, 40, 32, 32, 40, 40, //176
|
|
26, 49, 28, 28, 26, 49, 28, 28, 9, 9, 6, 6, 9, 9, 6, 6, //192
|
|
0, 45, 29, 29, 24, 33, 29, 29, 37, 37, 7, 7, 50, 50, 7, 7, //208
|
|
13, 27, 44, 44, 13, 27, 44, 44, 8, 8, 10, 10, 8, 8, 10, 10, //224
|
|
0, 45, 29, 29, 24, 33, 29, 29, 37, 37, 7, 7, 50, 50, 7, 7, //240
|
|
13, 27, 44, 44, 13, 27, 44, 44, 8, 8, 10, 10, 8, 8, 10, 10 //256
|
|
};
|
|
|
|
return visBitmaps[d.to_ulong()]; // >=0 -> partial hide, <0 - full hide
|
|
}
|
|
};
|
|
|
|
MapObjectsSorter::MapObjectsSorter(const IMapRendererContext & context)
|
|
: context(context)
|
|
{
|
|
}
|
|
|
|
bool MapObjectsSorter::operator()(const ObjectInstanceID & left, const ObjectInstanceID & right) const
|
|
{
|
|
return (*this)(context.getObject(left), context.getObject(right));
|
|
}
|
|
|
|
bool MapObjectsSorter::operator()(const CGObjectInstance * left, const CGObjectInstance * right) const
|
|
{
|
|
//FIXME: remove mh access
|
|
return CGI->mh->compareObjectBlitOrder(left, right);
|
|
}
|
|
|
|
MapTileStorage::MapTileStorage(size_t capacity)
|
|
: animations(capacity)
|
|
{
|
|
}
|
|
|
|
void MapTileStorage::load(size_t index, const std::string & filename)
|
|
{
|
|
auto & terrainAnimations = animations[index];
|
|
|
|
for(auto & entry : terrainAnimations)
|
|
{
|
|
entry = std::make_unique<CAnimation>(filename);
|
|
entry->preload();
|
|
}
|
|
|
|
for(size_t i = 0; i < terrainAnimations[0]->size(); ++i)
|
|
{
|
|
terrainAnimations[1]->getImage(i)->verticalFlip();
|
|
terrainAnimations[3]->getImage(i)->verticalFlip();
|
|
|
|
terrainAnimations[2]->getImage(i)->horizontalFlip();
|
|
terrainAnimations[3]->getImage(i)->horizontalFlip();
|
|
}
|
|
}
|
|
|
|
std::shared_ptr<IImage> MapTileStorage::find(size_t fileIndex, size_t rotationIndex, size_t imageIndex)
|
|
{
|
|
const auto & animation = animations[fileIndex][rotationIndex];
|
|
return animation->getImage(imageIndex);
|
|
}
|
|
|
|
MapRendererTerrain::MapRendererTerrain()
|
|
: storage(VLC->terrainTypeHandler->objects.size())
|
|
{
|
|
for(const auto & terrain : VLC->terrainTypeHandler->objects)
|
|
storage.load(terrain->getIndex(), terrain->tilesFilename);
|
|
}
|
|
|
|
void MapRendererTerrain::renderTile(const IMapRendererContext & context, Canvas & target, const int3 & coordinates)
|
|
{
|
|
const TerrainTile & mapTile = context.getMapTile(coordinates);
|
|
|
|
int32_t terrainIndex = mapTile.terType->getIndex();
|
|
int32_t imageIndex = mapTile.terView;
|
|
int32_t rotationIndex = mapTile.extTileFlags % 4;
|
|
|
|
const auto & image = storage.find(terrainIndex, rotationIndex, imageIndex);
|
|
|
|
if(mapTile.terType->getId() == ETerrainId::LAVA)
|
|
{
|
|
image->shiftPalette(246, 9, context.terrainImageIndex(9));
|
|
}
|
|
|
|
if(mapTile.terType->getId() == ETerrainId::WATER)
|
|
{
|
|
image->shiftPalette(229, 12, context.terrainImageIndex(12));
|
|
image->shiftPalette(242, 14, context.terrainImageIndex(14));
|
|
}
|
|
|
|
target.draw(image, Point(0, 0));
|
|
}
|
|
|
|
MapRendererRiver::MapRendererRiver()
|
|
: storage(VLC->riverTypeHandler->objects.size())
|
|
{
|
|
for(const auto & river : VLC->riverTypeHandler->objects)
|
|
storage.load(river->getIndex(), river->tilesFilename);
|
|
}
|
|
|
|
void MapRendererRiver::renderTile(const IMapRendererContext & context, Canvas & target, const int3 & coordinates)
|
|
{
|
|
const TerrainTile & mapTile = context.getMapTile(coordinates);
|
|
|
|
if(mapTile.riverType->getId() == River::NO_RIVER)
|
|
return;
|
|
|
|
int32_t terrainIndex = mapTile.riverType->getIndex();
|
|
int32_t imageIndex = mapTile.riverDir;
|
|
int32_t rotationIndex = (mapTile.extTileFlags >> 2) % 4;
|
|
|
|
const auto & image = storage.find(terrainIndex, rotationIndex, imageIndex);
|
|
|
|
if(mapTile.riverType->getId() == River::WATER_RIVER)
|
|
{
|
|
image->shiftPalette(183, 12, context.terrainImageIndex(12));
|
|
image->shiftPalette(195, 6, context.terrainImageIndex(6));
|
|
}
|
|
|
|
if(mapTile.riverType->getId() == River::MUD_RIVER)
|
|
{
|
|
image->shiftPalette(228, 12, context.terrainImageIndex(12));
|
|
image->shiftPalette(183, 6, context.terrainImageIndex(6));
|
|
image->shiftPalette(240, 6, context.terrainImageIndex(6));
|
|
}
|
|
|
|
if(mapTile.riverType->getId() == River::LAVA_RIVER)
|
|
{
|
|
image->shiftPalette(240, 9, context.terrainImageIndex(9));
|
|
}
|
|
|
|
target.draw(image, Point(0, 0));
|
|
}
|
|
|
|
MapRendererRoad::MapRendererRoad()
|
|
: storage(VLC->roadTypeHandler->objects.size())
|
|
{
|
|
for(const auto & road : VLC->roadTypeHandler->objects)
|
|
storage.load(road->getIndex(), road->tilesFilename);
|
|
}
|
|
|
|
void MapRendererRoad::renderTile(const IMapRendererContext & context, Canvas & target, const int3 & coordinates)
|
|
{
|
|
const int3 coordinatesAbove = coordinates - int3(0, 1, 0);
|
|
|
|
if(context.isInMap(coordinatesAbove))
|
|
{
|
|
const TerrainTile & mapTileAbove = context.getMapTile(coordinatesAbove);
|
|
if(mapTileAbove.roadType->getId() != Road::NO_ROAD)
|
|
{
|
|
int32_t terrainIndex = mapTileAbove.roadType->getIndex();
|
|
int32_t imageIndex = mapTileAbove.roadDir;
|
|
int32_t rotationIndex = (mapTileAbove.extTileFlags >> 4) % 4;
|
|
|
|
const auto & image = storage.find(terrainIndex, rotationIndex, imageIndex);
|
|
target.draw(image, Point(0, 0), Rect(0, 16, 32, 16));
|
|
}
|
|
}
|
|
|
|
const TerrainTile & mapTile = context.getMapTile(coordinates);
|
|
if(mapTile.roadType->getId() != Road::NO_ROAD)
|
|
{
|
|
int32_t terrainIndex = mapTile.roadType->getIndex();
|
|
int32_t imageIndex = mapTile.roadDir;
|
|
int32_t rotationIndex = (mapTile.extTileFlags >> 4) % 4;
|
|
|
|
const auto & image = storage.find(terrainIndex, rotationIndex, imageIndex);
|
|
target.draw(image, Point(0, 16), Rect(0, 0, 32, 16));
|
|
}
|
|
}
|
|
|
|
MapRendererBorder::MapRendererBorder()
|
|
{
|
|
animation = std::make_unique<CAnimation>("EDG");
|
|
animation->preload();
|
|
}
|
|
|
|
size_t MapRendererBorder::getIndexForTile(const IMapRendererContext & context, const int3 & tile)
|
|
{
|
|
assert(!context.isInMap(tile));
|
|
|
|
int3 size = context.getMapSize();
|
|
|
|
if(tile.x < -1 || tile.x > size.x || tile.y < -1 || tile.y > size.y)
|
|
return std::abs(tile.x) % 4 + 4 * (std::abs(tile.y) % 4);
|
|
|
|
if(tile.x == -1 && tile.y == -1)
|
|
return 16;
|
|
|
|
if(tile.x == size.x && tile.y == -1)
|
|
return 17;
|
|
|
|
if(tile.x == size.x && tile.y == size.y)
|
|
return 18;
|
|
|
|
if(tile.x == -1 && tile.y == size.y)
|
|
return 19;
|
|
|
|
if(tile.y == -1)
|
|
return 20 + (tile.x % 4);
|
|
|
|
if(tile.x == size.x)
|
|
return 24 + (tile.y % 4);
|
|
|
|
if(tile.y == size.y)
|
|
return 28 + (tile.x % 4);
|
|
|
|
if(tile.x == -1)
|
|
return 32 + (tile.y % 4);
|
|
|
|
//else - visible area, no renderable border
|
|
assert(0);
|
|
return 0;
|
|
}
|
|
|
|
void MapRendererBorder::renderTile(const IMapRendererContext & context, Canvas & target, const int3 & coordinates)
|
|
{
|
|
const auto & image = animation->getImage(getIndexForTile(context, coordinates));
|
|
target.draw(image, Point(0, 0));
|
|
}
|
|
|
|
MapRendererFow::MapRendererFow()
|
|
{
|
|
fogOfWarFullHide = std::make_unique<CAnimation>("TSHRC");
|
|
fogOfWarFullHide->preload();
|
|
fogOfWarPartialHide = std::make_unique<CAnimation>("TSHRE");
|
|
fogOfWarPartialHide->preload();
|
|
|
|
static const std::vector<int> rotations = {22, 15, 2, 13, 12, 16, 28, 17, 20, 19, 7, 24, 26, 25, 30, 32, 27};
|
|
|
|
size_t size = fogOfWarPartialHide->size(0); //group size after next rotation
|
|
|
|
for(const int rotation : rotations)
|
|
{
|
|
fogOfWarPartialHide->duplicateImage(0, rotation, 0);
|
|
auto image = fogOfWarPartialHide->getImage(size, 0);
|
|
image->verticalFlip();
|
|
size++;
|
|
}
|
|
}
|
|
|
|
void MapRendererFow::renderTile(const IMapRendererContext & context, Canvas & target, const int3 & coordinates)
|
|
{
|
|
assert(!context.isVisible(coordinates));
|
|
|
|
const NeighborTilesInfo neighborInfo(context, coordinates);
|
|
|
|
int retBitmapID = neighborInfo.getBitmapID(); // >=0 -> partial hide, <0 - full hide
|
|
if(retBitmapID < 0)
|
|
{
|
|
// generate a number that is predefined for each tile,
|
|
// but appears random to break visible pattern in large areas of fow
|
|
// current approach (use primes as magic numbers for formula) looks to be suitable
|
|
size_t pseudorandomNumber = ((coordinates.x * 997) ^ (coordinates.y * 1009)) / 101;
|
|
size_t imageIndex = pseudorandomNumber % fogOfWarFullHide->size();
|
|
|
|
target.draw(fogOfWarFullHide->getImage(imageIndex), Point(0, 0));
|
|
}
|
|
else
|
|
{
|
|
target.draw(fogOfWarPartialHide->getImage(retBitmapID), Point(0, 0));
|
|
}
|
|
}
|
|
|
|
std::shared_ptr<CAnimation> MapRendererObjects::getBaseAnimation(const CGObjectInstance* obj)
|
|
{
|
|
const auto & info = obj->appearance;
|
|
|
|
//the only(?) invisible object
|
|
if(info->id == Obj::EVENT)
|
|
return std::shared_ptr<CAnimation>();
|
|
|
|
if(info->animationFile.empty())
|
|
{
|
|
logGlobal->warn("Def name for obj (%d,%d) is empty!", info->id, info->subid);
|
|
return std::shared_ptr<CAnimation>();
|
|
}
|
|
|
|
bool generateMovementGroups = (info->id == Obj::BOAT) || (info->id == Obj::HERO);
|
|
|
|
//TODO: relocate to config file?
|
|
// Boat appearance files only contain single, unanimated image
|
|
// proper boat animations are actually in different file
|
|
static const std::vector<std::string> boatAnimations = {
|
|
"AB01_.def", "AB02_.def", "AB03_.def"
|
|
};
|
|
|
|
if (info->id == Obj::BOAT)
|
|
return getAnimation(boatAnimations[info->subid], generateMovementGroups);
|
|
|
|
return getAnimation(info->animationFile, generateMovementGroups);
|
|
}
|
|
|
|
std::shared_ptr<CAnimation> MapRendererObjects::getAnimation(const std::string & filename, bool generateMovementGroups)
|
|
{
|
|
if(animations.count(filename))
|
|
return animations[filename];
|
|
|
|
auto ret = std::make_shared<CAnimation>(filename);
|
|
animations[filename] = ret;
|
|
ret->preload();
|
|
|
|
if(generateMovementGroups)
|
|
{
|
|
ret->createFlippedGroup(1, 13);
|
|
ret->createFlippedGroup(2, 14);
|
|
ret->createFlippedGroup(3, 15);
|
|
|
|
ret->createFlippedGroup(6, 10);
|
|
ret->createFlippedGroup(7, 11);
|
|
ret->createFlippedGroup(8, 12);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
std::shared_ptr<CAnimation> MapRendererObjects::getFlagAnimation(const CGObjectInstance* obj)
|
|
{
|
|
//TODO: relocate to config file?
|
|
static const std::vector<std::string> heroFlags = {
|
|
"AF00", "AF01", "AF02", "AF03", "AF04", "AF05", "AF06", "AF07"
|
|
};
|
|
|
|
//TODO: relocate to config file?
|
|
static const std::vector<std::vector<std::string>> boatFlags = {
|
|
{"ABF01L", "ABF01G", "ABF01R", "ABF01D", "ABF01B", "ABF01P", "ABF01W", "ABF01K"},
|
|
{"ABF02L", "ABF02G", "ABF02R", "ABF02D", "ABF02B", "ABF02P", "ABF02W", "ABF02K"},
|
|
{"ABF03L", "ABF03G", "ABF03R", "ABF03D", "ABF03B", "ABF03P", "ABF03W", "ABF03K"}
|
|
};
|
|
|
|
if(obj->ID == Obj::HERO)
|
|
{
|
|
assert(dynamic_cast<const CGHeroInstance *>(obj) != nullptr);
|
|
assert(obj->tempOwner.isValidPlayer());
|
|
return getAnimation(heroFlags[obj->tempOwner.getNum()], true);
|
|
}
|
|
|
|
if(obj->ID == Obj::BOAT)
|
|
{
|
|
const auto * boat = dynamic_cast<const CGBoat *>(obj);
|
|
assert(boat->subID < boatFlags.size());
|
|
if (boat->hero)
|
|
{
|
|
assert(boat->hero->tempOwner.isValidPlayer());
|
|
return getAnimation(boatFlags[boat->subID][boat->hero->tempOwner.getNum()], true);
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
std::shared_ptr<CAnimation> MapRendererObjects::getOverlayAnimation(const CGObjectInstance * obj)
|
|
{
|
|
if(obj->ID == Obj::BOAT)
|
|
{
|
|
//TODO: relocate to config file?
|
|
// Boats have additional animation with waves around boat
|
|
static const std::vector<std::string> boatAnimations = {
|
|
"ABM01_.def", "ABM02_.def", "ABM03_.def"
|
|
};
|
|
|
|
const auto * boat = dynamic_cast<const CGBoat *>(obj);
|
|
if (boat->hero)
|
|
return getAnimation(boatAnimations[obj->subID], true);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
std::shared_ptr<IImage> MapRendererObjects::getImage(const IMapRendererContext & context, const CGObjectInstance * obj, const std::shared_ptr<CAnimation>& animation) const
|
|
{
|
|
if(!animation)
|
|
return nullptr;
|
|
|
|
size_t groupIndex = context.objectGroupIndex(obj->id);
|
|
|
|
if(animation->size(groupIndex) == 0)
|
|
return nullptr;
|
|
|
|
size_t frameIndex = context.objectImageIndex(obj->id, animation->size(groupIndex));
|
|
|
|
return animation->getImage(frameIndex, groupIndex);
|
|
}
|
|
|
|
void MapRendererObjects::renderImage(const IMapRendererContext & context, Canvas & target, const int3 & coordinates, const CGObjectInstance * object, const std::shared_ptr<IImage>& image)
|
|
{
|
|
if(!image)
|
|
return;
|
|
|
|
auto transparency = static_cast<uint8_t>(std::round(255 * context.objectTransparency(object->id)));
|
|
|
|
if (transparency == 0)
|
|
return;
|
|
|
|
image->setAlpha(transparency);
|
|
image->setFlagColor(object->tempOwner);
|
|
|
|
Point offsetPixels = context.objectImageOffset(object->id, coordinates);
|
|
|
|
if ( offsetPixels.x < image->dimensions().x && offsetPixels.y < image->dimensions().y)
|
|
{
|
|
Point imagePos = image->dimensions() - offsetPixels - Point(32, 32);
|
|
target.draw(image, Point(0, 0), Rect(imagePos, Point(32,32)));
|
|
}
|
|
}
|
|
|
|
void MapRendererObjects::renderObject(const IMapRendererContext & context, Canvas & target, const int3 & coordinates, const CGObjectInstance * instance)
|
|
{
|
|
renderImage(context, target, coordinates, instance, getImage(context, instance, getBaseAnimation(instance)));
|
|
renderImage(context, target, coordinates, instance, getImage(context, instance, getFlagAnimation(instance)));
|
|
renderImage(context, target, coordinates, instance, getImage(context, instance, getOverlayAnimation(instance)));
|
|
}
|
|
|
|
void MapRendererObjects::renderTile(const IMapRendererContext & context, Canvas & target, const int3 & coordinates)
|
|
{
|
|
for(const auto & objectID : context.getObjects(coordinates))
|
|
{
|
|
const auto * objectInstance = context.getObject(objectID);
|
|
|
|
assert(objectInstance);
|
|
if(!objectInstance)
|
|
{
|
|
logGlobal->error("Stray map object that isn't fading");
|
|
continue;
|
|
}
|
|
|
|
renderObject(context, target, coordinates, objectInstance);
|
|
}
|
|
}
|
|
|
|
void MapRendererDebugGrid::renderTile(const IMapRendererContext & context, Canvas & target, const int3 & coordinates)
|
|
{
|
|
if(context.showGrid())
|
|
{
|
|
target.drawLine(Point(0, 0), Point(0, 31), {128, 128, 128, 128}, {128, 128, 128, 128});
|
|
target.drawLine(Point(0, 0), Point(31, 0), {128, 128, 128, 128}, {128, 128, 128, 128});
|
|
}
|
|
}
|
|
|
|
MapRendererPath::MapRendererPath()
|
|
: pathNodes(new CAnimation("ADAG"))
|
|
{
|
|
pathNodes->preload();
|
|
}
|
|
|
|
void MapRendererPath::renderImage(Canvas & target, bool reachableToday, size_t imageIndex)
|
|
{
|
|
const static size_t unreachableTodayOffset = 25;
|
|
|
|
if(reachableToday)
|
|
target.draw(pathNodes->getImage(imageIndex), Point(0, 0));
|
|
else
|
|
target.draw(pathNodes->getImage(imageIndex + unreachableTodayOffset), Point(0, 0));
|
|
}
|
|
|
|
void MapRendererPath::renderImageCross(Canvas & target, bool reachableToday, const int3 & curr)
|
|
{
|
|
renderImage(target, reachableToday, 0);
|
|
}
|
|
|
|
void MapRendererPath::renderImageArrow(Canvas & target, bool reachableToday, const int3 & curr, const int3 & prev, const int3 & next)
|
|
{
|
|
// Vector directions
|
|
// 0 1 2
|
|
// |
|
|
// 3 - 4 - 5
|
|
// |
|
|
// 6 7 8
|
|
//For example:
|
|
// |
|
|
// +->
|
|
// is (directionToArrowIndex[7][5])
|
|
//
|
|
const static size_t directionToArrowIndex[9][9] = {
|
|
{16, 17, 18, 7, 0, 19, 6, 5, 0 },
|
|
{8, 9, 18, 7, 0, 19, 6, 0, 20},
|
|
{8, 1, 10, 7, 0, 19, 0, 21, 20},
|
|
{24, 17, 18, 15, 0, 0, 6, 5, 4 },
|
|
{0, 0, 0, 0, 0, 0, 0, 0, 0 },
|
|
{8, 1, 2, 0, 0, 11, 22, 21, 20},
|
|
{24, 17, 0, 23, 0, 3, 14, 5, 4 },
|
|
{24, 0, 2, 23, 0, 3, 22, 13, 4 },
|
|
{0, 1, 2, 23, 0, 3, 22, 21, 12}
|
|
};
|
|
|
|
size_t enterDirection = (curr.x - next.x + 1) + 3 * (curr.y - next.y + 1);
|
|
size_t leaveDirection = (prev.x - curr.x + 1) + 3 * (prev.y - curr.y + 1);
|
|
size_t imageIndex = directionToArrowIndex[enterDirection][leaveDirection];
|
|
|
|
renderImage(target, reachableToday, imageIndex);
|
|
}
|
|
|
|
void MapRendererPath::renderTile(const IMapRendererContext & context, Canvas & target, const int3 & coordinates)
|
|
{
|
|
const auto & functor = [&](const CGPathNode & node)
|
|
{
|
|
return node.coord == coordinates;
|
|
};
|
|
|
|
const auto * path = context.currentPath();
|
|
if(!path)
|
|
return;
|
|
|
|
const auto & iter = boost::range::find_if(path->nodes, functor);
|
|
|
|
if(iter == path->nodes.end())
|
|
return;
|
|
|
|
bool reachableToday = iter->turns == 0;
|
|
if(iter == path->nodes.begin())
|
|
renderImageCross(target, reachableToday, iter->coord);
|
|
|
|
auto next = iter + 1;
|
|
auto prev = iter - 1;
|
|
|
|
// start of path - current hero location
|
|
if(next == path->nodes.end())
|
|
return;
|
|
|
|
bool pathContinuous = iter->coord.areNeighbours(next->coord) && iter->coord.areNeighbours(prev->coord);
|
|
bool embarking = iter->action == CGPathNode::EMBARK || iter->action == CGPathNode::DISEMBARK;
|
|
|
|
if(pathContinuous && !embarking)
|
|
renderImageArrow(target, reachableToday, iter->coord, prev->coord, next->coord);
|
|
else
|
|
renderImageCross(target, reachableToday, iter->coord);
|
|
}
|
|
|
|
void MapRenderer::renderTile(const IMapRendererContext & context, Canvas & target, const int3 & coordinates)
|
|
{
|
|
if(!context.isInMap(coordinates))
|
|
{
|
|
rendererBorder.renderTile(context, target, coordinates);
|
|
return;
|
|
}
|
|
|
|
const NeighborTilesInfo neighborInfo(context, coordinates);
|
|
|
|
if(!context.isVisible(coordinates) && neighborInfo.areAllHidden())
|
|
{
|
|
rendererFow.renderTile(context, target, coordinates);
|
|
}
|
|
else
|
|
{
|
|
rendererTerrain.renderTile(context, target, coordinates);
|
|
rendererRiver.renderTile(context, target, coordinates);
|
|
rendererRoad.renderTile(context, target, coordinates);
|
|
rendererObjects.renderTile(context, target, coordinates);
|
|
rendererPath.renderTile(context, target, coordinates);
|
|
|
|
if(!context.isVisible(coordinates))
|
|
rendererFow.renderTile(context, target, coordinates);
|
|
}
|
|
rendererDebugGrid.renderTile(context, target, coordinates);
|
|
}
|