2017-07-13 10:26:03 +02:00
|
|
|
/*
|
2023-02-01 16:42:03 +02:00
|
|
|
* CMinimap.cpp, part of VCMI engine
|
2017-07-13 10:26:03 +02:00
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*
|
|
|
|
*/
|
2012-06-13 16:04:06 +03:00
|
|
|
|
2023-02-01 20:42:06 +02:00
|
|
|
#include "StdInc.h"
|
|
|
|
#include "CMinimap.h"
|
2014-07-15 10:14:49 +03:00
|
|
|
|
2023-05-08 14:18:34 +02:00
|
|
|
#include "AdventureMapInterface.h"
|
2014-07-13 20:53:37 +03:00
|
|
|
|
2023-02-01 20:42:06 +02:00
|
|
|
#include "../widgets/Images.h"
|
2014-07-13 20:53:37 +03:00
|
|
|
#include "../CGameInfo.h"
|
|
|
|
#include "../CPlayerInterface.h"
|
|
|
|
#include "../gui/CGuiHandler.h"
|
2023-05-18 22:31:05 +02:00
|
|
|
#include "../gui/MouseButton.h"
|
2023-05-16 14:10:26 +02:00
|
|
|
#include "../gui/WindowHandler.h"
|
2023-02-02 21:15:13 +02:00
|
|
|
#include "../render/Colors.h"
|
2023-04-17 00:09:25 +02:00
|
|
|
#include "../render/Canvas.h"
|
2023-07-31 18:50:55 +02:00
|
|
|
#include "../render/Graphics.h"
|
|
|
|
#include "../renderSDL/SDL_Extensions.h"
|
2023-02-10 15:29:30 +02:00
|
|
|
#include "../windows/InfoWindows.h"
|
2014-07-13 20:53:37 +03:00
|
|
|
|
|
|
|
#include "../../CCallback.h"
|
2024-07-20 14:55:17 +02:00
|
|
|
#include "../../lib/texts/CGeneralTextHandler.h"
|
2023-01-09 01:17:37 +02:00
|
|
|
#include "../../lib/TerrainHandler.h"
|
2014-07-13 20:53:37 +03:00
|
|
|
#include "../../lib/mapObjects/CGHeroInstance.h"
|
2023-02-01 20:42:06 +02:00
|
|
|
#include "../../lib/mapping/CMapDefines.h"
|
2012-06-13 16:04:06 +03:00
|
|
|
|
2023-02-10 15:29:30 +02:00
|
|
|
ColorRGBA CMinimapInstance::getTileColor(const int3 & pos) const
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
|
|
|
const TerrainTile * tile = LOCPLINT->cb->getTile(pos, false);
|
|
|
|
|
|
|
|
// if tile is not visible it will be black on minimap
|
|
|
|
if(!tile)
|
2023-07-31 18:50:55 +02:00
|
|
|
return Colors::BLACK;
|
2012-06-13 16:04:06 +03:00
|
|
|
|
|
|
|
// if object at tile is owned - it will be colored as its owner
|
2014-10-01 15:19:03 +03:00
|
|
|
for (const CGObjectInstance *obj : tile->blockingObjects)
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
2013-03-03 20:06:03 +03:00
|
|
|
PlayerColor player = obj->getOwner();
|
|
|
|
if(player == PlayerColor::NEUTRAL)
|
2023-07-31 18:50:55 +02:00
|
|
|
return graphics->neutralColor;
|
2023-02-10 15:29:30 +02:00
|
|
|
|
2023-08-27 00:35:38 +02:00
|
|
|
if (player.isValidPlayer())
|
2023-07-31 18:50:55 +02:00
|
|
|
return graphics->playerColors[player.getNum()];
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
|
2024-07-13 20:37:13 +02:00
|
|
|
if (tile->blocked() && !tile->visitable())
|
|
|
|
return tile->getTerrain()->minimapBlocked;
|
2012-06-13 16:04:06 +03:00
|
|
|
else
|
2024-07-13 20:37:13 +02:00
|
|
|
return tile->getTerrain()->minimapUnblocked;
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void CMinimapInstance::refreshTile(const int3 &tile)
|
|
|
|
{
|
2023-02-10 15:29:30 +02:00
|
|
|
if (level == tile.z)
|
|
|
|
minimap->drawPoint(Point(tile.x, tile.y), getTileColor(tile));
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
|
2023-02-10 15:29:30 +02:00
|
|
|
void CMinimapInstance::redrawMinimap()
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
|
|
|
int3 mapSizes = LOCPLINT->cb->getMapSize();
|
|
|
|
|
2023-02-10 15:29:30 +02:00
|
|
|
for (int y = 0; y < mapSizes.y; ++y)
|
|
|
|
for (int x = 0; x < mapSizes.x; ++x)
|
|
|
|
minimap->drawPoint(Point(x, y), getTileColor(int3(x, y, level)));
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
CMinimapInstance::CMinimapInstance(CMinimap *Parent, int Level):
|
2015-01-13 21:57:41 +02:00
|
|
|
parent(Parent),
|
2024-07-22 13:00:45 +02:00
|
|
|
minimap(new Canvas(Point(LOCPLINT->cb->getMapSize().x, LOCPLINT->cb->getMapSize().y), CanvasScalingPolicy::IGNORE)),
|
2015-01-13 21:57:41 +02:00
|
|
|
level(Level)
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
|
|
|
pos.w = parent->pos.w;
|
|
|
|
pos.h = parent->pos.h;
|
2023-02-10 15:29:30 +02:00
|
|
|
redrawMinimap();
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
|
2023-06-02 15:42:18 +02:00
|
|
|
void CMinimapInstance::showAll(Canvas & to)
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
2023-06-02 15:42:18 +02:00
|
|
|
to.drawScaled(*minimap, pos.topLeft(), pos.dimensions());
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
|
2018-04-07 13:34:11 +02:00
|
|
|
CMinimap::CMinimap(const Rect & position)
|
2023-06-22 21:11:48 +02:00
|
|
|
: CIntObject(LCLICK | SHOW_POPUP | DRAG | MOVE | GESTURE, position.topLeft()),
|
2023-02-10 15:29:30 +02:00
|
|
|
level(0)
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
2024-08-09 17:30:04 +02:00
|
|
|
OBJECT_CONSTRUCTION;
|
2018-04-07 13:34:11 +02:00
|
|
|
|
2024-06-24 03:23:26 +02:00
|
|
|
double maxSideLengthSrc = std::max(LOCPLINT->cb->getMapSize().x, LOCPLINT->cb->getMapSize().y);
|
|
|
|
double maxSideLengthDst = std::max(position.w, position.h);
|
|
|
|
double resize = maxSideLengthSrc / maxSideLengthDst;
|
2023-10-21 16:49:50 +02:00
|
|
|
Point newMinimapSize = Point(LOCPLINT->cb->getMapSize().x/ resize, LOCPLINT->cb->getMapSize().y / resize);
|
|
|
|
Point offset = Point((std::max(newMinimapSize.x, newMinimapSize.y) - newMinimapSize.x) / 2, (std::max(newMinimapSize.x, newMinimapSize.y) - newMinimapSize.y) / 2);
|
|
|
|
|
|
|
|
pos.x += offset.x;
|
|
|
|
pos.y += offset.y;
|
|
|
|
pos.w = newMinimapSize.x;
|
|
|
|
pos.h = newMinimapSize.y;
|
|
|
|
|
|
|
|
aiShield = std::make_shared<CPicture>(ImagePath::builtin("AIShield"), -offset);
|
2018-04-07 13:34:11 +02:00
|
|
|
aiShield->disable();
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
|
2023-02-10 15:29:30 +02:00
|
|
|
int3 CMinimap::pixelToTile(const Point & cursorPos) const
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
|
|
|
// 0 = top-left corner, 1 = bottom-right corner
|
2023-02-10 15:29:30 +02:00
|
|
|
double dx = static_cast<double>(cursorPos.x) / pos.w;
|
|
|
|
double dy = static_cast<double>(cursorPos.y) / pos.h;
|
|
|
|
|
|
|
|
int3 mapSizes = LOCPLINT->cb->getMapSize();
|
|
|
|
|
|
|
|
int tileX(std::round(mapSizes.x * dx));
|
|
|
|
int tileY(std::round(mapSizes.y * dy));
|
|
|
|
|
|
|
|
return int3(tileX, tileY, level);
|
|
|
|
}
|
2012-06-13 16:04:06 +03:00
|
|
|
|
2023-02-10 15:29:30 +02:00
|
|
|
Point CMinimap::tileToPixels(const int3 &tile) const
|
|
|
|
{
|
2012-06-13 16:04:06 +03:00
|
|
|
int3 mapSizes = LOCPLINT->cb->getMapSize();
|
|
|
|
|
2023-02-10 15:29:30 +02:00
|
|
|
double stepX = static_cast<double>(pos.w) / mapSizes.x;
|
|
|
|
double stepY = static_cast<double>(pos.h) / mapSizes.y;
|
|
|
|
|
|
|
|
int x = static_cast<int>(stepX * tile.x);
|
|
|
|
int y = static_cast<int>(stepY * tile.y);
|
|
|
|
|
|
|
|
return Point(x,y);
|
2012-07-08 19:36:20 +03:00
|
|
|
}
|
2012-06-13 16:04:06 +03:00
|
|
|
|
2023-05-31 13:30:24 +02:00
|
|
|
void CMinimap::moveAdvMapSelection(const Point & positionGlobal)
|
2012-07-08 19:36:20 +03:00
|
|
|
{
|
2023-05-31 13:30:24 +02:00
|
|
|
int3 newLocation = pixelToTile(positionGlobal - pos.topLeft());
|
2023-02-23 19:46:41 +02:00
|
|
|
adventureInt->centerOnTile(newLocation);
|
2012-06-13 16:04:06 +03:00
|
|
|
|
2023-05-19 19:03:04 +02:00
|
|
|
if (!(adventureInt->isActive()))
|
2023-05-16 14:10:26 +02:00
|
|
|
GH.windows().totalRedraw(); //redraw this as well as inactive adventure map
|
2012-08-07 14:28:52 +03:00
|
|
|
else
|
|
|
|
redraw();//redraw only this
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
|
2023-05-31 13:30:24 +02:00
|
|
|
void CMinimap::gesturePanning(const Point & initialPosition, const Point & currentPosition, const Point & lastUpdateDistance)
|
|
|
|
{
|
2023-06-01 11:10:51 +02:00
|
|
|
if (pos.isInside(currentPosition))
|
|
|
|
moveAdvMapSelection(currentPosition);
|
2023-05-31 13:30:24 +02:00
|
|
|
}
|
|
|
|
|
2023-07-08 13:33:04 +02:00
|
|
|
void CMinimap::clickPressed(const Point & cursorPosition)
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
2023-07-08 13:33:04 +02:00
|
|
|
moveAdvMapSelection(cursorPosition);
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
|
2023-07-08 13:33:04 +02:00
|
|
|
void CMinimap::showPopupWindow(const Point & cursorPosition)
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
2023-06-11 17:20:10 +02:00
|
|
|
CRClickPopup::createAndPush(CGI->generaltexth->zelp[291].second);
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void CMinimap::hover(bool on)
|
|
|
|
{
|
2018-04-07 13:34:11 +02:00
|
|
|
if(on)
|
2023-05-16 17:34:23 +02:00
|
|
|
GH.statusbar()->write(CGI->generaltexth->zelp[291].first);
|
2012-06-13 16:04:06 +03:00
|
|
|
else
|
2023-05-16 17:34:23 +02:00
|
|
|
GH.statusbar()->clear();
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
|
2023-06-22 21:11:48 +02:00
|
|
|
void CMinimap::mouseDragged(const Point & cursorPosition, const Point & lastUpdateDistance)
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
2023-06-22 21:11:48 +02:00
|
|
|
moveAdvMapSelection(cursorPosition);
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
|
2023-06-02 15:42:18 +02:00
|
|
|
void CMinimap::showAll(Canvas & to)
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
2023-10-21 17:28:23 +02:00
|
|
|
CSDL_Ext::CClipRectGuard guard(to.getInternalSurface(), aiShield->pos);
|
2012-06-13 16:04:06 +03:00
|
|
|
CIntObject::showAll(to);
|
2023-02-23 19:46:41 +02:00
|
|
|
|
2018-04-07 13:34:11 +02:00
|
|
|
if(minimap)
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
|
|
|
int3 mapSizes = LOCPLINT->cb->getMapSize();
|
|
|
|
|
|
|
|
//draw radar
|
2023-01-17 22:01:35 +02:00
|
|
|
Rect radar =
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
2023-02-10 23:29:13 +02:00
|
|
|
screenArea.x * pos.w / mapSizes.x,
|
|
|
|
screenArea.y * pos.h / mapSizes.y,
|
|
|
|
screenArea.w * pos.w / mapSizes.x - 1,
|
|
|
|
screenArea.h * pos.h / mapSizes.y - 1
|
2012-06-13 16:04:06 +03:00
|
|
|
};
|
|
|
|
|
2023-06-02 15:42:18 +02:00
|
|
|
Canvas clippedTarget(to, pos);
|
2023-07-31 18:50:55 +02:00
|
|
|
clippedTarget.drawBorderDashed(radar, Colors::PURPLE);
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMinimap::update()
|
|
|
|
{
|
2018-04-07 13:34:11 +02:00
|
|
|
if(aiShield->recActions & UPDATE) //AI turn is going on. There is no need to update minimap
|
2012-06-13 16:04:06 +03:00
|
|
|
return;
|
|
|
|
|
2024-08-09 17:30:04 +02:00
|
|
|
OBJECT_CONSTRUCTION;
|
2018-04-07 13:34:11 +02:00
|
|
|
minimap = std::make_shared<CMinimapInstance>(this, level);
|
2012-06-15 20:08:19 +03:00
|
|
|
redraw();
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
|
2023-02-23 19:46:41 +02:00
|
|
|
void CMinimap::onMapViewMoved(const Rect & visibleArea, int mapLevel)
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
2023-02-23 19:46:41 +02:00
|
|
|
if (screenArea == visibleArea && level == mapLevel)
|
2023-02-10 15:29:30 +02:00
|
|
|
return;
|
|
|
|
|
2023-02-23 19:46:41 +02:00
|
|
|
screenArea = visibleArea;
|
|
|
|
|
|
|
|
if(level != mapLevel)
|
|
|
|
{
|
|
|
|
level = mapLevel;
|
|
|
|
update();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
redraw();
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void CMinimap::setAIRadar(bool on)
|
|
|
|
{
|
2018-04-07 13:34:11 +02:00
|
|
|
if(on)
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
2018-04-07 13:34:11 +02:00
|
|
|
aiShield->enable();
|
|
|
|
minimap.reset();
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-04-07 13:34:11 +02:00
|
|
|
aiShield->disable();
|
2012-06-13 16:04:06 +03:00
|
|
|
update();
|
|
|
|
}
|
2023-03-22 00:57:08 +02:00
|
|
|
redraw();
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|
|
|
|
|
2023-07-03 18:36:10 +02:00
|
|
|
void CMinimap::updateTiles(const std::unordered_set<int3> & positions)
|
2012-06-13 16:04:06 +03:00
|
|
|
{
|
2018-04-07 13:34:11 +02:00
|
|
|
if(minimap)
|
2023-04-16 00:48:49 +02:00
|
|
|
{
|
|
|
|
for (auto const & tile : positions)
|
|
|
|
minimap->refreshTile(tile);
|
|
|
|
}
|
2023-03-16 18:36:31 +02:00
|
|
|
redraw();
|
2012-06-13 16:04:06 +03:00
|
|
|
}
|