| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665 | /* * BattleFieldController.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 "BattleFieldController.h"#include "BattleInterface.h"#include "BattleActionsController.h"#include "BattleInterfaceClasses.h"#include "BattleEffectsController.h"#include "BattleSiegeController.h"#include "BattleStacksController.h"#include "BattleObstacleController.h"#include "BattleProjectileController.h"#include "BattleRenderer.h"#include "../CGameInfo.h"#include "../CPlayerInterface.h"#include "../gui/CAnimation.h"#include "../gui/Canvas.h"#include "../gui/CGuiHandler.h"#include "../gui/CCursorHandler.h"#include "../../CCallback.h"#include "../../lib/BattleFieldHandler.h"#include "../../lib/CConfigHandler.h"#include "../../lib/CStack.h"#include "../../lib/spells/ISpellMechanics.h"BattleFieldController::BattleFieldController(BattleInterface & owner):	owner(owner),	attackingHex(BattleHex::INVALID){	OBJ_CONSTRUCTION_CAPTURING_ALL_NO_DISPOSE;	pos.w = owner.pos.w;	pos.h = owner.pos.h;	//preparing cells and hexes	cellBorder = IImage::createFromFile("CCELLGRD.BMP");	cellShade = IImage::createFromFile("CCELLSHD.BMP");	if(!owner.siegeController)	{		auto bfieldType = owner.curInt->cb->battleGetBattlefieldType();		if(bfieldType == BattleField::NONE)			logGlobal->error("Invalid battlefield returned for current battle");		else			background = IImage::createFromFile(bfieldType.getInfo()->graphics);	}	else	{		std::string backgroundName = owner.siegeController->getBattleBackgroundName();		background = IImage::createFromFile(backgroundName);	}	//preparing graphic with cell borders	cellBorders = std::make_unique<Canvas>(Point(background->width(), background->height()));	for (int i=0; i<GameConstants::BFIELD_SIZE; ++i)	{		if ( i % GameConstants::BFIELD_WIDTH == 0)			continue;		if ( i % GameConstants::BFIELD_WIDTH == GameConstants::BFIELD_WIDTH - 1)			continue;		cellBorders->draw(cellBorder, hexPositionLocal(i).topLeft());	}	backgroundWithHexes = std::make_unique<Canvas>(Point(background->width(), background->height()));	for (int h = 0; h < GameConstants::BFIELD_SIZE; ++h)	{		auto hex = std::make_shared<ClickableHex>();		hex->myNumber = h;		hex->pos = hexPositionAbsolute(h);		hex->myInterface = &owner;		bfield.push_back(hex);	}	auto accessibility = owner.curInt->cb->getAccesibility();	for(int i = 0; i < accessibility.size(); i++)		stackCountOutsideHexes[i] = (accessibility[i] == EAccessibility::ACCESSIBLE);}void BattleFieldController::renderBattlefield(Canvas & canvas){	showBackground(canvas);	BattleRenderer renderer(owner);	renderer.execute(canvas);	owner.projectilesController->showProjectiles(canvas);}void BattleFieldController::showBackground(Canvas & canvas){	if (owner.stacksController->getActiveStack() != nullptr ) //&& creAnims[stacksController->getActiveStack()->ID]->isIdle() //show everything with range		showBackgroundImageWithHexes(canvas);	else		showBackgroundImage(canvas);	showHighlightedHexes(canvas);}void BattleFieldController::showBackgroundImage(Canvas & canvas){	canvas.draw(background, owner.pos.topLeft());	owner.obstacleController->showAbsoluteObstacles(canvas, pos.topLeft());	if ( owner.siegeController )		owner.siegeController->showAbsoluteObstacles(canvas, pos.topLeft());	if (settings["battle"]["cellBorders"].Bool())		canvas.draw(*cellBorders, owner.pos.topLeft());}void BattleFieldController::showBackgroundImageWithHexes(Canvas & canvas){	canvas.draw(*backgroundWithHexes.get(), owner.pos.topLeft());}void BattleFieldController::redrawBackgroundWithHexes(){	const CStack *activeStack = owner.stacksController->getActiveStack();	std::vector<BattleHex> attackableHexes;	if (activeStack)		occupyableHexes = owner.curInt->cb->battleGetAvailableHexes(activeStack, true, &attackableHexes);	auto accessibility = owner.curInt->cb->getAccesibility();	for(int i = 0; i < accessibility.size(); i++)		stackCountOutsideHexes[i] = (accessibility[i] == EAccessibility::ACCESSIBLE);	//prepare background graphic with hexes and shaded hexes	backgroundWithHexes->draw(background, Point(0,0));	owner.obstacleController->showAbsoluteObstacles(*backgroundWithHexes, Point(0,0));	if ( owner.siegeController )		owner.siegeController->showAbsoluteObstacles(*backgroundWithHexes, Point(0,0));	if (settings["battle"]["stackRange"].Bool())	{		std::vector<BattleHex> hexesToShade = occupyableHexes;		hexesToShade.insert(hexesToShade.end(), attackableHexes.begin(), attackableHexes.end());		for (BattleHex hex : hexesToShade)		{			backgroundWithHexes->draw(cellShade, hexPositionLocal(hex).topLeft());		}	}	if(settings["battle"]["cellBorders"].Bool())		backgroundWithHexes->draw(*cellBorders, Point(0, 0));}void BattleFieldController::showHighlightedHex(Canvas & canvas, BattleHex hex, bool darkBorder){	Point hexPos = hexPositionAbsolute(hex).topLeft();	canvas.draw(cellShade, hexPos);	if(!darkBorder && settings["battle"]["cellBorders"].Bool())		canvas.draw(cellBorder, hexPos);}std::set<BattleHex> BattleFieldController::getHighlightedHexesStackRange(){	std::set<BattleHex> result;	if ( !owner.stacksController->getActiveStack())		return result;	if ( !settings["battle"]["stackRange"].Bool())		return result;	auto hoveredHex = getHoveredHex();	std::set<BattleHex> set = owner.curInt->cb->battleGetAttackedHexes(owner.stacksController->getActiveStack(), hoveredHex, attackingHex);	for(BattleHex hex : set)		result.insert(hex);	// display the movement shadow of the stack at b (i.e. stack under mouse)	const CStack * const shere = owner.curInt->cb->battleGetStackByPos(hoveredHex, false);	if(shere && shere != owner.stacksController->getActiveStack() && shere->alive())	{		std::vector<BattleHex> v = owner.curInt->cb->battleGetAvailableHexes(shere, true, nullptr);		for(BattleHex hex : v)			result.insert(hex);	}	return result;}std::set<BattleHex> BattleFieldController::getHighlightedHexesSpellRange(){	std::set<BattleHex> result;	auto hoveredHex = getHoveredHex();	if(!settings["battle"]["mouseShadow"].Bool())		return result;	const spells::Caster *caster = nullptr;	const CSpell *spell = nullptr;	spells::Mode mode = spells::Mode::HERO;	if(owner.actionsController->spellcastingModeActive())//hero casts spell	{		spell = owner.actionsController->selectedSpell().toSpell();		caster = owner.getActiveHero();	}	else if(owner.stacksController->activeStackSpellToCast() != SpellID::NONE)//stack casts spell	{		spell = SpellID(owner.stacksController->activeStackSpellToCast()).toSpell();		caster = owner.stacksController->getActiveStack();		mode = spells::Mode::CREATURE_ACTIVE;	}	if(caster && spell) //when casting spell	{		// printing shaded hex(es)		spells::BattleCast event(owner.curInt->cb.get(), caster, mode, spell);		auto shaded = spell->battleMechanics(&event)->rangeInHexes(hoveredHex);		for(BattleHex shadedHex : shaded)		{			if((shadedHex.getX() != 0) && (shadedHex.getX() != GameConstants::BFIELD_WIDTH - 1))				result.insert(shadedHex);		}	}	return result;}std::set<BattleHex> BattleFieldController::getHighlightedHexesMovementTarget(){	const CStack * stack = owner.stacksController->getActiveStack();	std::set<BattleHex> result;	auto hoveredHex = getHoveredHex();	if (stack)	{		std::vector<BattleHex> v = owner.curInt->cb->battleGetAvailableHexes(stack, false, nullptr);		if (vstd::contains(v,hoveredHex))		{			result.insert(hoveredHex);			if (stack->doubleWide())				result.insert(stack->occupiedHex(hoveredHex));		}		else		{			if (stack->doubleWide())			{				for (auto const & hex : v)				{					if (stack->occupiedHex(hex) == hoveredHex)					{						result.insert(hoveredHex);						result.insert(hex);					}				}			}		}	}	return result;}void BattleFieldController::showHighlightedHexes(Canvas & canvas){	std::set<BattleHex> hoveredStack = getHighlightedHexesStackRange();	std::set<BattleHex> hoveredSpell = getHighlightedHexesSpellRange();	std::set<BattleHex> hoveredMove  = getHighlightedHexesMovementTarget();	auto const & hoveredMouse = owner.actionsController->spellcastingModeActive() ? hoveredSpell : hoveredMove;	for(int b=0; b<GameConstants::BFIELD_SIZE; ++b)	{		bool stack = hoveredStack.count(b);		bool mouse = hoveredMouse.count(b);		if ( stack && mouse )		{			// area where enemy stack can move AND affected by mouse cursor - create darker highlight by blitting twice			showHighlightedHex(canvas, b, true);			showHighlightedHex(canvas, b, true);		}		if ( !stack && mouse )		{			showHighlightedHex(canvas, b, true);		}		if ( stack && !mouse )		{			showHighlightedHex(canvas, b, false);		}	}}Rect BattleFieldController::hexPositionLocal(BattleHex hex) const{	int x = 14 + ((hex.getY())%2==0 ? 22 : 0) + 44*hex.getX();	int y = 86 + 42 *hex.getY();	int w = cellShade->width();	int h = cellShade->height();	return Rect(x, y, w, h);}Rect BattleFieldController::hexPositionAbsolute(BattleHex hex) const{	return hexPositionLocal(hex) + owner.pos.topLeft();}bool BattleFieldController::isPixelInHex(Point const & position){	return !cellShade->isTransparent(position);}BattleHex BattleFieldController::getHoveredHex(){	for ( auto const & hex : bfield)		if (hex->hovered && hex->strictHovered)			return hex->myNumber;	return BattleHex::INVALID;}void BattleFieldController::setBattleCursor(BattleHex myNumber){	Rect hoveredHexPos = hexPositionAbsolute(myNumber);	CCursorHandler *cursor = CCS->curh;	const double subdividingAngle = 2.0*M_PI/6.0; // Divide a hex into six sectors.	const double hexMidX = hoveredHexPos.x + hoveredHexPos.w/2.0;	const double hexMidY = hoveredHexPos.y + hoveredHexPos.h/2.0;	const double cursorHexAngle = M_PI - atan2(hexMidY - cursor->position().y, cursor->position().y - hexMidX) + subdividingAngle/2; //TODO: refactor this nightmare	const double sector = fmod(cursorHexAngle/subdividingAngle, 6.0);	const int zigzagCorrection = !((myNumber/GameConstants::BFIELD_WIDTH)%2); // Off-by-one correction needed to deal with the odd battlefield rows.	std::vector<Cursor::Combat> sectorCursor; // From left to bottom left.	sectorCursor.push_back(Cursor::Combat::HIT_EAST);	sectorCursor.push_back(Cursor::Combat::HIT_SOUTHEAST);	sectorCursor.push_back(Cursor::Combat::HIT_SOUTHWEST);	sectorCursor.push_back(Cursor::Combat::HIT_WEST);	sectorCursor.push_back(Cursor::Combat::HIT_NORTHWEST);	sectorCursor.push_back(Cursor::Combat::HIT_NORTHEAST);	const bool doubleWide = owner.stacksController->getActiveStack()->doubleWide();	bool aboveAttackable = true, belowAttackable = true;	// Exclude directions which cannot be attacked from.	// Check to the left.	if (myNumber%GameConstants::BFIELD_WIDTH <= 1 || !vstd::contains(occupyableHexes, myNumber - 1))	{		sectorCursor[0] = Cursor::Combat::INVALID;	}	// Check top left, top right as well as above for 2-hex creatures.	if (myNumber/GameConstants::BFIELD_WIDTH == 0)	{			sectorCursor[1] = Cursor::Combat::INVALID;			sectorCursor[2] = Cursor::Combat::INVALID;			aboveAttackable = false;	}	else	{		if (doubleWide)		{			bool attackRow[4] = {true, true, true, true};			if (myNumber%GameConstants::BFIELD_WIDTH <= 1 || !vstd::contains(occupyableHexes, myNumber - GameConstants::BFIELD_WIDTH - 2 + zigzagCorrection))				attackRow[0] = false;			if (!vstd::contains(occupyableHexes, myNumber - GameConstants::BFIELD_WIDTH - 1 + zigzagCorrection))				attackRow[1] = false;			if (!vstd::contains(occupyableHexes, myNumber - GameConstants::BFIELD_WIDTH + zigzagCorrection))				attackRow[2] = false;			if (myNumber%GameConstants::BFIELD_WIDTH >= GameConstants::BFIELD_WIDTH - 2 || !vstd::contains(occupyableHexes, myNumber - GameConstants::BFIELD_WIDTH + 1 + zigzagCorrection))				attackRow[3] = false;			if (!(attackRow[0] && attackRow[1]))				sectorCursor[1] = Cursor::Combat::INVALID;			if (!(attackRow[1] && attackRow[2]))				aboveAttackable = false;			if (!(attackRow[2] && attackRow[3]))				sectorCursor[2] = Cursor::Combat::INVALID;		}		else		{			if (!vstd::contains(occupyableHexes, myNumber - GameConstants::BFIELD_WIDTH - 1 + zigzagCorrection))				sectorCursor[1] = Cursor::Combat::INVALID;			if (!vstd::contains(occupyableHexes, myNumber - GameConstants::BFIELD_WIDTH + zigzagCorrection))				sectorCursor[2] = Cursor::Combat::INVALID;		}	}	// Check to the right.	if (myNumber%GameConstants::BFIELD_WIDTH >= GameConstants::BFIELD_WIDTH - 2 || !vstd::contains(occupyableHexes, myNumber + 1))	{		sectorCursor[3] = Cursor::Combat::INVALID;	}	// Check bottom right, bottom left as well as below for 2-hex creatures.	if (myNumber/GameConstants::BFIELD_WIDTH == GameConstants::BFIELD_HEIGHT - 1)	{		sectorCursor[4] = Cursor::Combat::INVALID;		sectorCursor[5] = Cursor::Combat::INVALID;		belowAttackable = false;	}	else	{		if (doubleWide)		{			bool attackRow[4] = {true, true, true, true};			if (myNumber%GameConstants::BFIELD_WIDTH <= 1 || !vstd::contains(occupyableHexes, myNumber + GameConstants::BFIELD_WIDTH - 2 + zigzagCorrection))				attackRow[0] = false;			if (!vstd::contains(occupyableHexes, myNumber + GameConstants::BFIELD_WIDTH - 1 + zigzagCorrection))				attackRow[1] = false;			if (!vstd::contains(occupyableHexes, myNumber + GameConstants::BFIELD_WIDTH + zigzagCorrection))				attackRow[2] = false;			if (myNumber%GameConstants::BFIELD_WIDTH >= GameConstants::BFIELD_WIDTH - 2 || !vstd::contains(occupyableHexes, myNumber + GameConstants::BFIELD_WIDTH + 1 + zigzagCorrection))				attackRow[3] = false;			if (!(attackRow[0] && attackRow[1]))				sectorCursor[5] = Cursor::Combat::INVALID;			if (!(attackRow[1] && attackRow[2]))				belowAttackable = false;			if (!(attackRow[2] && attackRow[3]))				sectorCursor[4] = Cursor::Combat::INVALID;		}		else		{			if (!vstd::contains(occupyableHexes, myNumber + GameConstants::BFIELD_WIDTH + zigzagCorrection))				sectorCursor[4] = Cursor::Combat::INVALID;			if (!vstd::contains(occupyableHexes, myNumber + GameConstants::BFIELD_WIDTH - 1 + zigzagCorrection))				sectorCursor[5] = Cursor::Combat::INVALID;		}	}	// Determine index from sector.	int cursorIndex;	if (doubleWide)	{		sectorCursor.insert(sectorCursor.begin() + 5, belowAttackable ? Cursor::Combat::HIT_NORTH : Cursor::Combat::INVALID);		sectorCursor.insert(sectorCursor.begin() + 2, aboveAttackable ? Cursor::Combat::HIT_SOUTH : Cursor::Combat::INVALID);		if (sector < 1.5)			cursorIndex = static_cast<int>(sector);		else if (sector >= 1.5 && sector < 2.5)			cursorIndex = 2;		else if (sector >= 2.5 && sector < 4.5)			cursorIndex = (int) sector + 1;		else if (sector >= 4.5 && sector < 5.5)			cursorIndex = 6;		else			cursorIndex = (int) sector + 2;	}	else	{		cursorIndex = static_cast<int>(sector);	}	// Generally should NEVER happen, but to avoid the possibility of having endless loop below... [#1016]	if (!vstd::contains_if (sectorCursor, [](Cursor::Combat sc) { return sc != Cursor::Combat::INVALID; }))	{		logGlobal->error("Error: for hex %d cannot find a hex to attack from!", myNumber);		attackingHex = -1;		return;	}	// Find the closest direction attackable, starting with the right one.	// FIXME: Is this really how the original H3 client does it?	int i = 0;	while (sectorCursor[(cursorIndex + i)%sectorCursor.size()] == Cursor::Combat::INVALID) //Why hast thou forsaken me?		i = i <= 0 ? 1 - i : -i; // 0, 1, -1, 2, -2, 3, -3 etc..	int index = (cursorIndex + i)%sectorCursor.size(); //hopefully we get elements from sectorCursor	cursor->set(sectorCursor[index]);	switch (index)	{		case 0:			attackingHex = myNumber - 1; //left			break;		case 1:			attackingHex = myNumber - GameConstants::BFIELD_WIDTH - 1 + zigzagCorrection; //top left			break;		case 2:			attackingHex = myNumber - GameConstants::BFIELD_WIDTH + zigzagCorrection; //top right			break;		case 3:			attackingHex = myNumber + 1; //right			break;		case 4:			attackingHex = myNumber + GameConstants::BFIELD_WIDTH + zigzagCorrection; //bottom right			break;		case 5:			attackingHex = myNumber + GameConstants::BFIELD_WIDTH - 1 + zigzagCorrection; //bottom left			break;	}	BattleHex hex(attackingHex);	if (!hex.isValid())		attackingHex = -1;}BattleHex BattleFieldController::fromWhichHexAttack(BattleHex myNumber){	//TODO far too much repeating code	BattleHex destHex;	switch(CCS->curh->get<Cursor::Combat>())	{	case Cursor::Combat::HIT_NORTHWEST: //from bottom right		{			bool doubleWide = owner.stacksController->getActiveStack()->doubleWide();			destHex = myNumber + ( (myNumber/GameConstants::BFIELD_WIDTH)%2 ? GameConstants::BFIELD_WIDTH : GameConstants::BFIELD_WIDTH+1 ) +				(owner.stacksController->getActiveStack()->side == BattleSide::ATTACKER && doubleWide ? 1 : 0);			if(vstd::contains(occupyableHexes, destHex))				return destHex;			else if(owner.stacksController->getActiveStack()->side == BattleSide::ATTACKER)			{				if (vstd::contains(occupyableHexes, destHex+1))					return destHex+1;			}			else //if we are defender			{				if(vstd::contains(occupyableHexes, destHex-1))					return destHex-1;			}			break;		}	case Cursor::Combat::HIT_NORTHEAST: //from bottom left		{			destHex = myNumber + ( (myNumber/GameConstants::BFIELD_WIDTH)%2 ? GameConstants::BFIELD_WIDTH-1 : GameConstants::BFIELD_WIDTH );			if (vstd::contains(occupyableHexes, destHex))				return destHex;			else if(owner.stacksController->getActiveStack()->side == BattleSide::ATTACKER)			{				if(vstd::contains(occupyableHexes, destHex+1))					return destHex+1;			}			else //we are defender			{				if(vstd::contains(occupyableHexes, destHex-1))					return destHex-1;			}			break;		}	case Cursor::Combat::HIT_EAST: //from left		{			if(owner.stacksController->getActiveStack()->doubleWide() && owner.stacksController->getActiveStack()->side == BattleSide::DEFENDER)			{				std::vector<BattleHex> acc = owner.curInt->cb->battleGetAvailableHexes(owner.stacksController->getActiveStack());				if (vstd::contains(acc, myNumber))					return myNumber - 1;				else					return myNumber - 2;			}			else			{				return myNumber - 1;			}			break;		}	case Cursor::Combat::HIT_SOUTHEAST: //from top left		{			destHex = myNumber - ((myNumber/GameConstants::BFIELD_WIDTH) % 2 ? GameConstants::BFIELD_WIDTH + 1 : GameConstants::BFIELD_WIDTH);			if(vstd::contains(occupyableHexes, destHex))				return destHex;			else if(owner.stacksController->getActiveStack()->side == BattleSide::ATTACKER)			{				if(vstd::contains(occupyableHexes, destHex+1))					return destHex+1;			}			else //if we are defender			{				if(vstd::contains(occupyableHexes, destHex-1))					return destHex-1;			}			break;		}	case Cursor::Combat::HIT_SOUTHWEST: //from top right		{			bool doubleWide = owner.stacksController->getActiveStack()->doubleWide();			destHex = myNumber - ( (myNumber/GameConstants::BFIELD_WIDTH)%2 ? GameConstants::BFIELD_WIDTH : GameConstants::BFIELD_WIDTH-1 ) +				(owner.stacksController->getActiveStack()->side == BattleSide::ATTACKER && doubleWide ? 1 : 0);			if(vstd::contains(occupyableHexes, destHex))				return destHex;			else if(owner.stacksController->getActiveStack()->side == BattleSide::ATTACKER)			{				if(vstd::contains(occupyableHexes, destHex+1))					return destHex+1;			}			else //if we are defender			{				if(vstd::contains(occupyableHexes, destHex-1))					return destHex-1;			}			break;		}	case Cursor::Combat::HIT_WEST: //from right		{			if(owner.stacksController->getActiveStack()->doubleWide() && owner.stacksController->getActiveStack()->side == BattleSide::ATTACKER)			{				std::vector<BattleHex> acc = owner.curInt->cb->battleGetAvailableHexes(owner.stacksController->getActiveStack());				if(vstd::contains(acc, myNumber))					return myNumber + 1;				else					return myNumber + 2;			}			else			{				return myNumber + 1;			}			break;		}	case Cursor::Combat::HIT_NORTH: //from bottom		{			destHex = myNumber + ( (myNumber/GameConstants::BFIELD_WIDTH)%2 ? GameConstants::BFIELD_WIDTH : GameConstants::BFIELD_WIDTH+1 );			if(vstd::contains(occupyableHexes, destHex))				return destHex;			else if(owner.stacksController->getActiveStack()->side == BattleSide::ATTACKER)			{				if(vstd::contains(occupyableHexes, destHex+1))					return destHex+1;			}			else //if we are defender			{				if(vstd::contains(occupyableHexes, destHex-1))					return destHex-1;			}			break;		}	case Cursor::Combat::HIT_SOUTH: //from top		{			destHex = myNumber - ( (myNumber/GameConstants::BFIELD_WIDTH)%2 ? GameConstants::BFIELD_WIDTH : GameConstants::BFIELD_WIDTH-1 );			if (vstd::contains(occupyableHexes, destHex))				return destHex;			else if(owner.stacksController->getActiveStack()->side == BattleSide::ATTACKER)			{				if(vstd::contains(occupyableHexes, destHex+1))					return destHex+1;			}			else //if we are defender			{				if(vstd::contains(occupyableHexes, destHex-1))					return destHex-1;			}			break;		}	}	return BattleHex::INVALID;}bool BattleFieldController::isTileAttackable(const BattleHex & number) const{	for (auto & elem : occupyableHexes)	{		if (BattleHex::mutualPosition(elem, number) != -1 || elem == number)			return true;	}	return false;}bool BattleFieldController::stackCountOutsideHex(const BattleHex & number) const{	return stackCountOutsideHexes[number];}
 |