2011-12-14 00:23:17 +03:00
# include "StdInc.h"
2007-06-12 12:33:20 +03:00
# include "CObjectHandler.h"
2011-12-14 00:23:17 +03:00
2008-06-13 11:16:51 +03:00
# include "CDefObjInfoHandler.h"
2008-11-28 03:36:34 +02:00
# include "CGeneralTextHandler.h"
2007-10-26 20:55:33 +03:00
# include "CDefObjInfoHandler.h"
2007-11-19 00:58:28 +02:00
# include "CHeroHandler.h"
2008-11-11 17:17:58 +02:00
# include "CSpellHandler.h"
2012-08-11 12:06:23 +03:00
# include "CModHandler.h"
2010-12-20 23:22:53 +02:00
# include "../client/CSoundBase.h"
2008-10-26 22:58:34 +02:00
# include <boost/random/linear_congruential.hpp>
2008-04-14 20:45:01 +03:00
# include "CTownHandler.h"
2010-07-16 17:22:18 +03:00
# include "CCreatureHandler.h"
2010-12-20 23:22:53 +02:00
# include "VCMI_Lib.h"
# include "IGameCallback.h"
# include "CGameState.h"
# include "NetPacks.h"
2011-12-14 00:23:17 +03:00
# include "StartInfo.h"
2013-01-03 15:19:20 +03:00
# include "Mapping/CMap.h"
2009-09-17 13:47:47 +03:00
# include <SDL_stdinc.h>
2011-08-26 23:32:05 +03:00
# include "CBuildingHandler.h"
2012-08-02 14:03:26 +03:00
# include "JsonNode.h"
# include "Filesystem/CResourceLoader.h"
2011-12-14 00:23:17 +03:00
# include "GameConstants.h"
2011-02-22 11:47:25 +02:00
2009-07-26 06:33:13 +03:00
using namespace boost : : assign ;
2008-12-22 19:48:41 +02:00
2009-04-15 17:03:31 +03:00
/*
* CObjectHandler . 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
*
*/
2012-02-21 17:08:42 +03:00
// It looks that we can't rely on shadowCoverage correctness (Mantis #866). This may result
// in notable performance decrease (SDL blit with custom alpha blit) not notable on my system (Ivan)
# define USE_COVERAGE_MAP 0
2008-12-27 03:01:59 +02:00
std : : map < int , std : : map < int , std : : vector < int > > > CGTeleport : : objs ;
2009-09-07 05:29:44 +03:00
std : : vector < std : : pair < int , int > > CGTeleport : : gates ;
2008-12-22 19:48:41 +02:00
IGameCallback * IObjectInterface : : cb = NULL ;
2008-10-26 22:58:34 +02:00
extern boost : : rand48 ran ;
2009-08-11 10:50:29 +03:00
std : : map < ui8 , std : : set < ui8 > > CGKeys : : playerKeyMap ;
2009-08-13 04:03:11 +03:00
std : : map < si32 , std : : vector < si32 > > CGMagi : : eyelist ;
2010-02-10 04:56:00 +02:00
ui8 CGObelisk : : obeliskCount ; //how many obelisks are on map
2010-08-12 18:54:25 +03:00
std : : map < ui8 , ui8 > CGObelisk : : visited ; //map: team_id => how many obelisks has been visited
2010-02-10 04:56:00 +02:00
2010-06-26 19:02:10 +03:00
std : : vector < const CArtifact * > CGTownInstance : : merchantArtifacts ;
2010-07-20 17:08:13 +03:00
std : : vector < int > CGTownInstance : : universitySkills ;
2010-06-26 19:02:10 +03:00
2012-07-19 21:52:44 +03:00
void IObjectInterface : : onHeroVisit ( const CGHeroInstance * h ) const
2012-09-15 22:16:16 +03:00
{ }
2008-12-22 19:48:41 +02:00
2012-07-19 21:52:44 +03:00
void IObjectInterface : : onHeroLeave ( const CGHeroInstance * h ) const
2012-09-15 22:16:16 +03:00
{ }
2008-12-22 19:48:41 +02:00
2008-12-27 03:01:59 +02:00
void IObjectInterface : : newTurn ( ) const
2012-09-15 22:16:16 +03:00
{ }
2008-12-22 19:48:41 +02:00
IObjectInterface : : ~ IObjectInterface ( )
{ }
IObjectInterface : : IObjectInterface ( )
{ }
void IObjectInterface : : initObj ( )
{ }
2009-02-20 14:39:27 +02:00
void IObjectInterface : : setProperty ( ui8 what , ui32 val )
{ }
2012-03-14 16:02:38 +03:00
bool IObjectInterface : : wasVisited ( ui8 player ) const
{
return false ;
}
bool IObjectInterface : : wasVisited ( const CGHeroInstance * h ) const
{
return false ;
}
2009-09-07 05:29:44 +03:00
void IObjectInterface : : postInit ( )
{ }
void IObjectInterface : : preInit ( )
{ }
2009-03-14 19:19:53 +02:00
void CPlayersVisited : : setPropertyDer ( ui8 what , ui32 val )
{
if ( what = = 10 )
2010-05-26 12:47:53 +03:00
players . insert ( ( ui8 ) val ) ;
2009-03-14 19:19:53 +02:00
}
2012-03-14 16:02:38 +03:00
bool CPlayersVisited : : wasVisited ( ui8 player ) const
2009-03-14 19:19:53 +02:00
{
return vstd : : contains ( players , player ) ;
}
2011-09-03 21:32:37 +03:00
// Bank helper. Find the creature ID and their number, and store the
// result in storage (either guards or reward creatures).
static void readCreatures ( const JsonNode & creature , std : : vector < std : : pair < ui16 , ui32 > > & storage )
2009-08-20 13:07:23 +03:00
{
2011-09-03 21:32:37 +03:00
std : : pair < si16 , si32 > creInfo = std : : make_pair ( - 1 , 0 ) ;
2012-01-19 17:33:22 +03:00
creInfo . second = creature [ " number " ] . Float ( ) ;
creInfo . first = creature [ " id " ] . Float ( ) ;
2012-01-23 03:02:10 +03:00
storage . push_back ( creInfo ) ;
2009-08-20 13:07:23 +03:00
}
2010-08-18 16:42:46 +03:00
2011-09-03 21:32:37 +03:00
// Bank helper. Process a bank level.
static void readBankLevel ( const JsonNode & level , BankConfig & bc )
{
int idx ;
2010-08-18 16:42:46 +03:00
2011-09-03 21:32:37 +03:00
bc . chance = level [ " chance " ] . Float ( ) ;
2010-08-18 16:42:46 +03:00
2011-09-03 21:32:37 +03:00
BOOST_FOREACH ( const JsonNode & creature , level [ " guards " ] . Vector ( ) )
{
readCreatures ( creature , bc . guards ) ;
}
2010-08-18 16:42:46 +03:00
2011-09-03 21:32:37 +03:00
bc . upgradeChance = level [ " upgrade_chance " ] . Float ( ) ;
bc . combatValue = level [ " combat_value " ] . Float ( ) ;
2010-08-18 16:42:46 +03:00
2011-12-14 00:23:17 +03:00
bc . resources . resize ( GameConstants : : RESOURCE_QUANTITY ) ;
2011-09-03 21:32:37 +03:00
idx = 0 ;
2012-04-08 05:14:20 +03:00
BOOST_FOREACH ( const JsonNode & resource , level [ " reward_resources " ] . Vector ( ) )
2010-08-18 16:42:46 +03:00
{
2012-04-08 05:14:20 +03:00
bc . resources [ idx ] = resource . Float ( ) ;
2011-09-03 21:32:37 +03:00
idx + + ;
}
BOOST_FOREACH ( const JsonNode & creature , level [ " reward_creatures " ] . Vector ( ) )
{
readCreatures ( creature , bc . creatures ) ;
2010-08-18 16:42:46 +03:00
}
bc . artifacts . resize ( 4 ) ;
2011-09-03 21:32:37 +03:00
idx = 0 ;
BOOST_FOREACH ( const JsonNode & artifact , level [ " reward_artifacts " ] . Vector ( ) )
2010-08-18 16:42:46 +03:00
{
2011-09-03 21:32:37 +03:00
bc . artifacts [ idx ] = artifact . Float ( ) ;
idx + + ;
2010-08-18 16:42:46 +03:00
}
2011-09-03 21:32:37 +03:00
bc . value = level [ " value " ] . Float ( ) ;
bc . rewardDifficulty = level [ " profitability " ] . Float ( ) ;
bc . easiest = level [ " easiest " ] . Float ( ) ;
2010-08-18 16:42:46 +03:00
}
2009-08-20 13:07:23 +03:00
2007-06-11 20:21:27 +03:00
void CObjectHandler : : loadObjects ( )
{
2011-09-03 17:14:43 +03:00
tlog5 < < " \t \t Reading cregens \n " ;
2010-05-18 10:01:54 +03:00
2012-08-02 14:03:26 +03:00
const JsonNode config ( ResourceID ( " config/dwellings.json " ) ) ;
2011-09-03 17:14:43 +03:00
BOOST_FOREACH ( const JsonNode & dwelling , config [ " dwellings " ] . Vector ( ) )
{
cregens [ dwelling [ " dwelling " ] . Float ( ) ] = dwelling [ " creature " ] . Float ( ) ;
}
tlog5 < < " \t \t Done loading cregens! \n " ;
2010-05-18 10:01:54 +03:00
2012-04-08 05:14:20 +03:00
tlog5 < < " \t \t Reading resources prices \n " ;
2012-08-02 14:03:26 +03:00
const JsonNode config2 ( ResourceID ( " config/resources.json " ) ) ;
2012-04-08 05:14:20 +03:00
BOOST_FOREACH ( const JsonNode & price , config2 [ " resources_prices " ] . Vector ( ) )
2011-09-03 17:14:43 +03:00
{
2011-09-03 17:26:39 +03:00
resVals . push_back ( price . Float ( ) ) ;
2008-02-05 05:56:45 +02:00
}
2011-09-03 17:26:39 +03:00
tlog5 < < " \t \t Done loading resource prices! \n " ;
2009-08-13 04:03:11 +03:00
2011-09-03 21:32:37 +03:00
tlog5 < < " \t \t Reading banks configs \n " ;
2012-08-02 14:03:26 +03:00
const JsonNode config3 ( ResourceID ( " config/bankconfig.json " ) ) ;
2011-09-03 21:32:37 +03:00
int bank_num = 0 ;
BOOST_FOREACH ( const JsonNode & bank , config3 [ " banks " ] . Vector ( ) )
2009-09-25 12:57:13 +03:00
{
2011-09-03 21:32:37 +03:00
creBanksNames [ bank_num ] = bank [ " name " ] . String ( ) ;
2009-08-20 13:07:23 +03:00
2011-09-03 21:32:37 +03:00
int level_num = 0 ;
BOOST_FOREACH ( const JsonNode & level , bank [ " levels " ] . Vector ( ) )
2009-08-20 13:07:23 +03:00
{
2011-09-03 21:32:37 +03:00
banksInfo [ bank_num ] . push_back ( new BankConfig ) ;
BankConfig & bc = * banksInfo [ bank_num ] . back ( ) ;
bc . level = level_num ;
readBankLevel ( level , bc ) ;
level_num + + ;
2009-08-20 13:07:23 +03:00
}
2011-09-03 21:32:37 +03:00
bank_num + + ;
2009-08-20 13:07:23 +03:00
}
2011-09-03 21:32:37 +03:00
tlog5 < < " \t \t Done loading banks configs \n " ;
2007-08-09 19:28:01 +03:00
}
2009-08-20 13:07:23 +03:00
2012-03-03 13:08:01 +03:00
int CObjectHandler : : bankObjToIndex ( const CGObjectInstance * obj )
{
2012-09-23 21:01:04 +03:00
switch ( obj - > ID ) //find appriopriate key
{
case Obj : : CREATURE_BANK :
return obj - > subID ;
case Obj : : DERELICT_SHIP :
return 8 ;
case Obj : : DRAGON_UTOPIA :
return 10 ;
case Obj : : CRYPT :
return 9 ;
case Obj : : SHIPWRECK :
return 7 ;
default :
2012-12-24 18:07:30 +03:00
tlog2 < < " Unrecognized Bank indetifier! \n " ;
2012-09-23 21:01:04 +03:00
return 0 ;
2012-03-03 13:08:01 +03:00
}
}
2012-09-24 21:52:30 +03:00
int CGObjectInstance : : getOwner ( ) const
2007-10-27 22:38:48 +03:00
{
2007-11-19 00:58:28 +02:00
//if (state)
// return state->owner;
2008-08-02 18:08:03 +03:00
//else
2007-11-19 00:58:28 +02:00
return tempOwner ; //won't have owner
}
2008-12-22 19:48:41 +02:00
CGObjectInstance : : CGObjectInstance ( ) : animPhaseShift ( rand ( ) % 0xff )
{
pos = int3 ( - 1 , - 1 , - 1 ) ;
//std::cout << "Tworze obiekt "<<this<<std::endl;
//state = new CLuaObjectScript();
ID = subID = id = - 1 ;
defInfo = NULL ;
tempOwner = 254 ;
blockVisit = false ;
}
CGObjectInstance : : ~ CGObjectInstance ( )
{
//std::cout << "Usuwam obiekt "<<this<<std::endl;
//if (state)
// delete state;
//state=NULL;
}
const std : : string & CGObjectInstance : : getHoverText ( ) const
{
return hoverName ;
}
2012-09-24 21:52:30 +03:00
void CGObjectInstance : : setOwner ( int ow )
2007-11-19 00:58:28 +02:00
{
//if (state)
// state->owner = ow;
//else
tempOwner = ow ;
2007-10-27 22:38:48 +03:00
}
int CGObjectInstance : : getWidth ( ) const //returns width of object graphic in tiles
{
2008-06-17 20:48:32 +03:00
return defInfo - > width ;
2007-10-27 22:38:48 +03:00
}
int CGObjectInstance : : getHeight ( ) const //returns height of object graphic in tiles
{
2009-03-12 01:25:59 +02:00
return defInfo - > height ;
2007-10-27 22:38:48 +03:00
}
2008-09-26 17:16:01 +03:00
bool CGObjectInstance : : visitableAt ( int x , int y ) const //returns true if object is visitable at location (x, y) form left top tile of image (x, y in tiles)
2007-10-27 22:38:48 +03:00
{
2009-03-12 01:25:59 +02:00
if ( defInfo = = NULL )
{
tlog2 < < " Warning: VisitableAt for obj " < < id < < " : NULL defInfo! \n " ;
2007-10-27 22:38:48 +03:00
return false ;
2009-03-12 01:25:59 +02:00
}
if ( ( defInfo - > visitMap [ y ] > > ( 7 - x ) ) & 1 )
{
2007-08-09 19:28:01 +03:00
return true ;
2009-03-12 01:25:59 +02:00
}
2007-10-27 22:38:48 +03:00
return false ;
}
2008-09-26 17:16:01 +03:00
bool CGObjectInstance : : blockingAt ( int x , int y ) const
{
if ( x < 0 | | y < 0 | | x > = getWidth ( ) | | y > = getHeight ( ) | | defInfo = = NULL )
return false ;
if ( ( defInfo - > blockMap [ y + 6 - getHeight ( ) ] > > ( 7 - ( 8 - getWidth ( ) + x ) ) ) & 1 )
2009-07-30 15:49:45 +03:00
return false ;
return true ;
2008-09-26 17:16:01 +03:00
}
2009-07-01 18:58:20 +03:00
bool CGObjectInstance : : coveringAt ( int x , int y ) const
{
2012-05-19 14:47:26 +03:00
//input coordinates are always negative
x = - x ;
y = - y ;
2012-02-21 17:08:42 +03:00
# if USE_COVERAGE_MAP
2012-05-19 14:47:26 +03:00
//NOTE: this code may be broken
2012-07-19 21:52:44 +03:00
if ( ( defInfo - > coverageMap [ y ] > > ( 7 - ( x ) ) ) & 1
2009-12-22 23:53:50 +02:00
| | ( defInfo - > shadowCoverage [ y ] > > ( 7 - ( x ) ) ) & 1 )
2009-07-01 18:58:20 +03:00
return true ;
return false ;
2012-02-21 17:18:58 +03:00
# else
2012-05-19 14:47:26 +03:00
return x > = 0 & & y > = 0 & & x < getWidth ( ) & & y < getHeight ( ) ;
# endif
}
bool CGObjectInstance : : hasShadowAt ( int x , int y ) const
{
# if USE_COVERAGE_MAP
//NOTE: this code may be broken
if ( ( defInfo - > shadowCoverage [ y ] > > ( 7 - ( x ) ) ) & 1 )
return true ;
return false ;
# else
return coveringAt ( x , y ) ; // ignore unreliable shadowCoverage map
2012-02-21 17:08:42 +03:00
# endif
2009-07-01 18:58:20 +03:00
}
2009-07-30 15:49:45 +03:00
std : : set < int3 > CGObjectInstance : : getBlockedPos ( ) const
{
std : : set < int3 > ret ;
for ( int w = 0 ; w < getWidth ( ) ; + + w )
{
for ( int h = 0 ; h < getHeight ( ) ; + + h )
{
if ( blockingAt ( w , h ) )
ret . insert ( int3 ( pos . x - getWidth ( ) + w + 1 , pos . y - getHeight ( ) + h + 1 , pos . z ) ) ;
}
}
return ret ;
}
2007-10-27 22:38:48 +03:00
bool CGObjectInstance : : operator < ( const CGObjectInstance & cmp ) const //screen printing priority comparing
{
2007-11-24 00:33:55 +02:00
if ( defInfo - > printPriority = = 1 & & cmp . defInfo - > printPriority = = 0 )
2007-10-27 22:38:48 +03:00
return true ;
2007-11-24 00:33:55 +02:00
if ( cmp . defInfo - > printPriority = = 1 & & defInfo - > printPriority = = 0 )
2007-08-09 19:28:01 +03:00
return false ;
if ( this - > pos . y < cmp . pos . y )
return true ;
if ( this - > pos . y > cmp . pos . y )
return false ;
2012-09-23 21:01:04 +03:00
if ( cmp . ID = = Obj : : HERO & & ID ! = Obj : : HERO )
2008-02-23 21:20:41 +02:00
return true ;
2012-09-23 21:01:04 +03:00
if ( cmp . ID ! = Obj : : HERO & & ID = = Obj : : HERO )
2008-02-23 21:20:41 +02:00
return false ;
2007-10-28 13:34:33 +02:00
if ( ! defInfo - > isVisitable ( ) & & cmp . defInfo - > isVisitable ( ) )
2007-08-09 19:28:01 +03:00
return true ;
2007-10-28 13:34:33 +02:00
if ( ! cmp . defInfo - > isVisitable ( ) & & defInfo - > isVisitable ( ) )
2007-08-09 19:28:01 +03:00
return false ;
if ( this - > pos . x < cmp . pos . x )
return true ;
return false ;
}
2007-10-26 20:55:33 +03:00
2008-12-22 19:48:41 +02:00
void CGObjectInstance : : initObj ( )
{
2010-07-09 02:03:27 +03:00
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : TAVERN :
2010-07-09 02:03:27 +03:00
blockVisit = true ;
break ;
}
2008-12-22 19:48:41 +02:00
}
2008-08-02 18:08:03 +03:00
2009-02-20 14:39:27 +02:00
void CGObjectInstance : : setProperty ( ui8 what , ui32 val )
{
switch ( what )
{
2010-05-02 21:20:26 +03:00
case ObjProperty : : OWNER :
2009-02-20 14:39:27 +02:00
tempOwner = val ;
break ;
2010-05-02 21:20:26 +03:00
case ObjProperty : : BLOCKVIS :
2009-02-20 14:39:27 +02:00
blockVisit = val ;
break ;
2010-05-02 21:20:26 +03:00
case ObjProperty : : ID :
2009-02-20 14:39:27 +02:00
ID = val ;
break ;
2010-07-31 03:26:34 +03:00
case ObjProperty : : SUBID :
subID = val ;
break ;
2009-02-20 14:39:27 +02:00
}
setPropertyDer ( what , val ) ;
}
void CGObjectInstance : : setPropertyDer ( ui8 what , ui32 val )
{ }
2009-03-12 01:25:59 +02:00
int3 CGObjectInstance : : getSightCenter ( ) const
{
//return vistiable tile if possible
for ( int i = 0 ; i < 8 ; i + + )
for ( int j = 0 ; j < 6 ; j + + )
if ( visitableAt ( i , j ) )
return ( pos + int3 ( i - 7 , j - 5 , 0 ) ) ;
return pos ;
}
int CGObjectInstance : : getSightRadious ( ) const
{
return 3 ;
}
2011-01-20 19:25:15 +02:00
void CGObjectInstance : : getSightTiles ( boost : : unordered_set < int3 , ShashInt3 > & tiles ) const //returns reference to the set
2009-11-11 19:45:03 +02:00
{
2009-12-01 12:23:23 +02:00
cb - > getTilesInRange ( tiles , getSightCenter ( ) , getSightRadious ( ) , tempOwner , 1 ) ;
2009-11-11 19:45:03 +02:00
}
2010-06-19 12:13:10 +03:00
void CGObjectInstance : : hideTiles ( int ourplayer , int radius ) const
{
2012-09-24 19:14:53 +03:00
for ( auto i = cb - > gameState ( ) - > teams . begin ( ) ; i ! = cb - > gameState ( ) - > teams . end ( ) ; i + + )
2010-06-19 12:13:10 +03:00
{
2010-08-06 16:14:10 +03:00
if ( ! vstd : : contains ( i - > second . players , ourplayer ) ) //another team
2010-06-19 12:13:10 +03:00
{
2012-09-24 19:14:53 +03:00
for ( auto j = i - > second . players . begin ( ) ; j ! = i - > second . players . end ( ) ; j + + )
2011-05-10 01:20:47 +03:00
if ( cb - > getPlayer ( * j ) - > status = = PlayerState : : INGAME ) //seek for living player (if any)
2010-08-06 16:14:10 +03:00
{
FoWChange fw ;
fw . mode = 0 ;
2010-08-26 15:06:36 +03:00
fw . player = * j ;
2010-08-06 16:14:10 +03:00
cb - > getTilesInRange ( fw . tiles , pos , radius , ( * j ) , - 1 ) ;
cb - > sendAndApply ( & fw ) ;
break ;
}
2010-06-19 12:13:10 +03:00
}
}
}
2009-07-14 19:20:15 +03:00
int3 CGObjectInstance : : getVisitableOffset ( ) const
{
for ( int y = 0 ; y < 6 ; y + + )
for ( int x = 0 ; x < 8 ; x + + )
if ( ( defInfo - > visitMap [ 5 - y ] > > x ) & 1 )
return int3 ( x , y , 0 ) ;
tlog2 < < " Warning: getVisitableOffset called on non-visitable obj! \n " ;
return int3 ( - 1 , - 1 , - 1 ) ;
}
2009-07-19 10:16:33 +03:00
void CGObjectInstance : : getNameVis ( std : : string & hname ) const
{
const CGHeroInstance * h = cb - > getSelectedHero ( cb - > getCurrentPlayer ( ) ) ;
hname = VLC - > generaltexth - > names [ ID ] ;
2012-07-19 21:52:44 +03:00
if ( h )
2009-07-19 10:16:33 +03:00
{
2010-05-02 21:20:26 +03:00
if ( ! h - > hasBonusFrom ( Bonus : : OBJECT , ID ) )
2009-07-19 10:16:33 +03:00
hname + = " " + VLC - > generaltexth - > allTexts [ 353 ] ; //not visited
else
hname + = " " + VLC - > generaltexth - > allTexts [ 352 ] ; //visited
}
}
void CGObjectInstance : : giveDummyBonus ( int heroID , ui8 duration ) const
{
GiveBonus gbonus ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . type = Bonus : : NONE ;
2010-02-10 04:56:00 +02:00
gbonus . id = heroID ;
2009-07-19 10:16:33 +03:00
gbonus . bonus . duration = duration ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . source = Bonus : : OBJECT ;
2011-02-20 20:32:39 +02:00
gbonus . bonus . sid = ID ;
2009-07-19 10:16:33 +03:00
cb - > giveHeroBonus ( & gbonus ) ;
}
2009-07-26 06:33:13 +03:00
void CGObjectInstance : : onHeroVisit ( const CGHeroInstance * h ) const
{
2010-07-09 02:03:27 +03:00
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : HILL_FORT :
2010-07-22 03:32:45 +03:00
{
OpenWindow ow ;
ow . window = OpenWindow : : HILL_FORT_WINDOW ;
ow . id1 = id ;
ow . id2 = h - > id ;
cb - > sendAndApply ( & ow ) ;
}
break ;
2012-09-23 21:01:04 +03:00
case Obj : : SANCTUARY :
2010-07-09 03:54:17 +03:00
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . soundID = soundBase : : GETPROTECTION ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , 114 ) ; //You enter the sanctuary and immediately feel as if a great weight has been lifted off your shoulders. You feel safe here.
cb - > sendAndApply ( & iw ) ;
}
break ;
2012-09-23 21:01:04 +03:00
case Obj : : TAVERN :
2010-07-09 03:54:17 +03:00
{
OpenWindow ow ;
ow . window = OpenWindow : : TAVERN_WINDOW ;
ow . id1 = h - > id ;
ow . id2 = id ;
cb - > sendAndApply ( & ow ) ;
}
break ;
2010-07-09 02:03:27 +03:00
}
2009-07-26 06:33:13 +03:00
}
2009-12-20 19:14:14 +02:00
ui8 CGObjectInstance : : getPassableness ( ) const
{
return 0 ;
}
2010-05-15 11:33:32 +03:00
int3 CGObjectInstance : : visitablePos ( ) const
{
return pos - getVisitableOffset ( ) ;
}
2011-02-24 17:05:21 +02:00
bool CGObjectInstance : : isVisitable ( ) const
{
for ( int g = 0 ; g < ARRAY_COUNT ( defInfo - > visitMap ) ; + + g )
{
if ( defInfo - > visitMap [ g ] ! = 0 )
{
return true ;
}
}
return false ;
}
2009-05-24 01:57:39 +03:00
static int lowestSpeed ( const CGHeroInstance * chi )
2007-10-27 22:38:48 +03:00
{
2010-05-02 21:20:26 +03:00
if ( ! chi - > Slots ( ) . size ( ) )
2008-12-22 19:48:41 +02:00
{
tlog1 < < " Error! Hero " < < chi - > id < < " ( " < < chi - > name < < " ) has no army! \n " ;
return 20 ;
}
2010-05-02 21:20:26 +03:00
TSlots : : const_iterator i = chi - > Slots ( ) . begin ( ) ;
//TODO? should speed modifiers (eg from artifacts) affect hero movement?
2010-11-22 02:34:46 +02:00
int ret = ( i + + ) - > second - > valOfBonuses ( Bonus : : STACKS_SPEED ) ;
2010-05-02 21:20:26 +03:00
for ( ; i ! = chi - > Slots ( ) . end ( ) ; i + + )
2008-12-22 19:48:41 +02:00
{
2010-11-22 02:34:46 +02:00
ret = std : : min ( ret , i - > second - > valOfBonuses ( Bonus : : STACKS_SPEED ) ) ;
2008-12-22 19:48:41 +02:00
}
return ret ;
2007-10-27 22:38:48 +03:00
}
2008-12-22 19:48:41 +02:00
2011-12-14 00:23:17 +03:00
ui32 CGHeroInstance : : getTileCost ( const TerrainTile & dest , const TerrainTile & from ) const
2007-10-27 22:38:48 +03:00
{
2009-02-12 16:44:58 +02:00
//base move cost
unsigned ret = 100 ;
2012-07-19 21:52:44 +03:00
2009-02-12 16:44:58 +02:00
//if there is road both on dest and src tiles - use road movement cost
2012-11-06 19:39:29 +03:00
if ( dest . roadType ! = ERoadType : : NO_ROAD & & from . roadType ! = ERoadType : : NO_ROAD )
2008-09-28 16:29:37 +03:00
{
2012-10-26 20:51:05 +03:00
int road = std : : min ( dest . roadType , from . roadType ) ; //used road ID
2009-02-12 16:44:58 +02:00
switch ( road )
2008-09-28 16:29:37 +03:00
{
2012-10-26 20:51:05 +03:00
case ERoadType : : DIRT_ROAD :
2009-02-12 16:44:58 +02:00
ret = 75 ;
2008-09-28 16:29:37 +03:00
break ;
2012-10-26 20:51:05 +03:00
case ERoadType : : GRAVEL_ROAD :
2009-02-12 16:44:58 +02:00
ret = 65 ;
2008-09-28 16:29:37 +03:00
break ;
2012-10-26 20:51:05 +03:00
case ERoadType : : COBBLESTONE_ROAD :
2009-02-12 16:44:58 +02:00
ret = 50 ;
2008-09-28 16:29:37 +03:00
break ;
2009-02-12 16:44:58 +02:00
default :
tlog1 < < " Unknown road type: " < < road < < " ... Something wrong! \n " ;
2008-09-28 16:29:37 +03:00
break ;
}
}
2012-07-19 21:52:44 +03:00
else
2007-10-27 22:38:48 +03:00
{
2012-09-06 13:39:48 +03:00
//FIXME: in H3 presence of Nomad in army will remove terrain penalty for sand. Bonus not implemented in VCMI
2012-09-05 15:49:23 +03:00
2012-09-06 13:39:48 +03:00
// NOTE: in H3 neutral stacks will ignore terrain penalty only if placed as topmost stack(s) in hero army.
// This is clearly bug in H3 however intended behaviour is not clear.
// Current VCMI behaviour will ignore neutrals in calculations so army in VCMI
// will always have best penalty without any influence from player-defined stacks order
bool nativeArmy = true ;
BOOST_FOREACH ( auto stack , stacks )
{
int nativeTerrain = VLC - > townh - > factions [ stack . second - > type - > faction ] . nativeTerrain ;
2012-11-06 19:39:29 +03:00
if ( nativeTerrain ! = - 1 & & nativeTerrain ! = from . terType )
2012-09-06 13:39:48 +03:00
{
nativeArmy = false ;
break ;
}
}
if ( ! nativeArmy )
2012-11-06 19:39:29 +03:00
ret = VLC - > heroh - > terrCosts [ from . terType ] ;
2012-09-06 13:39:48 +03:00
}
2007-10-27 22:38:48 +03:00
return ret ;
}
2012-09-06 13:39:48 +03:00
2007-10-27 22:38:48 +03:00
int3 CGHeroInstance : : convertPosition ( int3 src , bool toh3m ) //toh3m=true: manifest->h3m; toh3m=false: h3m->manifest
{
if ( toh3m )
{
src . x + = 1 ;
return src ;
}
else
{
src . x - = 1 ;
return src ;
}
}
int3 CGHeroInstance : : getPosition ( bool h3m ) const //h3m=true - returns position of hero object; h3m=false - returns position of hero 'manifestation'
{
2008-12-23 15:59:03 +02:00
if ( h3m )
{
2007-10-27 22:38:48 +03:00
return pos ;
2008-12-23 15:59:03 +02:00
}
else
{
return convertPosition ( pos , false ) ;
}
2007-10-27 22:38:48 +03:00
}
2008-10-18 14:41:24 +03:00
2007-10-27 22:38:48 +03:00
bool CGHeroInstance : : canWalkOnSea ( ) const
2007-10-26 20:55:33 +03:00
{
2010-05-15 18:00:19 +03:00
return hasBonusOfType ( Bonus : : FLYING_MOVEMENT ) | | hasBonusOfType ( Bonus : : WATER_WALKING ) ;
2007-10-26 20:55:33 +03:00
}
2010-05-15 18:00:19 +03:00
2010-12-23 22:18:10 +02:00
ui8 CGHeroInstance : : getSecSkillLevel ( SecondarySkill skill ) const
2008-02-25 01:06:27 +02:00
{
2008-12-21 21:17:35 +02:00
for ( size_t i = 0 ; i < secSkills . size ( ) ; + + i )
2010-12-29 23:04:22 +02:00
if ( secSkills [ i ] . first = = skill )
2008-02-25 01:06:27 +02:00
return secSkills [ i ] . second ;
2008-09-28 16:29:37 +03:00
return 0 ;
2008-02-25 01:06:27 +02:00
}
2010-07-31 16:55:05 +03:00
2010-12-23 22:18:10 +02:00
void CGHeroInstance : : setSecSkillLevel ( SecondarySkill which , int val , bool abs )
2010-07-31 16:55:05 +03:00
{
if ( getSecSkillLevel ( which ) = = 0 )
{
secSkills . push_back ( std : : pair < int , int > ( which , val ) ) ;
updateSkill ( which , val ) ;
}
else
{
for ( unsigned i = 0 ; i < secSkills . size ( ) ; i + + )
{
if ( secSkills [ i ] . first = = which )
{
if ( abs )
secSkills [ i ] . second = val ;
else
secSkills [ i ] . second + = val ;
if ( secSkills [ i ] . second > 3 ) //workaround to avoid crashes when same sec skill is given more than once
{
tlog1 < < " Warning: Skill " < < which < < " increased over limit! Decreasing to Expert. \n " ;
secSkills [ i ] . second = 3 ;
}
updateSkill ( which , secSkills [ i ] . second ) ; //when we know final value
}
}
}
}
2012-12-18 14:24:13 +03:00
bool CGHeroInstance : : canLearnSkill ( ) const
{
return secSkills . size ( ) < GameConstants : : SKILL_PER_HERO ;
}
2008-10-26 22:58:34 +02:00
int CGHeroInstance : : maxMovePoints ( bool onLand ) const
{
2012-09-06 13:39:48 +03:00
int base ;
2010-02-10 04:56:00 +02:00
if ( onLand )
{
2012-09-06 13:39:48 +03:00
// used function is f(x) = 66.6x + 1300, rounded to second digit, where x is lowest speed in army
static const int baseSpeed = 1300 ; // base speed from creature with 0 speed
int armySpeed = lowestSpeed ( this ) * 20 / 3 ;
base = armySpeed * 10 + baseSpeed ; // separate *10 is intentional to receive same rounding as in h3
vstd : : abetween ( base , 1500 , 2000 ) ; // base speed is limited by these values
2010-02-10 04:56:00 +02:00
}
else
{
base = 1500 ; //on water base movement is always 1500 (speed of army doesn't matter)
}
2012-07-19 21:52:44 +03:00
2010-05-02 21:20:26 +03:00
int bonus = valOfBonuses ( Bonus : : MOVEMENT ) + ( onLand ? valOfBonuses ( Bonus : : LAND_MOVEMENT ) : valOfBonuses ( Bonus : : SEA_MOVEMENT ) ) ;
2008-10-26 22:58:34 +02:00
2009-04-04 01:34:31 +03:00
double modifier = 0 ;
2008-10-26 22:58:34 +02:00
if ( onLand )
{
2011-12-14 00:23:17 +03:00
modifier = valOfBonuses ( Bonus : : SECONDARY_SKILL_PREMY , LOGISTICS ) / 100.0 ;
2008-10-26 22:58:34 +02:00
}
else
{
2011-12-14 00:23:17 +03:00
modifier = valOfBonuses ( Bonus : : SECONDARY_SKILL_PREMY , NAVIGATION ) / 100.0 ;
2008-10-26 22:58:34 +02:00
}
2010-02-10 04:56:00 +02:00
return int ( base + base * modifier ) + bonus ;
2008-10-26 22:58:34 +02:00
}
2009-04-04 01:34:31 +03:00
2008-12-22 19:48:41 +02:00
CGHeroInstance : : CGHeroInstance ( )
2010-03-11 01:16:30 +02:00
: IBoatGenerator ( this )
2008-12-22 19:48:41 +02:00
{
2011-07-13 21:39:02 +03:00
setNodeType ( HERO ) ;
2012-09-23 21:01:04 +03:00
ID = Obj : : HERO ;
2008-12-22 19:48:41 +02:00
tacticFormationEnabled = inTownGarrison = false ;
mana = movement = portrait = level = - 1 ;
isStanding = true ;
moveDir = 4 ;
exp = 0xffffffff ;
visitedTown = NULL ;
type = NULL ;
2009-07-19 04:00:19 +03:00
boat = NULL ;
2012-04-22 20:38:36 +03:00
commander = NULL ;
2011-09-13 06:31:17 +03:00
sex = 0xff ;
2008-12-22 19:48:41 +02:00
secSkills . push_back ( std : : make_pair ( - 1 , - 1 ) ) ;
2011-07-13 21:39:02 +03:00
speciality . setNodeType ( CBonusSystemNode : : SPECIALITY ) ;
2011-09-13 04:46:11 +03:00
attachTo ( & speciality ) ; //do we ever need to detach it?
2008-12-22 19:48:41 +02:00
}
void CGHeroInstance : : initHero ( int SUBID )
{
subID = SUBID ;
initHero ( ) ;
}
void CGHeroInstance : : initHero ( )
{
2010-05-02 21:20:26 +03:00
assert ( validTypes ( true ) ) ;
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : HERO )
2009-02-14 21:12:40 +02:00
initHeroDefInfo ( ) ;
2008-12-22 19:48:41 +02:00
if ( ! type )
type = VLC - > heroh - > heroes [ subID ] ;
2012-12-16 16:47:53 +03:00
if ( ! vstd : : contains ( spells , 0xffffffff ) ) //hero starts with a spell
{
BOOST_FOREACH ( auto spellID , type - > spells )
spells . insert ( spellID ) ;
}
2009-09-14 01:45:58 +03:00
else //remove placeholder
spells - = 0xffffffff ;
2012-12-16 16:47:53 +03:00
if ( ! getArt ( ArtifactPosition : : MACH4 ) & & ! getArt ( ArtifactPosition : : SPELLBOOK ) & & ! type - > spells . empty ( ) ) //no catapult means we haven't read pre-existent set -> use default rules for spellbook
2011-12-14 00:23:17 +03:00
putArtifact ( ArtifactPosition : : SPELLBOOK , CArtifactInstance : : createNewArtifactInstance ( 0 ) ) ;
2008-12-22 19:48:41 +02:00
2011-12-14 00:23:17 +03:00
if ( ! getArt ( ArtifactPosition : : MACH4 ) )
putArtifact ( ArtifactPosition : : MACH4 , CArtifactInstance : : createNewArtifactInstance ( 3 ) ) ; //everyone has a catapult
2012-07-19 21:52:44 +03:00
2008-12-22 19:48:41 +02:00
if ( portrait < 0 | | portrait = = 255 )
2012-12-16 16:47:53 +03:00
portrait = type - > imageIndex ;
2010-05-02 21:20:26 +03:00
if ( ! hasBonus ( Selector : : sourceType ( Bonus : : HERO_BASE_SKILL ) ) )
2008-12-22 19:48:41 +02:00
{
2012-09-23 21:01:04 +03:00
for ( int g = 0 ; g < GameConstants : : PRIMARY_SKILLS ; + + g )
{
2012-12-14 18:32:53 +03:00
pushPrimSkill ( static_cast < PrimarySkill : : PrimarySkill > ( g ) , type - > heroClass - > primarySkillInitial [ g ] ) ;
2012-09-23 21:01:04 +03:00
}
2008-12-22 19:48:41 +02:00
}
if ( secSkills . size ( ) = = 1 & & secSkills [ 0 ] = = std : : pair < ui8 , ui8 > ( - 1 , - 1 ) ) //set secondary skills to default
secSkills = type - > secSkillsInit ;
if ( ! name . length ( ) )
name = type - > name ;
if ( exp = = 0xffffffff )
{
2010-08-28 17:52:20 +03:00
initExp ( ) ;
2008-12-22 19:48:41 +02:00
}
else
{
level = VLC - > heroh - > level ( exp ) ;
}
2010-08-25 17:57:58 +03:00
if ( sex = = 0xFF ) //sex is default
sex = type - > sex ;
2010-05-02 21:20:26 +03:00
setFormation ( false ) ;
if ( ! stacksCount ( ) ) //standard army//initial army
2008-12-22 19:48:41 +02:00
{
2010-07-08 08:52:11 +03:00
initArmy ( ) ;
2008-12-22 19:48:41 +02:00
}
2010-05-02 21:20:26 +03:00
assert ( validTypes ( ) ) ;
2010-07-08 08:52:11 +03:00
2012-08-24 12:37:52 +03:00
if ( VLC - > modh - > modules . COMMANDERS )
2012-04-22 20:38:36 +03:00
{
2012-12-19 19:35:58 +03:00
commander = new CCommanderInstance ( VLC - > townh - > factions [ type - > heroClass - > faction ] . commander ) ;
2012-04-28 14:41:20 +03:00
commander - > setArmyObj ( castToArmyObj ( ) ) ; //TODO: separate function for setting commanders
2012-04-22 20:38:36 +03:00
}
2008-12-22 19:48:41 +02:00
hoverName = VLC - > generaltexth - > allTexts [ 15 ] ;
boost : : algorithm : : replace_first ( hoverName , " %s " , name ) ;
boost : : algorithm : : replace_first ( hoverName , " %s " , type - > heroClass - > name ) ;
2009-04-04 01:34:31 +03:00
2010-08-15 10:39:07 +03:00
if ( mana < 0 )
mana = manaLimit ( ) ;
2008-12-22 19:48:41 +02:00
}
2011-01-28 04:11:58 +02:00
void CGHeroInstance : : initArmy ( IArmyDescriptor * dst /*= NULL*/ )
2010-07-08 08:52:11 +03:00
{
if ( ! dst )
dst = this ;
int howManyStacks = 0 ; //how many stacks will hero receives <1 - 3>
int pom = ran ( ) % 100 ;
int warMachinesGiven = 0 ;
if ( pom < 9 )
howManyStacks = 1 ;
else if ( pom < 79 )
howManyStacks = 2 ;
else
howManyStacks = 3 ;
2012-12-16 16:47:53 +03:00
vstd : : amin ( howManyStacks , type - > initialArmy . size ( ) ) ;
2010-07-26 23:56:39 +03:00
for ( int stackNo = 0 ; stackNo < howManyStacks ; stackNo + + )
2010-07-08 08:52:11 +03:00
{
2012-12-03 19:00:17 +03:00
auto & stack = type - > initialArmy [ stackNo ] ;
2012-04-10 05:30:43 +03:00
2012-12-03 19:00:17 +03:00
int range = stack . maxAmount - stack . minAmount ;
int count = ran ( ) % ( range + 1 ) + stack . minAmount ;
2010-07-08 08:52:11 +03:00
2012-12-03 19:00:17 +03:00
if ( stack . creature > = 145 & &
stack . creature < = 149 ) //war machine
2010-07-08 08:52:11 +03:00
{
warMachinesGiven + + ;
2010-12-30 16:41:46 +02:00
if ( dst ! = this )
continue ;
int slot = - 1 , aid = - 1 ;
2012-12-03 19:00:17 +03:00
switch ( stack . creature )
2010-07-08 08:52:11 +03:00
{
case 145 : //catapult
2011-12-14 00:23:17 +03:00
slot = ArtifactPosition : : MACH4 ;
2010-12-30 16:41:46 +02:00
aid = 3 ;
2010-07-08 08:52:11 +03:00
break ;
default :
2012-12-03 19:00:17 +03:00
aid = CArtHandler : : convertMachineID ( stack . creature , true ) ;
2010-12-30 16:41:46 +02:00
slot = 9 + aid ;
2010-07-08 08:52:11 +03:00
break ;
}
2010-12-30 16:41:46 +02:00
if ( ! getArt ( slot ) )
2011-01-18 20:56:14 +02:00
putArtifact ( slot , CArtifactInstance : : createNewArtifactInstance ( aid ) ) ;
2010-12-30 16:41:46 +02:00
else
2011-09-24 04:15:36 +03:00
tlog3 < < " Hero " < < name < < " already has artifact at " < < slot < < " , omitting giving " < < aid < < std : : endl ;
2010-07-08 08:52:11 +03:00
}
else
2012-12-03 19:00:17 +03:00
dst - > setCreature ( stackNo - warMachinesGiven , stack . creature , count ) ;
2010-07-08 08:52:11 +03:00
}
}
2009-02-01 16:11:41 +02:00
void CGHeroInstance : : initHeroDefInfo ( )
{
2012-09-23 21:01:04 +03:00
if ( ! defInfo | | defInfo - > id ! = Obj : : HERO )
2009-02-01 16:11:41 +02:00
{
defInfo = new CGDefInfo ( ) ;
2012-09-23 21:01:04 +03:00
defInfo - > id = Obj : : HERO ;
2009-02-01 16:11:41 +02:00
defInfo - > subid = subID ;
defInfo - > printPriority = 0 ;
defInfo - > visitDir = 0xff ;
}
for ( int i = 0 ; i < 6 ; i + + )
{
2009-07-13 00:17:35 +03:00
defInfo - > blockMap [ i ] = 255 ;
defInfo - > visitMap [ i ] = 0 ;
defInfo - > coverageMap [ i ] = 0 ;
2009-12-22 23:53:50 +02:00
defInfo - > shadowCoverage [ i ] = 0 ;
2009-02-01 16:11:41 +02:00
}
defInfo - > blockMap [ 5 ] = 253 ;
defInfo - > visitMap [ 5 ] = 2 ;
2009-07-13 00:17:35 +03:00
defInfo - > coverageMap [ 4 ] = defInfo - > coverageMap [ 5 ] = 224 ;
2009-02-01 16:11:41 +02:00
}
2008-12-22 19:48:41 +02:00
CGHeroInstance : : ~ CGHeroInstance ( )
{
}
bool CGHeroInstance : : needsLastStack ( ) const
{
return true ;
}
2008-12-27 03:01:59 +02:00
void CGHeroInstance : : onHeroVisit ( const CGHeroInstance * h ) const
2008-12-22 19:48:41 +02:00
{
2009-06-28 11:21:50 +03:00
if ( h = = this ) return ; //exclude potential self-visiting
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : HERO )
2008-12-22 19:48:41 +02:00
{
2010-08-13 13:46:08 +03:00
if ( cb - > gameState ( ) - > getPlayerRelations ( tempOwner , h - > tempOwner ) ) //our or ally hero
2009-02-14 21:12:40 +02:00
{
//exchange
2010-08-12 18:54:25 +03:00
cb - > heroExchange ( h - > id , id ) ;
2009-02-14 21:12:40 +02:00
}
2009-09-13 01:17:23 +03:00
else //battle
2009-02-14 21:12:40 +02:00
{
2009-09-13 01:17:23 +03:00
if ( visitedTown ) //we're in town
visitedTown - > onHeroVisit ( h ) ; //town will handle attacking
else
cb - > startBattleI ( h , this ) ;
2009-02-14 21:12:40 +02:00
}
2008-12-22 19:48:41 +02:00
}
2012-09-23 21:01:04 +03:00
else if ( ID = = Obj : : PRISON )
2008-12-22 19:48:41 +02:00
{
2009-04-24 00:09:10 +03:00
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . soundID = soundBase : : ROGUE ;
2009-02-14 21:12:40 +02:00
if ( cb - > getHeroCount ( h - > tempOwner , false ) < 8 ) //free hero slot
{
2009-02-20 12:36:15 +02:00
cb - > changeObjPos ( id , pos + int3 ( 1 , 0 , 0 ) , 0 ) ;
2012-09-23 21:01:04 +03:00
cb - > setObjProperty ( id , ObjProperty : : ID , Obj : : HERO ) ; //set ID to 34
2009-02-14 21:12:40 +02:00
cb - > giveHero ( id , h - > tempOwner ) ; //recreates def and adds hero to player
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 102 ) ;
}
else //already 8 wandering heroes
{
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 103 ) ;
}
2009-04-24 00:09:10 +03:00
cb - > showInfoDialog ( & iw ) ;
2008-12-22 19:48:41 +02:00
}
}
const std : : string & CGHeroInstance : : getBiography ( ) const
{
if ( biography . length ( ) )
return biography ;
2012-12-16 16:47:53 +03:00
return type - > biography ;
2008-12-22 19:48:41 +02:00
}
2012-12-25 20:23:26 +03:00
void CGHeroInstance : : initObj ( ) //TODO: use bonus system
2008-12-22 19:48:41 +02:00
{
2008-12-27 03:01:59 +02:00
blockVisit = true ;
2010-07-06 10:32:40 +03:00
speciality . growthsWithLevel = false ;
2010-07-07 05:29:07 +03:00
if ( ! type )
return ; //TODO support prison
2010-08-17 17:58:13 +03:00
for ( std : : vector < SSpecialtyInfo > : : const_iterator it = type - > spec . begin ( ) ; it ! = type - > spec . end ( ) ; it + + )
2010-07-06 10:32:40 +03:00
{
2010-11-20 02:03:31 +02:00
Bonus * bonus = new Bonus ( ) ;
bonus - > val = it - > val ;
2011-02-20 20:32:39 +02:00
bonus - > sid = id ; //from the hero, speciality has no unique id
2010-11-20 02:03:31 +02:00
bonus - > duration = Bonus : : PERMANENT ;
bonus - > source = Bonus : : HERO_SPECIAL ;
2010-07-06 10:32:40 +03:00
switch ( it - > type )
{
case 1 : // creature speciality
{
2010-07-12 13:20:25 +03:00
speciality . growthsWithLevel = true ;
const CCreature & specCreature = * VLC - > creh - > creatures [ it - > additionalinfo ] ; //creature in which we have specialty
int creLevel = specCreature . level ;
2012-12-25 20:23:26 +03:00
if ( ! creLevel )
2010-07-07 05:29:07 +03:00
{
2010-07-08 22:10:26 +03:00
if ( it - > additionalinfo = = 146 )
2010-07-07 05:29:07 +03:00
creLevel = 5 ; //treat ballista as 5-level
else
{
2010-07-12 13:20:25 +03:00
tlog2 < < " Warning: unknown level of " < < specCreature . namePl < < std : : endl ;
2010-07-07 05:29:07 +03:00
continue ;
}
}
2010-11-20 02:03:31 +02:00
bonus - > limiter . reset ( new CCreatureTypeLimiter ( specCreature , true ) ) ; //with upgrades
2012-07-19 21:52:44 +03:00
bonus - > type = Bonus : : PRIMARY_SKILL ;
2010-11-20 02:03:31 +02:00
bonus - > additionalInfo = it - > additionalinfo ;
bonus - > valType = Bonus : : ADDITIVE_VALUE ;
2010-07-12 13:20:25 +03:00
2010-11-20 02:03:31 +02:00
bonus - > subtype = PrimarySkill : : ATTACK ;
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2010-07-12 13:20:25 +03:00
2011-02-21 06:13:00 +02:00
bonus = new Bonus ( * bonus ) ;
2010-11-20 02:03:31 +02:00
bonus - > subtype = PrimarySkill : : DEFENSE ;
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2010-07-13 22:55:13 +03:00
//values will be calculated later
2010-07-12 13:20:25 +03:00
2011-02-21 06:13:00 +02:00
bonus = new Bonus ( * bonus ) ;
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : STACKS_SPEED ;
bonus - > val = 1 ; //+1 speed
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2010-07-06 10:32:40 +03:00
}
break ;
case 2 : //secondary skill
speciality . growthsWithLevel = true ;
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : SPECIAL_SECONDARY_SKILL ; //needs to be recalculated with level, based on this value
bonus - > valType = Bonus : : BASE_NUMBER ; // to receive nonzero value
bonus - > subtype = it - > subtype ; //skill id
bonus - > val = it - > val ; //value per level, in percent
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2011-02-21 06:13:00 +02:00
bonus = new Bonus ( * bonus ) ;
2010-07-07 15:51:28 +03:00
switch ( it - > additionalinfo )
2010-07-06 10:32:40 +03:00
{
case 0 : //normal
2010-11-20 02:03:31 +02:00
bonus - > valType = Bonus : : PERCENT_TO_BASE ;
2010-07-06 10:32:40 +03:00
break ;
case 1 : //when it's navigation or there's no 'base' at all
2010-11-20 02:03:31 +02:00
bonus - > valType = Bonus : : PERCENT_TO_ALL ;
2010-07-06 10:32:40 +03:00
break ;
}
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : SECONDARY_SKILL_PREMY ; //value will be calculated later
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2010-07-06 10:32:40 +03:00
break ;
2011-09-24 04:15:36 +03:00
case 3 : //spell damage bonus, level dependent but calculated elsewhere
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : SPECIAL_SPELL_LEV ;
bonus - > subtype = it - > subtype ;
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2010-07-06 10:32:40 +03:00
break ;
case 4 : //creature stat boost
2010-07-08 22:10:26 +03:00
switch ( it - > subtype )
{
case 1 : //attack
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : PRIMARY_SKILL ;
bonus - > subtype = PrimarySkill : : ATTACK ;
2010-07-15 22:38:15 +03:00
break ;
2010-07-08 22:10:26 +03:00
case 2 : //defense
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : PRIMARY_SKILL ;
bonus - > subtype = PrimarySkill : : DEFENSE ;
2010-07-15 22:38:15 +03:00
break ;
2010-07-17 16:11:12 +03:00
case 3 :
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : CREATURE_DAMAGE ;
bonus - > subtype = 0 ; //both min and max
2010-07-15 22:38:15 +03:00
break ;
case 4 : //hp
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : STACK_HEALTH ;
2010-07-08 22:10:26 +03:00
break ;
case 5 :
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : STACKS_SPEED ;
2010-07-08 22:10:26 +03:00
break ;
default :
2010-07-15 22:38:15 +03:00
continue ;
2010-07-08 22:10:26 +03:00
}
2010-11-20 02:03:31 +02:00
bonus - > valType = Bonus : : ADDITIVE_VALUE ;
bonus - > limiter . reset ( new CCreatureTypeLimiter ( * VLC - > creh - > creatures [ it - > additionalinfo ] , true ) ) ;
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2010-07-06 10:32:40 +03:00
break ;
case 5 : //spell damage bonus in percent
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : SPECIFIC_SPELL_DAMAGE ;
bonus - > valType = Bonus : : BASE_NUMBER ; // current spell system is screwed
bonus - > subtype = it - > subtype ; //spell id
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2010-07-06 10:32:40 +03:00
break ;
case 6 : //damage bonus for bless (Adela)
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : SPECIAL_BLESS_DAMAGE ;
bonus - > subtype = it - > subtype ; //spell id if you ever wanted to use it otherwise
bonus - > additionalInfo = it - > additionalinfo ; //damage factor
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2010-07-06 10:32:40 +03:00
break ;
case 7 : //maxed mastery for spell
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : MAXED_SPELL ;
bonus - > subtype = it - > subtype ; //spell i
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2010-07-06 10:32:40 +03:00
break ;
case 8 : //peculiar spells - enchantments
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : SPECIAL_PECULIAR_ENCHANT ;
bonus - > subtype = it - > subtype ; //spell id
bonus - > additionalInfo = it - > additionalinfo ; //0, 1 for Coronius
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2010-07-06 13:02:38 +03:00
break ;
2010-07-06 10:32:40 +03:00
case 9 : //upgrade creatures
2010-07-07 15:20:15 +03:00
{
2010-12-17 20:47:07 +02:00
std : : vector < ConstTransitivePtr < CCreature > > * creatures = & VLC - > creh - > creatures ;
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : SPECIAL_UPGRADE ;
bonus - > subtype = it - > subtype ; //base id
bonus - > additionalInfo = it - > additionalinfo ; //target id
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2011-02-21 06:13:00 +02:00
bonus = new Bonus ( * bonus ) ;
2010-07-17 09:49:16 +03:00
for ( std : : set < ui32 > : : iterator i = ( * creatures ) [ it - > subtype ] - > upgrades . begin ( ) ;
i ! = ( * creatures ) [ it - > subtype ] - > upgrades . end ( ) ; i + + )
2010-07-06 10:32:40 +03:00
{
2010-11-20 02:03:31 +02:00
bonus - > subtype = * i ; //propagate for regular upgrades of base creature
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2011-02-21 06:13:00 +02:00
bonus = new Bonus ( * bonus ) ;
2010-07-06 10:32:40 +03:00
}
2011-12-14 00:23:17 +03:00
vstd : : clear_pointer ( bonus ) ;
2010-07-06 10:32:40 +03:00
break ;
2010-07-07 15:20:15 +03:00
}
2010-07-06 10:32:40 +03:00
case 10 : //resource generation
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : GENERATE_RESOURCE ;
bonus - > subtype = it - > subtype ;
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2010-07-06 10:32:40 +03:00
break ;
case 11 : //starting skill with mastery (Adrienne)
2010-11-19 00:06:56 +02:00
cb - > changeSecSkill ( id , it - > val , it - > additionalinfo ) ; //simply give it and forget
2010-07-06 10:32:40 +03:00
break ;
case 12 : //army speed
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : STACKS_SPEED ;
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2010-07-06 10:32:40 +03:00
break ;
case 13 : //Dragon bonuses (Mutare)
2010-11-20 02:03:31 +02:00
bonus - > type = Bonus : : PRIMARY_SKILL ;
bonus - > valType = Bonus : : ADDITIVE_VALUE ;
2010-07-16 09:18:41 +03:00
switch ( it - > subtype )
{
case 1 :
2010-11-20 02:03:31 +02:00
bonus - > subtype = PrimarySkill : : ATTACK ;
2010-07-16 09:18:41 +03:00
break ;
case 2 :
2010-11-20 02:03:31 +02:00
bonus - > subtype = PrimarySkill : : DEFENSE ;
2010-07-16 09:18:41 +03:00
break ;
}
2010-11-20 02:03:31 +02:00
bonus - > limiter . reset ( new HasAnotherBonusLimiter ( Bonus : : DRAGON_NATURE ) ) ;
2010-11-19 00:06:56 +02:00
speciality . addNewBonus ( bonus ) ;
2010-07-06 13:02:38 +03:00
break ;
2010-07-06 10:32:40 +03:00
default :
tlog2 < < " Unexpected hero speciality " < < type < < ' \n ' ;
}
}
2010-07-17 20:02:11 +03:00
//initialize bonuses
for ( std : : vector < std : : pair < ui8 , ui8 > > : : iterator it = secSkills . begin ( ) ; it ! = secSkills . end ( ) ; it + + )
2010-07-19 21:30:51 +03:00
updateSkill ( it - > first , it - > second ) ;
2010-07-13 22:55:13 +03:00
UpdateSpeciality ( ) ;
2010-08-15 10:39:07 +03:00
mana = manaLimit ( ) ; //after all bonuses are taken into account, make sure this line is the last one
2011-02-11 14:27:38 +02:00
type - > name = name ;
2010-07-13 22:55:13 +03:00
}
2012-12-25 20:23:26 +03:00
void CGHeroInstance : : UpdateSpeciality ( ) //TODO: calculate special value of bonuses on-the-fly?
2010-07-13 22:55:13 +03:00
{
if ( speciality . growthsWithLevel )
{
2010-12-17 20:47:07 +02:00
std : : vector < ConstTransitivePtr < CCreature > > & creatures = VLC - > creh - > creatures ;
2010-11-19 00:06:56 +02:00
2011-07-13 21:39:02 +03:00
BOOST_FOREACH ( Bonus * it , speciality . getBonusList ( ) )
2010-07-13 22:55:13 +03:00
{
switch ( it - > type )
{
case Bonus : : SECONDARY_SKILL_PREMY :
2010-07-21 12:00:24 +03:00
it - > val = ( speciality . valOfBonuses ( Bonus : : SPECIAL_SECONDARY_SKILL , it - > subtype ) * level ) ;
break ; //use only hero skills as bonuses to avoid feedback loop
2010-07-13 22:55:13 +03:00
case Bonus : : PRIMARY_SKILL : //for crearures, that is
2010-12-17 20:47:07 +02:00
int creLevel = creatures [ it - > additionalInfo ] - > level ;
2010-07-13 22:55:13 +03:00
if ( ! creLevel )
{
if ( it - > additionalInfo = = 146 )
creLevel = 5 ; //treat ballista as 5-level
else
{
2010-12-17 20:47:07 +02:00
tlog2 < < " Warning: unknown level of " < < creatures [ it - > additionalInfo ] - > namePl < < std : : endl ;
2010-07-13 22:55:13 +03:00
continue ;
}
}
double primSkillModifier = ( int ) ( level / creLevel ) / 20.0 ;
2010-07-15 23:09:21 +03:00
int param ;
2010-07-13 22:55:13 +03:00
switch ( it - > subtype )
{
case PrimarySkill : : ATTACK :
2012-08-30 17:57:24 +03:00
param = creatures [ it - > additionalInfo ] - > Attack ( ) ;
2010-07-15 23:09:21 +03:00
break ;
2010-07-13 22:55:13 +03:00
case PrimarySkill : : DEFENSE :
2012-08-30 17:57:24 +03:00
param = creatures [ it - > additionalInfo ] - > Defense ( ) ;
2010-07-13 22:55:13 +03:00
break ;
2012-07-06 23:19:54 +03:00
default :
assert ( 0 ) ;
param = 0 ;
2010-07-13 22:55:13 +03:00
}
2010-07-15 23:09:21 +03:00
it - > val = ceil ( param * ( 1 + primSkillModifier ) ) - param ; //yep, overcomplicated but matches original
2010-07-13 22:55:13 +03:00
break ;
}
}
}
2008-12-22 19:48:41 +02:00
}
2010-07-19 21:30:51 +03:00
void CGHeroInstance : : updateSkill ( int which , int val )
2010-07-17 20:02:11 +03:00
{
2011-01-06 22:00:19 +02:00
if ( which = = LEADERSHIP | | which = = LUCK )
2011-02-21 06:13:00 +02:00
{ //luck-> VLC->generaltexth->arraytxt[73+luckSkill]; VLC->generaltexth->arraytxt[104+moraleSkill]
2011-01-06 22:00:19 +02:00
bool luck = which = = LUCK ;
Bonus : : BonusType type [ ] = { Bonus : : MORALE , Bonus : : LUCK } ;
2012-09-20 19:55:21 +03:00
Bonus * b = getBonusLocalFirst ( Selector : : type ( type [ luck ] ) & & Selector : : sourceType ( Bonus : : SECONDARY_SKILL ) ) ;
2011-01-06 22:00:19 +02:00
if ( ! b )
{
b = new Bonus ( Bonus : : PERMANENT , type [ luck ] , Bonus : : SECONDARY_SKILL , + val , which , which , Bonus : : BASE_NUMBER ) ;
addNewBonus ( b ) ;
}
else
b - > val = + val ;
}
2011-03-05 18:38:22 +02:00
else if ( which = = DIPLOMACY ) //surrender discount: 20% per level
{
2012-07-19 21:52:44 +03:00
2012-09-20 19:55:21 +03:00
if ( Bonus * b = getBonusLocalFirst ( Selector : : type ( Bonus : : SURRENDER_DISCOUNT ) & & Selector : : sourceType ( Bonus : : SECONDARY_SKILL ) ) )
2011-03-05 18:38:22 +02:00
b - > val = + val ;
else
addNewBonus ( new Bonus ( Bonus : : PERMANENT , Bonus : : SURRENDER_DISCOUNT , Bonus : : SECONDARY_SKILL , val * 20 , which ) ) ;
}
2011-01-06 22:00:19 +02:00
2010-07-17 20:02:11 +03:00
int skillVal = 0 ;
switch ( which )
{
2011-02-12 20:48:11 +02:00
case ARCHERY :
2010-07-17 20:02:11 +03:00
switch ( val )
{
case 1 :
skillVal = 10 ; break ;
case 2 :
skillVal = 25 ; break ;
case 3 :
skillVal = 50 ; break ;
}
break ;
2011-02-12 20:48:11 +02:00
case LOGISTICS :
2010-07-17 20:02:11 +03:00
skillVal = 10 * val ; break ;
2011-02-12 20:48:11 +02:00
case NAVIGATION :
2010-07-17 20:02:11 +03:00
skillVal = 50 * val ; break ;
2011-02-12 20:48:11 +02:00
case MYSTICISM :
2010-07-17 20:02:11 +03:00
skillVal = val ; break ;
2011-02-12 20:48:11 +02:00
case EAGLE_EYE :
2010-07-17 20:02:11 +03:00
skillVal = 30 + 10 * val ; break ;
2011-02-12 20:48:11 +02:00
case NECROMANCY :
2010-07-17 20:02:11 +03:00
skillVal = 10 * val ; break ;
2011-02-12 20:48:11 +02:00
case LEARNING :
skillVal = 5 * val ; break ;
case OFFENCE :
2010-07-17 20:02:11 +03:00
skillVal = 10 * val ; break ;
2011-02-12 20:48:11 +02:00
case ARMORER :
2010-07-17 20:02:11 +03:00
skillVal = 5 * val ; break ;
2011-02-12 20:48:11 +02:00
case INTELLIGENCE :
2010-10-25 22:37:21 +03:00
skillVal = 25 < < ( val - 1 ) ; break ;
2011-02-12 20:48:11 +02:00
case SORCERY :
2010-07-17 20:02:11 +03:00
skillVal = 5 * val ; break ;
2011-02-12 20:48:11 +02:00
case RESISTANCE :
2010-10-25 22:37:21 +03:00
skillVal = 5 < < ( val - 1 ) ; break ;
2011-02-12 20:48:11 +02:00
case FIRST_AID :
2010-07-17 20:02:11 +03:00
skillVal = 25 + 25 * val ; break ;
2011-06-24 21:16:28 +03:00
case ESTATES :
skillVal = 125 < < ( val - 1 ) ; break ;
2010-07-17 20:02:11 +03:00
}
2012-07-19 21:52:44 +03:00
2011-02-12 18:12:48 +02:00
int skillValType = skillVal ? Bonus : : BASE_NUMBER : Bonus : : INDEPENDENT_MIN ;
2011-07-13 21:39:02 +03:00
if ( Bonus * b = getBonusList ( ) . getFirst ( Selector : : typeSubtype ( Bonus : : SECONDARY_SKILL_PREMY , which ) & & Selector : : sourceType ( Bonus : : SECONDARY_SKILL ) ) ) //only local hero bonus
2010-07-17 20:02:11 +03:00
{
2011-02-12 18:12:48 +02:00
b - > val = skillVal ;
b - > valType = skillValType ;
2010-07-17 20:02:11 +03:00
}
2011-02-12 18:12:48 +02:00
else
{
Bonus * bonus = new Bonus ( Bonus : : PERMANENT , Bonus : : SECONDARY_SKILL_PREMY , id , skillVal , ID , which , skillValType ) ;
bonus - > source = Bonus : : SECONDARY_SKILL ;
addNewBonus ( bonus ) ;
}
2012-07-19 21:52:44 +03:00
2010-07-17 20:02:11 +03:00
}
2009-02-20 14:39:27 +02:00
void CGHeroInstance : : setPropertyDer ( ui8 what , ui32 val )
{
2010-05-02 21:20:26 +03:00
if ( what = = ObjProperty : : PRIMARY_STACK_COUNT )
setStackCount ( 0 , val ) ;
2009-02-20 14:39:27 +02:00
}
2009-03-12 01:25:59 +02:00
2009-04-12 04:48:50 +03:00
double CGHeroInstance : : getHeroStrength ( ) const
{
return sqrt ( ( 1.0 + 0.05 * getPrimSkillLevel ( 0 ) ) * ( 1.0 + 0.05 * getPrimSkillLevel ( 1 ) ) ) ;
}
2011-06-21 12:31:08 +03:00
ui64 CGHeroInstance : : getTotalStrength ( ) const
2009-04-12 04:48:50 +03:00
{
2009-04-16 03:28:54 +03:00
double ret = getHeroStrength ( ) * getArmyStrength ( ) ;
2011-06-21 12:31:08 +03:00
return ( ui64 ) ret ;
2009-04-12 04:48:50 +03:00
}
2012-09-23 21:01:04 +03:00
TExpType CGHeroInstance : : calculateXp ( TExpType exp ) const
2011-02-12 20:48:11 +02:00
{
2011-12-14 00:23:17 +03:00
return exp * ( 100 + valOfBonuses ( Bonus : : SECONDARY_SKILL_PREMY , CGHeroInstance : : LEARNING ) ) / 100.0 ;
2011-02-12 20:48:11 +02:00
}
2010-07-21 13:09:29 +03:00
ui8 CGHeroInstance : : getSpellSchoolLevel ( const CSpell * spell , int * outSelectedSchool ) const
2009-04-15 17:03:31 +03:00
{
2010-07-21 13:09:29 +03:00
si16 skill = - 1 ; //skill level
# define TRY_SCHOOL(schoolName, schoolMechanicsId, schoolOutId) \
2010-07-22 03:32:45 +03:00
if ( spell - > schoolName ) \
2010-07-21 13:09:29 +03:00
{ \
2010-12-23 22:18:10 +02:00
int thisSchool = std : : max < int > ( getSecSkillLevel ( \
static_cast < CGHeroInstance : : SecondarySkill > ( 14 + ( schoolMechanicsId ) ) ) , \
valOfBonuses ( Bonus : : MAGIC_SCHOOL_SKILL , 1 < < ( schoolMechanicsId ) ) ) ; \
2010-07-21 13:09:29 +03:00
if ( thisSchool > skill ) \
{ \
skill = thisSchool ; \
if ( outSelectedSchool ) \
* outSelectedSchool = schoolOutId ; \
} \
}
TRY_SCHOOL ( fire , 0 , 1 )
TRY_SCHOOL ( air , 1 , 0 )
TRY_SCHOOL ( water , 2 , 2 )
TRY_SCHOOL ( earth , 3 , 3 )
2010-07-28 13:09:15 +03:00
# undef TRY_SCHOOL
2010-07-21 13:09:29 +03:00
2009-07-21 16:53:26 +03:00
2011-12-14 00:23:17 +03:00
vstd : : amax ( skill , valOfBonuses ( Bonus : : MAGIC_SCHOOL_SKILL , 0 ) ) ; //any school bonus
vstd : : amax ( skill , valOfBonuses ( Bonus : : SPELL , spell - > id ) ) ; //given by artifact or other effect
2010-08-29 22:03:10 +03:00
if ( hasBonusOfType ( Bonus : : MAXED_SPELL , spell - > id ) ) //hero speciality (Daremyth, Melodia)
skill = 3 ;
2010-07-21 13:09:29 +03:00
assert ( skill > = 0 & & skill < = 3 ) ;
2009-04-15 17:03:31 +03:00
return skill ;
}
* support for new hero bonuses (BLOCK_MORALE, SECONDARY_SKILL_PREMY (archery), AIR_SPELL_DMG_PREMY, EARTH_SPELL_DMG_PREMY, FIRE_SPELL_DMG_PREMY, WATER_SPELL_DMG_PREMY, BLOCK_SPELLS_ABOVE_LEVEL, SPELL_IMMUNITY, BLOCK_MORALE, FIRE_SPELLS, AIR_SPELLS, WATER_SPELLS, EARTH_SPELLS, SPELL, SPELLS_OF_LEVEL). It means that following artifacts are now supported:
- Orb of the Firmament
- Orb of Silt
- Orb of Tempestuous Fire
- Orb of Driving Rain
- Bow of Elven Cherrywood
- Bowstring of the Unicorn's Mane
- Angel Feather Arrows
- Tome of Fire Magic
- Tome of Air Magic
- Tome of Water Magic
- Tome of Earth Magic
- Recanter's Cloak
- Orb of Inhibition
- Pendant of Dispassion
- Pendant of Second Sight
- Pendant of Holiness
- Pendant of Life
- Pendant of Death
- Pendant of Free Will
- Pendant of Negativity
- Pendant of Total Recall
- Spellbinder's Hat
- Spirit of Oppression
- Sphere of Permanence
I hope I listed them all here :). Please try them and report if something's wrong.
2009-05-03 19:14:16 +03:00
bool CGHeroInstance : : canCastThisSpell ( const CSpell * spell ) const
{
if ( ! getArt ( 17 ) ) //if hero has no spellbook
return false ;
2010-02-10 04:56:00 +02:00
if ( vstd : : contains ( spells , spell - > id ) //hero has this spell in spellbook
2010-05-02 21:20:26 +03:00
| | ( spell - > air & & hasBonusOfType ( Bonus : : AIR_SPELLS ) ) // this is air spell and hero can cast all air spells
| | ( spell - > fire & & hasBonusOfType ( Bonus : : FIRE_SPELLS ) ) // this is fire spell and hero can cast all fire spells
| | ( spell - > water & & hasBonusOfType ( Bonus : : WATER_SPELLS ) ) // this is water spell and hero can cast all water spells
| | ( spell - > earth & & hasBonusOfType ( Bonus : : EARTH_SPELLS ) ) // this is earth spell and hero can cast all earth spells
| | hasBonusOfType ( Bonus : : SPELL , spell - > id )
| | hasBonusOfType ( Bonus : : SPELLS_OF_LEVEL , spell - > level )
* support for new hero bonuses (BLOCK_MORALE, SECONDARY_SKILL_PREMY (archery), AIR_SPELL_DMG_PREMY, EARTH_SPELL_DMG_PREMY, FIRE_SPELL_DMG_PREMY, WATER_SPELL_DMG_PREMY, BLOCK_SPELLS_ABOVE_LEVEL, SPELL_IMMUNITY, BLOCK_MORALE, FIRE_SPELLS, AIR_SPELLS, WATER_SPELLS, EARTH_SPELLS, SPELL, SPELLS_OF_LEVEL). It means that following artifacts are now supported:
- Orb of the Firmament
- Orb of Silt
- Orb of Tempestuous Fire
- Orb of Driving Rain
- Bow of Elven Cherrywood
- Bowstring of the Unicorn's Mane
- Angel Feather Arrows
- Tome of Fire Magic
- Tome of Air Magic
- Tome of Water Magic
- Tome of Earth Magic
- Recanter's Cloak
- Orb of Inhibition
- Pendant of Dispassion
- Pendant of Second Sight
- Pendant of Holiness
- Pendant of Life
- Pendant of Death
- Pendant of Free Will
- Pendant of Negativity
- Pendant of Total Recall
- Spellbinder's Hat
- Spirit of Oppression
- Sphere of Permanence
I hope I listed them all here :). Please try them and report if something's wrong.
2009-05-03 19:14:16 +03:00
)
return true ;
return false ;
}
2009-09-01 01:04:00 +03:00
/**
* Calculates what creatures and how many to be raised from a battle .
* @ param battleResult The results of the battle .
* @ return Returns a pair with the first value indicating the ID of the creature
* type and second value the amount . Both values are returned as - 1 if necromancy
* could not be applied .
*/
2010-11-22 02:34:46 +02:00
CStackBasicDescriptor CGHeroInstance : : calculateNecromancy ( const BattleResult & battleResult ) const
2009-09-01 01:04:00 +03:00
{
2010-12-23 22:18:10 +02:00
const ui8 necromancyLevel = getSecSkillLevel ( CGHeroInstance : : NECROMANCY ) ;
2009-09-01 01:04:00 +03:00
// Hero knows necromancy.
2012-07-19 21:52:44 +03:00
if ( necromancyLevel > 0 )
2010-05-14 05:18:37 +03:00
{
2011-02-12 18:12:48 +02:00
double necromancySkill = valOfBonuses ( Bonus : : SECONDARY_SKILL_PREMY , NECROMANCY ) / 100.0 ;
2011-12-14 00:23:17 +03:00
vstd : : amin ( necromancySkill , 1.0 ) ; //it's impossible to raise more creatures than all...
2009-09-24 16:23:52 +03:00
const std : : map < ui32 , si32 > & casualties = battleResult . casualties [ ! battleResult . winner ] ;
2009-09-01 01:04:00 +03:00
ui32 raisedUnits = 0 ;
// Figure out what to raise and how many.
const ui32 creatureTypes [ ] = { 56 , 58 , 60 , 64 } ; // IDs for Skeletons, Walking Dead, Wights and Liches respectively.
2010-05-02 21:20:26 +03:00
const bool improvedNecromancy = hasBonusOfType ( Bonus : : IMPROVED_NECROMANCY ) ;
2010-05-14 05:18:37 +03:00
const CCreature * raisedUnitType = VLC - > creh - > creatures [ creatureTypes [ improvedNecromancy ? necromancyLevel : 0 ] ] ;
const ui32 raisedUnitHP = raisedUnitType - > valOfBonuses ( Bonus : : STACK_HEALTH ) ;
2009-09-01 01:04:00 +03:00
2010-05-14 05:18:37 +03:00
//calculate creatures raised from each defeated stack
for ( std : : map < ui32 , si32 > : : const_iterator it = casualties . begin ( ) ; it ! = casualties . end ( ) ; it + + )
{
// Get lost enemy hit points convertible to units.
2011-06-23 16:50:13 +03:00
CCreature * c = VLC - > creh - > creatures [ it - > first ] ;
2011-06-25 18:05:01 +03:00
if ( c - > isLiving ( ) )
2011-06-23 16:50:13 +03:00
{
const ui32 raisedHP = c - > valOfBonuses ( Bonus : : STACK_HEALTH ) * it - > second * necromancySkill ;
raisedUnits + = std : : min < ui32 > ( raisedHP / raisedUnitHP , it - > second * necromancySkill ) ; //limit to % of HP and % of original stack count
}
2010-05-14 05:18:37 +03:00
}
2009-09-01 01:04:00 +03:00
// Make room for new units.
2010-05-02 21:20:26 +03:00
int slot = getSlotFor ( raisedUnitType - > idNumber ) ;
2012-07-19 21:52:44 +03:00
if ( slot = = - 1 )
2009-09-24 16:23:52 +03:00
{
2009-09-01 01:04:00 +03:00
// If there's no room for unit, try it's upgraded version 2/3rds the size.
2010-05-02 21:20:26 +03:00
raisedUnitType = VLC - > creh - > creatures [ * raisedUnitType - > upgrades . begin ( ) ] ;
2009-09-01 01:04:00 +03:00
raisedUnits = ( raisedUnits * 2 ) / 3 ;
2010-05-02 21:20:26 +03:00
slot = getSlotFor ( raisedUnitType - > idNumber ) ;
2009-09-01 01:04:00 +03:00
}
if ( raisedUnits < = 0 )
raisedUnits = 1 ;
2010-11-22 02:34:46 +02:00
return CStackBasicDescriptor ( raisedUnitType - > idNumber , raisedUnits ) ;
2009-09-01 01:04:00 +03:00
}
2010-11-22 02:34:46 +02:00
return CStackBasicDescriptor ( ) ;
2009-09-01 01:04:00 +03:00
}
/**
* Show the necromancy dialog with information about units raised .
* @ param raisedStack Pair where the first element represents ID of the raised creature
* and the second element the amount .
*/
2010-11-22 02:34:46 +02:00
void CGHeroInstance : : showNecromancyDialog ( const CStackBasicDescriptor & raisedStack ) const
2009-09-01 01:04:00 +03:00
{
InfoWindow iw ;
2011-06-23 16:50:13 +03:00
iw . soundID = soundBase : : pickup01 + ran ( ) % 7 ;
2009-09-01 01:04:00 +03:00
iw . player = tempOwner ;
2010-04-02 05:07:40 +03:00
iw . components . push_back ( Component ( raisedStack ) ) ;
2009-09-01 01:04:00 +03:00
2010-04-02 05:07:40 +03:00
if ( raisedStack . count > 1 ) // Practicing the dark arts of necromancy, ... (plural)
2012-07-19 21:52:44 +03:00
{
2009-09-01 01:04:00 +03:00
iw . text . addTxt ( MetaString : : GENERAL_TXT , 145 ) ;
2010-04-02 05:07:40 +03:00
iw . text . addReplacement ( raisedStack . count ) ;
iw . text . addReplacement ( MetaString : : CRE_PL_NAMES , raisedStack . type - > idNumber ) ;
2012-07-19 21:52:44 +03:00
}
2010-04-02 05:07:40 +03:00
else // Practicing the dark arts of necromancy, ... (singular)
2012-07-19 21:52:44 +03:00
{
2009-09-01 01:04:00 +03:00
iw . text . addTxt ( MetaString : : GENERAL_TXT , 146 ) ;
2010-04-02 05:07:40 +03:00
iw . text . addReplacement ( MetaString : : CRE_SING_NAMES , raisedStack . type - > idNumber ) ;
2009-09-01 01:04:00 +03:00
}
cb - > showInfoDialog ( & iw ) ;
}
2009-03-12 01:25:59 +02:00
int3 CGHeroInstance : : getSightCenter ( ) const
{
return getPosition ( false ) ;
}
int CGHeroInstance : : getSightRadious ( ) const
{
2010-12-23 22:18:10 +02:00
return 5 + getSecSkillLevel ( CGHeroInstance : : SCOUTING ) + valOfBonuses ( Bonus : : SIGHT_RADIOUS ) ; //default + scouting
2009-04-04 01:34:31 +03:00
}
si32 CGHeroInstance : : manaRegain ( ) const
{
2010-05-02 21:20:26 +03:00
if ( hasBonusOfType ( Bonus : : FULL_MANA_REGENERATION ) )
2009-08-21 22:31:41 +03:00
return manaLimit ( ) ;
2009-08-22 15:50:23 +03:00
2012-07-19 21:52:44 +03:00
return 1 + valOfBonuses ( Bonus : : SECONDARY_SKILL_PREMY , 8 ) + valOfBonuses ( Bonus : : MANA_REGENERATION ) ; //1 + Mysticism level
2009-04-04 01:34:31 +03:00
}
2012-02-17 22:30:40 +03:00
// /**
// * Places an artifact in hero's backpack. If it's a big artifact equips it
// * or discards it if it cannot be equipped.
// */
// void CGHeroInstance::giveArtifact (ui32 aid) //use only for fixed artifacts
// {
// CArtifact * const artifact = VLC->arth->artifacts[aid]; //pointer to constant object
// CArtifactInstance *ai = CArtifactInstance::createNewArtifactInstance(artifact);
// ai->putAt(this, ai->firstAvailableSlot(this));
// }
2009-12-30 17:33:28 +02:00
2010-03-11 01:16:30 +02:00
int CGHeroInstance : : getBoatType ( ) const
{
2012-09-23 21:01:04 +03:00
switch ( type - > heroClass - > getAlignment ( ) )
2010-03-11 01:16:30 +02:00
{
2012-09-23 21:01:04 +03:00
case EAlignment : : GOOD :
return 1 ;
case EAlignment : : EVIL :
return 0 ;
case EAlignment : : NEUTRAL :
2010-03-11 01:16:30 +02:00
return 2 ;
default :
2012-04-22 10:32:45 +03:00
throw std : : runtime_error ( " Wrong alignment! " ) ;
2010-03-11 01:16:30 +02:00
}
}
void CGHeroInstance : : getOutOffsets ( std : : vector < int3 > & offsets ) const
{
static int3 dirs [ ] = { int3 ( 0 , 1 , 0 ) , int3 ( 0 , - 1 , 0 ) , int3 ( - 1 , 0 , 0 ) , int3 ( + 1 , 0 , 0 ) , int3 ( 1 , 1 , 0 ) , int3 ( - 1 , 1 , 0 ) , int3 ( 1 , - 1 , 0 ) , int3 ( - 1 , - 1 , 0 ) } ;
for ( size_t i = 0 ; i < ARRAY_COUNT ( dirs ) ; i + + )
offsets + = dirs [ i ] ;
}
int CGHeroInstance : : getSpellCost ( const CSpell * sp ) const
{
return sp - > costs [ getSpellSchoolLevel ( sp ) ] ;
}
2012-09-23 21:01:04 +03:00
void CGHeroInstance : : pushPrimSkill ( int which , int val )
2010-05-02 21:20:26 +03:00
{
2010-11-20 02:03:31 +02:00
addNewBonus ( new Bonus ( Bonus : : PERMANENT , Bonus : : PRIMARY_SKILL , Bonus : : HERO_BASE_SKILL , val , id , which ) ) ;
2010-05-02 21:20:26 +03:00
}
2011-12-14 00:23:17 +03:00
EAlignment : : EAlignment CGHeroInstance : : getAlignment ( ) const
2010-07-20 09:05:45 +03:00
{
return type - > heroClass - > getAlignment ( ) ;
}
2010-08-28 17:52:20 +03:00
void CGHeroInstance : : initExp ( )
{
exp = 40 + ( ran ( ) ) % 50 ;
level = 1 ;
}
2010-12-12 01:11:26 +02:00
std : : string CGHeroInstance : : nodeName ( ) const
{
return " Hero " + name ;
}
2010-12-30 16:41:46 +02:00
void CGHeroInstance : : putArtifact ( ui16 pos , CArtifactInstance * art )
{
assert ( ! getArt ( pos ) ) ;
2012-04-14 05:20:22 +03:00
art - > putAt ( ArtifactLocation ( this , pos ) ) ;
2010-12-30 16:41:46 +02:00
}
void CGHeroInstance : : putInBackpack ( CArtifactInstance * art )
{
putArtifact ( art - > firstBackpackSlot ( this ) , art ) ;
}
2011-01-18 20:56:14 +02:00
bool CGHeroInstance : : hasSpellbook ( ) const
{
2011-12-14 00:23:17 +03:00
return getArt ( ArtifactPosition : : SPELLBOOK ) ;
2011-01-18 20:56:14 +02:00
}
2011-02-04 16:58:14 +02:00
void CGHeroInstance : : deserializationFix ( )
{
2012-04-14 05:20:22 +03:00
artDeserializationFix ( this ) ;
2011-02-04 16:58:14 +02:00
}
2011-02-22 11:47:25 +02:00
CBonusSystemNode * CGHeroInstance : : whereShouldBeAttached ( CGameState * gs )
{
if ( visitedTown )
{
if ( inTownGarrison )
return visitedTown ;
else
return & visitedTown - > townAndVis ;
}
else
return CArmedInstance : : whereShouldBeAttached ( gs ) ;
}
2011-09-19 23:50:25 +03:00
int CGHeroInstance : : movementPointsAfterEmbark ( int MPsBefore , int basicCost , bool disembark /*= false*/ ) const
{
if ( hasBonusOfType ( Bonus : : FREE_SHIP_BOARDING ) )
2011-12-14 00:23:17 +03:00
return ( MPsBefore - basicCost ) * static_cast < double > ( maxMovePoints ( disembark ) ) / maxMovePoints ( ! disembark ) ;
2011-09-19 23:50:25 +03:00
return 0 ; //take all MPs otherwise
}
2012-01-03 04:55:26 +03:00
CGHeroInstance : : ECanDig CGHeroInstance : : diggingStatus ( ) const
{
if ( movement < maxMovePoints ( true ) )
return LACK_OF_MOVEMENT ;
2012-11-06 19:39:29 +03:00
else if ( cb - > getTile ( getPosition ( false ) ) - > terType = = ETerrainType : : WATER )
2012-01-03 04:55:26 +03:00
return WRONG_TERRAIN ;
else
{
const TerrainTile * t = cb - > getTile ( getPosition ( ) ) ;
//TODO look for hole
//CGI->mh->getTerrainDescr(h->getPosition(false), hlp, false);
if ( /*hlp.length() || */ t - > blockingObjects . size ( ) > 1 )
return TILE_OCCUPIED ;
else
return CAN_DIG ;
}
}
2012-04-14 05:20:22 +03:00
ui8 CGHeroInstance : : bearerType ( ) const
{
return ArtBearer : : HERO ;
}
2009-07-06 22:41:27 +03:00
void CGDwelling : : initObj ( )
{
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : CREATURE_GENERATOR1 :
2009-07-09 22:15:22 +03:00
{
int crid = VLC - > objh - > cregens [ subID ] ;
2010-05-02 21:20:26 +03:00
const CCreature * crs = VLC - > creh - > creatures [ crid ] ;
2009-07-09 22:15:22 +03:00
creatures . resize ( 1 ) ;
creatures [ 0 ] . second . push_back ( crid ) ;
2012-12-19 19:35:58 +03:00
if ( subID > = VLC - > generaltexth - > creGens . size ( ) ) //very messy workaround
{
int faction = VLC - > creh - > creatures [ subID ] - > faction ;
assert ( VLC - > townh - > towns [ faction ] . dwellingNames . size ( ) ) ;
hoverName = VLC - > townh - > towns [ faction ] . dwellingNames [ VLC - > creh - > creatures [ subID ] - > level - 1 ] ;
}
else
hoverName = VLC - > generaltexth - > creGens [ subID ] ;
2009-07-09 22:15:22 +03:00
if ( crs - > level > 4 )
2010-11-27 03:46:19 +02:00
putStack ( 0 , new CStackInstance ( crs , ( crs - > growth ) * 3 ) ) ;
2012-09-29 20:36:48 +03:00
if ( getOwner ( ) ! = GameConstants : : NEUTRAL_PLAYER )
2009-10-26 11:11:10 +02:00
cb - > gameState ( ) - > players [ getOwner ( ) ] . dwellings . push_back ( this ) ;
2009-07-09 22:15:22 +03:00
}
2009-07-06 22:41:27 +03:00
break ;
2009-07-09 22:15:22 +03:00
2012-09-23 21:01:04 +03:00
case Obj : : CREATURE_GENERATOR4 :
2009-07-06 22:41:27 +03:00
creatures . resize ( 4 ) ;
2009-07-26 13:43:22 +03:00
if ( subID = = 1 ) //Golem Factory
2009-07-06 22:41:27 +03:00
{
creatures [ 0 ] . second . push_back ( 32 ) ; //Stone Golem
2012-07-19 21:52:44 +03:00
creatures [ 1 ] . second . push_back ( 33 ) ; //Iron Golem
2009-07-06 22:41:27 +03:00
creatures [ 2 ] . second . push_back ( 116 ) ; //Gold Golem
creatures [ 3 ] . second . push_back ( 117 ) ; //Diamond Golem
2009-07-26 13:43:22 +03:00
//guards
2010-11-27 03:46:19 +02:00
putStack ( 0 , new CStackInstance ( 116 , 9 ) ) ;
putStack ( 1 , new CStackInstance ( 117 , 6 ) ) ;
2009-07-06 22:41:27 +03:00
}
2012-07-19 21:52:44 +03:00
else if ( subID = = 0 ) // Elemental Conflux
2009-07-06 22:41:27 +03:00
{
creatures [ 0 ] . second . push_back ( 112 ) ; //Air Elemental
2009-08-07 01:36:51 +03:00
creatures [ 1 ] . second . push_back ( 114 ) ; //Fire Elemental
creatures [ 2 ] . second . push_back ( 113 ) ; //Earth Elemental
2009-07-06 22:41:27 +03:00
creatures [ 3 ] . second . push_back ( 115 ) ; //Water Elemental
2009-07-26 13:43:22 +03:00
//guards
2010-11-27 03:46:19 +02:00
putStack ( 0 , new CStackInstance ( 113 , 12 ) ) ;
2009-07-06 22:41:27 +03:00
}
else
{
assert ( 0 ) ;
}
2009-07-09 22:15:22 +03:00
hoverName = VLC - > generaltexth - > creGens4 [ subID ] ;
2009-07-06 22:41:27 +03:00
break ;
2009-07-09 22:15:22 +03:00
2012-09-23 21:01:04 +03:00
case Obj : : REFUGEE_CAMP :
2010-07-24 00:05:49 +03:00
//is handled within newturn func
2012-07-19 21:52:44 +03:00
break ;
2010-07-24 00:05:49 +03:00
2012-09-23 21:01:04 +03:00
case Obj : : WAR_MACHINE_FACTORY :
2010-05-15 11:33:32 +03:00
creatures . resize ( 3 ) ;
2012-07-19 21:52:44 +03:00
creatures [ 0 ] . second . push_back ( 146 ) ; //Ballista
2010-05-15 11:33:32 +03:00
creatures [ 1 ] . second . push_back ( 147 ) ; //First Aid Tent
creatures [ 2 ] . second . push_back ( 148 ) ; //Ammo Cart
break ;
2009-07-06 22:41:27 +03:00
default :
assert ( 0 ) ;
break ;
}
}
2009-10-26 11:11:10 +02:00
void CGDwelling : : setProperty ( ui8 what , ui32 val )
{
switch ( what )
{
2010-05-02 21:20:26 +03:00
case ObjProperty : : OWNER : //change owner
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : CREATURE_GENERATOR1 ) //single generators
2009-10-26 11:11:10 +02:00
{
2011-12-14 00:23:17 +03:00
if ( tempOwner ! = GameConstants : : NEUTRAL_PLAYER )
2009-10-26 11:11:10 +02:00
{
2010-12-20 23:22:53 +02:00
std : : vector < ConstTransitivePtr < CGDwelling > > * dwellings = & cb - > gameState ( ) - > players [ tempOwner ] . dwellings ;
2009-10-26 11:11:10 +02:00
dwellings - > erase ( std : : find ( dwellings - > begin ( ) , dwellings - > end ( ) , this ) ) ;
}
2011-12-14 00:23:17 +03:00
if ( val ! = GameConstants : : NEUTRAL_PLAYER ) //can new owner be neutral?
2009-10-26 11:11:10 +02:00
cb - > gameState ( ) - > players [ val ] . dwellings . push_back ( this ) ;
}
break ;
2010-07-24 00:05:49 +03:00
case ObjProperty : : AVAILABLE_CREATURE :
creatures . resize ( 1 ) ;
creatures [ 0 ] . second . resize ( 1 ) ;
creatures [ 0 ] . second [ 0 ] = val ;
break ;
2009-10-26 11:11:10 +02:00
}
2010-07-24 00:05:49 +03:00
CGObjectInstance : : setProperty ( what , val ) ;
2009-10-26 11:11:10 +02:00
}
2009-07-06 22:41:27 +03:00
void CGDwelling : : onHeroVisit ( const CGHeroInstance * h ) const
{
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : REFUGEE_CAMP & & ! creatures [ 0 ] . first ) //Refugee Camp, no available cres
2010-07-24 00:05:49 +03:00
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , 44 ) ; //{%s} \n\n The camp is deserted. Perhaps you should try next week.
iw . text . addReplacement ( MetaString : : OBJ_NAMES , ID ) ;
cb - > sendAndApply ( & iw ) ;
return ;
}
2010-08-13 13:46:08 +03:00
int relations = cb - > gameState ( ) - > getPlayerRelations ( h - > tempOwner , tempOwner ) ;
2012-07-19 21:52:44 +03:00
2010-08-06 16:14:10 +03:00
if ( relations = = 1 ) //ally
return ; //do not allow recruiting or capturing
2012-07-19 21:52:44 +03:00
2010-08-06 16:14:10 +03:00
if ( ! relations & & stacksCount ( ) > 0 ) //object is guarded, owned by enemy
2009-07-09 22:15:22 +03:00
{
BlockingDialog bd ;
bd . player = h - > tempOwner ;
bd . flags = BlockingDialog : : ALLOW_CANCEL ;
bd . text . addTxt ( MetaString : : GENERAL_TXT , 421 ) ; //Much to your dismay, the %s is guarded by %s %s. Do you wish to fight the guards?
2012-09-23 21:01:04 +03:00
bd . text . addReplacement ( ID = = Obj : : CREATURE_GENERATOR1 ? MetaString : : CREGENS : MetaString : : CREGENS4 , subID ) ;
2010-11-22 02:34:46 +02:00
bd . text . addReplacement ( MetaString : : ARRAY_TXT , 176 + Slots ( ) . begin ( ) - > second - > getQuantityID ( ) * 3 ) ;
bd . text . addReplacement ( * Slots ( ) . begin ( ) - > second ) ;
2009-07-09 22:15:22 +03:00
cb - > showBlockingDialog ( & bd , boost : : bind ( & CGDwelling : : wantsFight , this , h , _1 ) ) ;
return ;
}
2012-09-23 21:01:04 +03:00
if ( ! relations & & ID ! = Obj : : WAR_MACHINE_FACTORY )
2009-10-24 22:21:32 +03:00
{
2009-07-06 22:41:27 +03:00
cb - > setOwner ( id , h - > tempOwner ) ;
2009-10-24 22:21:32 +03:00
}
2009-07-06 22:41:27 +03:00
2009-07-09 22:15:22 +03:00
BlockingDialog bd ;
bd . player = h - > tempOwner ;
bd . flags = BlockingDialog : : ALLOW_CANCEL ;
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : CREATURE_GENERATOR1 | | ID = = Obj : : CREATURE_GENERATOR4 )
2010-05-15 11:33:32 +03:00
{
2012-09-23 21:01:04 +03:00
bd . text . addTxt ( MetaString : : ADVOB_TXT , ID = = Obj : : CREATURE_GENERATOR1 ? 35 : 36 ) ; //{%s} Would you like to recruit %s? / {%s} Would you like to recruit %s, %s, %s, or %s?
bd . text . addReplacement ( ID = = Obj : : CREATURE_GENERATOR1 ? MetaString : : CREGENS : MetaString : : CREGENS4 , subID ) ;
2010-05-15 11:33:32 +03:00
for ( size_t i = 0 ; i < creatures . size ( ) ; i + + )
bd . text . addReplacement ( MetaString : : CRE_PL_NAMES , creatures [ i ] . second [ 0 ] ) ;
}
2012-09-23 21:01:04 +03:00
else if ( ID = = Obj : : REFUGEE_CAMP )
2010-07-24 00:05:49 +03:00
{
bd . text . addTxt ( MetaString : : ADVOB_TXT , 35 ) ; //{%s} Would you like to recruit %s?
bd . text . addReplacement ( MetaString : : OBJ_NAMES , ID ) ;
for ( size_t i = 0 ; i < creatures . size ( ) ; i + + )
bd . text . addReplacement ( MetaString : : CRE_PL_NAMES , creatures [ i ] . second [ 0 ] ) ;
}
2012-09-23 21:01:04 +03:00
else if ( ID = = Obj : : WAR_MACHINE_FACTORY )
2010-05-15 11:33:32 +03:00
bd . text . addTxt ( MetaString : : ADVOB_TXT , 157 ) ; //{War Machine Factory} Would you like to purchase War Machines?
else
2012-04-22 10:32:45 +03:00
throw std : : runtime_error ( " Illegal dwelling! " ) ;
2009-08-07 01:36:51 +03:00
2009-07-09 22:15:22 +03:00
cb - > showBlockingDialog ( & bd , boost : : bind ( & CGDwelling : : heroAcceptsCreatures , this , h , _1 ) ) ;
2009-07-06 22:41:27 +03:00
}
void CGDwelling : : newTurn ( ) const
{
if ( cb - > getDate ( 1 ) ! = 1 ) //not first day of week
return ;
2010-05-15 11:33:32 +03:00
//town growths and War Machines Factories are handled separately
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : TOWN | | ID = = Obj : : WAR_MACHINE_FACTORY )
2009-07-31 23:10:22 +03:00
return ;
2009-07-06 22:41:27 +03:00
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : REFUGEE_CAMP ) //if it's a refugee camp, we need to pick an available creature
2010-07-24 00:05:49 +03:00
{
cb - > setObjProperty ( id , ObjProperty : : AVAILABLE_CREATURE , VLC - > creh - > pickRandomMonster ( ) ) ;
}
2009-07-06 22:41:27 +03:00
bool change = false ;
SetAvailableCreatures sac ;
sac . creatures = creatures ;
sac . tid = id ;
for ( size_t i = 0 ; i < creatures . size ( ) ; i + + )
{
if ( creatures [ i ] . second . size ( ) )
{
2010-08-24 17:26:57 +03:00
CCreature * cre = VLC - > creh - > creatures [ creatures [ i ] . second [ 0 ] ] ;
TQuantity amount = cre - > growth * ( 1 + cre - > valOfBonuses ( Bonus : : CREATURE_GROWTH_PERCENT ) / 100 ) + cre - > valOfBonuses ( Bonus : : CREATURE_GROWTH ) ;
2012-08-11 12:06:23 +03:00
if ( VLC - > modh - > settings . DWELLINGS_ACCUMULATE_CREATURES )
2010-08-24 17:26:57 +03:00
sac . creatures [ i ] . first + = amount ;
2009-08-07 01:36:51 +03:00
else
2010-08-24 17:26:57 +03:00
sac . creatures [ i ] . first = amount ;
2009-07-06 22:41:27 +03:00
change = true ;
}
}
if ( change )
cb - > sendAndApply ( & sac ) ;
}
2009-07-09 22:15:22 +03:00
void CGDwelling : : heroAcceptsCreatures ( const CGHeroInstance * h , ui32 answer ) const
{
if ( ! answer )
return ;
int crid = creatures [ 0 ] . second [ 0 ] ;
2010-05-02 21:20:26 +03:00
CCreature * crs = VLC - > creh - > creatures [ crid ] ;
2010-11-27 22:17:28 +02:00
TQuantity count = creatures [ 0 ] . first ;
2009-07-09 22:15:22 +03:00
2012-09-23 21:01:04 +03:00
if ( crs - > level = = 1 & & ID ! = Obj : : REFUGEE_CAMP ) //first level - creatures are for free
2009-07-09 22:15:22 +03:00
{
2010-11-27 22:17:28 +02:00
if ( count ) //there are available creatures
2009-07-09 22:15:22 +03:00
{
2010-05-02 21:20:26 +03:00
int slot = h - > getSlotFor ( crid ) ;
2009-07-09 22:15:22 +03:00
if ( slot < 0 ) //no available slot
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text . addTxt ( MetaString : : GENERAL_TXT , 425 ) ; //The %s would join your hero, but there aren't enough provisions to support them.
iw . text . addReplacement ( MetaString : : CRE_PL_NAMES , crid ) ;
cb - > showInfoDialog ( & iw ) ;
}
else //give creatures
{
SetAvailableCreatures sac ;
sac . tid = id ;
sac . creatures = creatures ;
sac . creatures [ 0 ] . first = 0 ;
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text . addTxt ( MetaString : : GENERAL_TXT , 423 ) ; //%d %s join your army.
2010-11-27 22:17:28 +02:00
iw . text . addReplacement ( count ) ;
2009-07-09 22:15:22 +03:00
iw . text . addReplacement ( MetaString : : CRE_PL_NAMES , crid ) ;
cb - > showInfoDialog ( & iw ) ;
cb - > sendAndApply ( & sac ) ;
2010-11-27 22:17:28 +02:00
cb - > addToSlot ( StackLocation ( h , slot ) , crs , count ) ;
2009-07-09 22:15:22 +03:00
}
}
else //there no creatures
{
InfoWindow iw ;
iw . text . addTxt ( MetaString : : GENERAL_TXT , 422 ) ; //There are no %s here to recruit.
iw . text . addReplacement ( MetaString : : CRE_PL_NAMES , crid ) ;
iw . player = h - > tempOwner ;
cb - > sendAndApply ( & iw ) ;
}
}
2009-07-26 13:43:22 +03:00
else
2009-07-09 22:15:22 +03:00
{
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : WAR_MACHINE_FACTORY ) //pick available War Machines
2010-05-15 11:33:32 +03:00
{
//there is 1 war machine available to recruit if hero doesn't have one
SetAvailableCreatures sac ;
sac . tid = id ;
sac . creatures = creatures ;
sac . creatures [ 0 ] . first = ! h - > getArt ( 13 ) ; //ballista
sac . creatures [ 1 ] . first = ! h - > getArt ( 15 ) ; //first aid tent
sac . creatures [ 2 ] . first = ! h - > getArt ( 14 ) ; //ammo cart
cb - > sendAndApply ( & sac ) ;
}
2009-07-09 22:15:22 +03:00
OpenWindow ow ;
ow . id1 = id ;
2009-07-26 13:43:22 +03:00
ow . id2 = h - > id ;
2012-09-23 21:01:04 +03:00
ow . window = ( ID = = Obj : : CREATURE_GENERATOR1 | | ID = = Obj : : REFUGEE_CAMP )
2012-07-19 21:52:44 +03:00
? OpenWindow : : RECRUITMENT_FIRST
2009-07-26 13:43:22 +03:00
: OpenWindow : : RECRUITMENT_ALL ;
2009-07-09 22:15:22 +03:00
cb - > sendAndApply ( & ow ) ;
}
}
void CGDwelling : : wantsFight ( const CGHeroInstance * h , ui32 answer ) const
{
if ( answer )
2009-09-13 01:17:23 +03:00
cb - > startBattleI ( h , this , boost : : bind ( & CGDwelling : : fightOver , this , h , _1 ) ) ;
2009-07-09 22:15:22 +03:00
}
void CGDwelling : : fightOver ( const CGHeroInstance * h , BattleResult * result ) const
{
if ( result - > winner = = 0 )
{
onHeroVisit ( h ) ;
}
}
2009-03-12 01:25:59 +02:00
int CGTownInstance : : getSightRadious ( ) const //returns sight distance
2007-10-27 22:38:48 +03:00
{
2012-09-05 15:49:23 +03:00
if ( subID = = ETownType : : TOWER )
2009-08-24 13:23:44 +03:00
{
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : GRAIL ) ) //skyship
2009-11-11 19:45:03 +02:00
return - 1 ; //entire map
2012-09-05 15:49:23 +03:00
else if ( hasBuilt ( EBuilding : : LOOKOUT_TOWER ) ) //lookout tower
2009-08-23 18:02:21 +03:00
return 20 ;
2009-08-24 13:23:44 +03:00
}
2009-03-12 01:25:59 +02:00
return 5 ;
2007-10-27 22:38:48 +03:00
}
2009-07-09 22:15:22 +03:00
2009-09-21 12:00:33 +03:00
void CGTownInstance : : setPropertyDer ( ui8 what , ui32 val )
{
///this is freakin' overcomplicated solution
switch ( what )
{
case 11 : //add visitor of town building
2010-05-02 21:20:26 +03:00
bonusingBuildings [ val ] - > setProperty ( ObjProperty : : VISITORS , visitingHero - > id ) ;
2009-09-21 12:00:33 +03:00
break ;
2009-10-06 10:55:39 +03:00
case 12 :
bonusingBuildings [ val ] - > setProperty ( 12 , 0 ) ;
break ;
2009-12-30 22:49:10 +02:00
case 13 : //add garrisoned hero to visitors
2010-05-02 21:20:26 +03:00
bonusingBuildings [ val ] - > setProperty ( ObjProperty : : VISITORS , garrisonHero - > id ) ;
2009-12-30 22:49:10 +02:00
break ;
2010-01-28 18:15:46 +02:00
case 14 :
bonusValue . first = val ;
break ;
case 15 :
bonusValue . second = val ;
2012-07-19 21:52:44 +03:00
break ;
2009-09-21 12:00:33 +03:00
}
}
2012-03-04 13:43:46 +03:00
CGTownInstance : : EFortLevel CGTownInstance : : fortLevel ( ) const //0 - none, 1 - fort, 2 - citadel, 3 - castle
2008-02-18 23:14:28 +02:00
{
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : CASTLE ) )
2012-03-04 13:43:46 +03:00
return CASTLE ;
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : CITADEL ) )
2012-03-04 13:43:46 +03:00
return CITADEL ;
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : FORT ) )
2012-03-04 13:43:46 +03:00
return FORT ;
return NONE ;
2008-02-18 23:14:28 +02:00
}
2009-07-09 22:15:22 +03:00
2008-02-18 23:14:28 +02:00
int CGTownInstance : : hallLevel ( ) const // -1 - none, 0 - village, 1 - town, 2 - city, 3 - capitol
{
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : CAPITOL ) )
2008-02-18 23:14:28 +02:00
return 3 ;
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : CITY_HALL ) )
2008-02-18 23:14:28 +02:00
return 2 ;
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : TOWN_HALL ) )
2008-02-18 23:14:28 +02:00
return 1 ;
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : VILLAGE_HALL ) )
2008-02-18 23:14:28 +02:00
return 0 ;
return - 1 ;
}
2008-08-20 09:57:53 +03:00
int CGTownInstance : : mageGuildLevel ( ) const
{
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : MAGES_GUILD_5 ) )
2008-08-20 09:57:53 +03:00
return 5 ;
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : MAGES_GUILD_4 ) )
2008-08-20 09:57:53 +03:00
return 4 ;
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : MAGES_GUILD_3 ) )
2008-08-20 09:57:53 +03:00
return 3 ;
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : MAGES_GUILD_2 ) )
2008-08-20 09:57:53 +03:00
return 2 ;
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : MAGES_GUILD_1 ) )
2008-08-20 09:57:53 +03:00
return 1 ;
return 0 ;
}
2012-08-08 00:46:24 +03:00
int CGTownInstance : : creatureDwellingLevel ( int dwelling ) const
2008-04-04 20:30:53 +03:00
{
2012-08-08 00:46:24 +03:00
if ( dwelling < 0 | | dwelling > = GameConstants : : CREATURES_PER_TOWN )
return - 1 ;
for ( int i = 0 ; ; i + + )
{
2012-09-05 15:49:23 +03:00
if ( ! hasBuilt ( EBuilding : : DWELL_FIRST + dwelling + i * GameConstants : : CREATURES_PER_TOWN ) )
2012-08-08 00:46:24 +03:00
return i - 1 ;
}
2008-04-04 20:30:53 +03:00
}
2008-06-03 16:15:34 +03:00
int CGTownInstance : : getHordeLevel ( const int & HID ) const //HID - 0 or 1; returns creature level or -1 if that horde structure is not present
2008-04-04 20:30:53 +03:00
{
2008-04-14 20:45:01 +03:00
return town - > hordeLvl [ HID ] ;
2008-04-04 20:30:53 +03:00
}
2008-06-03 16:15:34 +03:00
int CGTownInstance : : creatureGrowth ( const int & level ) const
2008-04-11 20:41:02 +03:00
{
2011-08-26 23:32:05 +03:00
return getGrowthInfo ( level ) . totalGrowth ( ) ;
}
GrowthInfo CGTownInstance : : getGrowthInfo ( int level ) const
{
GrowthInfo ret ;
2011-12-14 00:23:17 +03:00
if ( level < 0 | | level > = GameConstants : : CREATURES_PER_TOWN )
2011-08-26 23:32:05 +03:00
return ret ;
2012-09-05 15:49:23 +03:00
if ( ! hasBuilt ( EBuilding : : DWELL_FIRST + level ) )
2011-08-26 23:32:05 +03:00
return ret ; //no dwelling
2012-01-13 17:18:32 +03:00
const CCreature * creature = VLC - > creh - > creatures [ creatures [ level ] . second . back ( ) ] ;
2011-08-26 23:32:05 +03:00
const int base = creature - > growth ;
int castleBonus = 0 ;
ret . entries . push_back ( GrowthInfo : : Entry ( VLC - > generaltexth - > allTexts [ 590 ] , base ) ) ; // \n\nBasic growth %d"
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : CASTLE ) )
2011-12-14 00:23:17 +03:00
ret . entries . push_back ( GrowthInfo : : Entry ( subID , EBuilding : : CASTLE , castleBonus = base ) ) ;
2012-09-05 15:49:23 +03:00
else if ( hasBuilt ( EBuilding : : CITADEL ) )
2011-12-14 00:23:17 +03:00
ret . entries . push_back ( GrowthInfo : : Entry ( subID , EBuilding : : CITADEL , castleBonus = base / 2 ) ) ;
2011-08-26 23:32:05 +03:00
if ( town - > hordeLvl [ 0 ] = = level ) //horde 1
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : HORDE_1 ) )
2011-12-14 00:23:17 +03:00
ret . entries . push_back ( GrowthInfo : : Entry ( subID , EBuilding : : HORDE_1 , creature - > hordeGrowth ) ) ;
2011-08-26 23:32:05 +03:00
if ( town - > hordeLvl [ 1 ] = = level ) //horde 2
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : HORDE_2 ) )
2011-12-14 00:23:17 +03:00
ret . entries . push_back ( GrowthInfo : : Entry ( subID , EBuilding : : HORDE_2 , creature - > hordeGrowth ) ) ;
2011-08-26 23:32:05 +03:00
int dwellingBonus = 0 ;
if ( const PlayerState * p = cb - > getPlayer ( tempOwner , false ) )
2010-06-21 07:43:10 +03:00
{
2011-08-26 23:32:05 +03:00
BOOST_FOREACH ( const CGDwelling * dwelling , p - > dwellings )
2012-01-13 17:18:32 +03:00
if ( vstd : : contains ( creatures [ level ] . second , dwelling - > creatures [ 0 ] . second [ 0 ] ) )
2011-08-26 23:32:05 +03:00
dwellingBonus + + ;
2010-06-21 07:43:10 +03:00
}
2009-08-16 16:44:17 +03:00
2011-08-26 23:32:05 +03:00
if ( dwellingBonus )
ret . entries . push_back ( GrowthInfo : : Entry ( VLC - > generaltexth - > allTexts [ 591 ] , dwellingBonus ) ) ; // \nExternal dwellings %+d
2012-07-19 21:52:44 +03:00
2011-08-26 23:32:05 +03:00
//other *-of-legion-like bonuses (%d to growth cumulative with grail)
2011-09-06 16:59:26 +03:00
TBonusListPtr bonuses = getBonuses ( Selector : : type ( Bonus : : CREATURE_GROWTH ) & & Selector : : subtype ( level ) ) ;
2011-08-26 23:32:05 +03:00
BOOST_FOREACH ( const Bonus * b , * bonuses )
ret . entries . push_back ( GrowthInfo : : Entry ( b - > Description ( ) + " %+d " , b - > val ) ) ;
//statue-of-legion-like bonus: % to base+castle
2011-09-06 16:59:26 +03:00
TBonusListPtr bonuses2 = getBonuses ( Selector : : type ( Bonus : : CREATURE_GROWTH_PERCENT ) ) ;
2011-08-26 23:32:05 +03:00
BOOST_FOREACH ( const Bonus * b , * bonuses2 )
ret . entries . push_back ( GrowthInfo : : Entry ( b - > Description ( ) + " %+d " , b - > val * ( base + castleBonus ) / 100 ) ) ;
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : GRAIL ) ) //grail - +50% to ALL (so far added) growth
2011-12-14 00:23:17 +03:00
ret . entries . push_back ( GrowthInfo : : Entry ( subID , EBuilding : : GRAIL , ret . totalGrowth ( ) / 2 ) ) ;
2011-08-26 23:32:05 +03:00
return ret ;
2008-04-11 20:41:02 +03:00
}
2011-08-26 23:32:05 +03:00
2008-02-18 23:14:28 +02:00
int CGTownInstance : : dailyIncome ( ) const
{
int ret = 0 ;
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : GRAIL ) )
2008-02-18 23:14:28 +02:00
ret + = 5000 ;
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : CAPITOL ) )
2008-02-18 23:14:28 +02:00
ret + = 4000 ;
2012-09-05 15:49:23 +03:00
else if ( hasBuilt ( EBuilding : : CITY_HALL ) )
2008-02-18 23:14:28 +02:00
ret + = 2000 ;
2012-09-05 15:49:23 +03:00
else if ( hasBuilt ( EBuilding : : TOWN_HALL ) )
2008-02-18 23:14:28 +02:00
ret + = 1000 ;
2012-09-05 15:49:23 +03:00
else if ( hasBuilt ( EBuilding : : VILLAGE_HALL ) )
2008-02-18 23:14:28 +02:00
ret + = 500 ;
return ret ;
}
2008-01-09 19:21:31 +02:00
bool CGTownInstance : : hasFort ( ) const
{
2012-09-05 15:49:23 +03:00
return hasBuilt ( EBuilding : : FORT ) ;
2008-01-09 19:21:31 +02:00
}
2008-02-05 05:56:45 +02:00
bool CGTownInstance : : hasCapitol ( ) const
{
2012-09-05 15:49:23 +03:00
return hasBuilt ( EBuilding : : CAPITOL ) ;
2008-02-05 05:56:45 +02:00
}
2007-10-27 22:38:48 +03:00
CGTownInstance : : CGTownInstance ( )
2013-01-06 22:30:12 +03:00
: IShipyard ( this ) , IMarket ( this ) , town ( nullptr ) , builded ( 0 ) , destroyed ( 0 ) , identifier ( 0 ) , alignment ( 0xff )
2007-10-27 22:38:48 +03:00
{
2013-01-06 22:30:12 +03:00
2007-10-27 22:38:48 +03:00
}
2008-12-22 19:48:41 +02:00
CGTownInstance : : ~ CGTownInstance ( )
2009-09-23 15:42:14 +03:00
{
2012-07-19 21:52:44 +03:00
for ( std : : vector < CGTownBuilding * > : : const_iterator i = bonusingBuildings . begin ( ) ; i ! = bonusingBuildings . end ( ) ; i + + )
2009-09-23 15:42:14 +03:00
delete * i ;
}
2008-12-22 19:48:41 +02:00
int CGTownInstance : : spellsAtLevel ( int level , bool checkGuild ) const
2007-10-27 22:38:48 +03:00
{
2008-12-22 19:48:41 +02:00
if ( checkGuild & & mageGuildLevel ( ) < level )
return 0 ;
int ret = 6 - level ; //how many spells are available at this level
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( EBuilding : : LIBRARY , ETownType : : TOWER ) )
2012-07-19 21:52:44 +03:00
ret + + ;
2012-09-05 15:49:23 +03:00
2008-12-22 19:48:41 +02:00
return ret ;
2007-10-27 22:38:48 +03:00
}
2008-12-22 19:48:41 +02:00
bool CGTownInstance : : needsLastStack ( ) const
2007-10-27 22:38:48 +03:00
{
2008-12-22 19:48:41 +02:00
if ( garrisonHero )
return true ;
else return false ;
2007-10-27 22:38:48 +03:00
}
2008-12-22 19:48:41 +02:00
2008-12-27 03:01:59 +02:00
void CGTownInstance : : onHeroVisit ( const CGHeroInstance * h ) const
2008-02-07 20:45:22 +02:00
{
2010-08-13 13:46:08 +03:00
if ( ! cb - > gameState ( ) - > getPlayerRelations ( getOwner ( ) , h - > getOwner ( ) ) ) //if this is enemy
2008-12-22 19:48:41 +02:00
{
2011-02-21 06:13:00 +02:00
if ( armedGarrison ( ) | | visitingHero )
2009-08-22 16:59:15 +03:00
{
const CGHeroInstance * defendingHero = NULL ;
2011-02-21 06:13:00 +02:00
const CArmedInstance * defendingArmy = this ;
2009-08-22 16:59:15 +03:00
if ( visitingHero )
defendingHero = visitingHero ;
else if ( garrisonHero )
defendingHero = garrisonHero ;
if ( defendingHero )
defendingArmy = defendingHero ;
bool outsideTown = ( defendingHero = = visitingHero & & garrisonHero ) ;
2011-02-21 06:13:00 +02:00
2012-07-19 21:52:44 +03:00
//TODO
//"borrowing" army from garrison to visiting hero
2011-02-21 06:13:00 +02:00
2009-08-22 16:59:15 +03:00
cb - > startBattleI ( h , defendingArmy , getSightCenter ( ) , h , defendingHero , false , boost : : bind ( & CGTownInstance : : fightOver , this , h , _1 ) , ( outsideTown ? NULL : this ) ) ;
}
else
{
2009-09-22 19:12:33 +03:00
cb - > setOwner ( id , h - > tempOwner ) ;
2011-02-21 06:13:00 +02:00
removeCapitols ( h - > getOwner ( ) ) ;
2009-10-06 09:15:56 +03:00
cb - > heroVisitCastle ( id , h - > id ) ;
2009-08-22 16:59:15 +03:00
}
2008-12-22 19:48:41 +02:00
}
2009-10-06 09:15:56 +03:00
else
2012-07-17 11:52:27 +03:00
{
if ( h - > commander & & ! h - > commander - > alive ) //rise commander. TODO: interactive script
{
SetCommanderProperty scp ;
scp . heroid = h - > id ;
scp . which = SetCommanderProperty : : ALIVE ;
scp . amount = 1 ;
cb - > sendAndApply ( & scp ) ;
}
2009-10-06 09:15:56 +03:00
cb - > heroVisitCastle ( id , h - > id ) ;
2012-07-17 11:52:27 +03:00
}
2008-10-26 22:58:34 +02:00
}
2008-12-27 03:01:59 +02:00
void CGTownInstance : : onHeroLeave ( const CGHeroInstance * h ) const
2008-10-26 22:58:34 +02:00
{
2008-12-22 19:48:41 +02:00
cb - > stopHeroVisitCastle ( id , h - > id ) ;
2008-10-26 22:58:34 +02:00
}
2008-12-22 19:48:41 +02:00
void CGTownInstance : : initObj ( )
2010-06-07 08:28:12 +03:00
///initialize town structures
{
2010-02-01 19:07:46 +02:00
blockVisit = true ;
2012-10-05 16:11:26 +03:00
hoverName = name + " , " + VLC - > townh - > factions [ town - > typeID ] . name ;
2009-07-06 22:41:27 +03:00
2012-09-05 15:49:23 +03:00
if ( subID = = ETownType : : DUNGEON )
2011-12-14 00:23:17 +03:00
creatures . resize ( GameConstants : : CREATURES_PER_TOWN + 1 ) ; //extra dwelling for Dungeon
2010-07-10 19:50:23 +03:00
else
2011-12-14 00:23:17 +03:00
creatures . resize ( GameConstants : : CREATURES_PER_TOWN ) ;
for ( int i = 0 ; i < GameConstants : : CREATURES_PER_TOWN ; i + + )
2009-07-06 22:41:27 +03:00
{
2012-08-08 00:46:24 +03:00
int dwellingLevel = creatureDwellingLevel ( i ) ;
int creaturesTotal = town - > creatures [ i ] . size ( ) ;
for ( int j = 0 ; j < std : : min ( dwellingLevel + 1 , creaturesTotal ) ; j + + )
creatures [ i ] . second . push_back ( town - > creatures [ i ] [ j ] ) ;
2009-07-06 22:41:27 +03:00
}
2010-07-10 19:50:23 +03:00
2009-09-21 12:00:33 +03:00
switch ( subID )
2009-08-23 18:02:21 +03:00
{ //add new visitable objects
2009-10-05 19:38:54 +03:00
case 0 :
bonusingBuildings . push_back ( new COPWBonus ( 21 , this ) ) ; //Stables
break ;
2010-07-10 19:50:23 +03:00
case 5 :
bonusingBuildings . push_back ( new COPWBonus ( 21 , this ) ) ; //Vortex
case 2 : case 3 : case 6 :
2009-10-05 19:38:54 +03:00
bonusingBuildings . push_back ( new CTownBonus ( 23 , this ) ) ;
2009-08-23 18:02:21 +03:00
break ;
case 7 :
2009-10-05 19:38:54 +03:00
bonusingBuildings . push_back ( new CTownBonus ( 17 , this ) ) ;
2009-08-23 18:02:21 +03:00
break ;
2010-06-13 16:59:59 +03:00
}
//add special bonuses from buildings
2011-02-04 16:58:14 +02:00
recreateBuildingsBonuses ( ) ;
2008-12-22 19:48:41 +02:00
}
2009-10-06 10:55:39 +03:00
void CGTownInstance : : newTurn ( ) const
{
2010-01-28 18:15:46 +02:00
if ( cb - > getDate ( 1 ) = = 1 ) //reset on new week
2009-10-06 10:55:39 +03:00
{
2012-09-05 15:49:23 +03:00
//give resources for Rampart, Mystic Pond
if ( hasBuilt ( EBuilding : : MYSTIC_POND , ETownType : : RAMPART )
& & cb - > getDate ( 0 ) ! = 1 & & ( tempOwner < GameConstants : : PLAYER_LIMIT ) )
2010-01-28 18:15:46 +02:00
{
int resID = rand ( ) % 4 + 2 ; //bonus to random rare resource
resID = ( resID = = 2 ) ? 1 : resID ;
int resVal = rand ( ) % 4 + 1 ; //with size 1..4
cb - > giveResource ( tempOwner , resID , resVal ) ;
cb - > setObjProperty ( id , 14 , resID ) ;
cb - > setObjProperty ( id , 15 , resVal ) ;
}
2012-09-05 15:49:23 +03:00
if ( subID = = ETownType : : DUNGEON )
2010-01-28 18:15:46 +02:00
for ( std : : vector < CGTownBuilding * > : : const_iterator i = bonusingBuildings . begin ( ) ; i ! = bonusingBuildings . end ( ) ; i + + )
2010-01-01 14:15:20 +02:00
{
2012-09-05 15:49:23 +03:00
if ( ( * i ) - > ID = = EBuilding : : MANA_VORTEX )
2010-01-01 14:15:20 +02:00
cb - > setObjProperty ( id , 12 , ( * i ) - > id ) ; //reset visitors for Mana Vortex
}
2010-06-29 12:59:14 +03:00
2011-12-14 00:23:17 +03:00
if ( tempOwner = = GameConstants : : NEUTRAL_PLAYER ) //garrison growth for neutral towns
2010-06-29 12:59:14 +03:00
{
std : : vector < ui8 > nativeCrits ; //slots
2010-11-22 02:34:46 +02:00
for ( TSlots : : const_iterator it = Slots ( ) . begin ( ) ; it ! = Slots ( ) . end ( ) ; it + + )
2010-06-29 12:59:14 +03:00
{
2010-11-22 02:34:46 +02:00
if ( it - > second - > type - > faction = = subID ) //native
2010-06-29 12:59:14 +03:00
{
nativeCrits . push_back ( it - > first ) ; //collect matching slots
}
}
if ( nativeCrits . size ( ) )
{
2010-11-22 02:34:46 +02:00
TSlot pos = nativeCrits [ rand ( ) % nativeCrits . size ( ) ] ;
2010-11-27 03:46:19 +02:00
StackLocation sl ( this , pos ) ;
2010-11-22 02:34:46 +02:00
const CCreature * c = getCreature ( pos ) ;
2011-09-24 04:15:36 +03:00
if ( rand ( ) % 100 < 90 | | c - > upgrades . empty ( ) ) //increase number if no upgrade available
2010-06-29 12:59:14 +03:00
{
2010-11-27 03:46:19 +02:00
cb - > changeStackCount ( sl , c - > growth ) ;
2010-06-29 12:59:14 +03:00
}
else //upgrade
{
2010-11-27 03:46:19 +02:00
cb - > changeStackType ( sl , VLC - > creh - > creatures [ * c - > upgrades . begin ( ) ] ) ;
2010-06-29 12:59:14 +03:00
}
}
2011-12-14 00:23:17 +03:00
if ( ( stacksCount ( ) < GameConstants : : ARMY_SIZE & & rand ( ) % 100 < 25 ) | | Slots ( ) . empty ( ) ) //add new stack
2010-06-29 12:59:14 +03:00
{
2011-12-14 00:23:17 +03:00
int i = rand ( ) % std : : min ( GameConstants : : ARMY_SIZE , cb - > getDate ( 3 ) < < 1 ) ;
2012-08-08 00:46:24 +03:00
TCreature c = town - > creatures [ i ] [ 0 ] ;
2010-11-27 03:46:19 +02:00
TSlot n = - 1 ;
2012-07-19 21:52:44 +03:00
2010-11-27 03:46:19 +02:00
TQuantity count = creatureGrowth ( i ) ;
2011-05-29 15:36:38 +03:00
if ( ! count ) // no dwelling
count = VLC - > creh - > creatures [ c ] - > growth ;
2010-11-27 03:46:19 +02:00
2010-06-29 12:59:14 +03:00
{ //no lower tiers or above current month
2010-11-27 03:46:19 +02:00
if ( ( n = getSlotFor ( c ) ) > = 0 )
2012-07-19 21:52:44 +03:00
{
2010-11-27 03:46:19 +02:00
StackLocation sl ( this , n ) ;
2010-06-29 12:59:14 +03:00
if ( slotEmpty ( n ) )
2010-11-27 03:46:19 +02:00
cb - > insertNewStack ( sl , VLC - > creh - > creatures [ c ] , count ) ;
else //add to existing
cb - > changeStackCount ( sl , count ) ;
2010-06-29 12:59:14 +03:00
}
}
2012-07-19 21:52:44 +03:00
}
2010-06-29 12:59:14 +03:00
}
2009-10-06 10:55:39 +03:00
}
}
2010-01-28 18:15:46 +02:00
2009-03-12 01:25:59 +02:00
int3 CGTownInstance : : getSightCenter ( ) const
{
return pos - int3 ( 2 , 0 , 0 ) ;
}
2010-02-13 06:47:31 +02:00
ui8 CGTownInstance : : getPassableness ( ) const
{
2011-02-21 06:13:00 +02:00
if ( ! armedGarrison ( ) ) //empty castle - anyone can visit
2011-12-14 00:23:17 +03:00
return GameConstants : : ALL_PLAYERS ;
2012-09-29 20:36:48 +03:00
if ( tempOwner = = GameConstants : : NEUTRAL_PLAYER ) //neutral guarded - no one can visit
2010-08-06 16:14:10 +03:00
return 0 ;
2012-07-19 21:52:44 +03:00
2010-08-08 17:39:41 +03:00
ui8 mask = 0 ;
2010-08-06 16:14:10 +03:00
TeamState * ts = cb - > gameState ( ) - > getPlayerTeam ( tempOwner ) ;
BOOST_FOREACH ( ui8 it , ts - > players )
mask | = 1 < < it ; //allies - add to possible visitors
2012-07-19 21:52:44 +03:00
2010-08-06 16:14:10 +03:00
return mask ;
2010-02-13 06:47:31 +02:00
}
2009-07-26 06:33:13 +03:00
void CGTownInstance : : getOutOffsets ( std : : vector < int3 > & offsets ) const
{
2009-09-20 15:47:40 +03:00
offsets + = int3 ( - 1 , 2 , 0 ) , int3 ( - 3 , 2 , 0 ) ;
2009-07-26 06:33:13 +03:00
}
2009-08-22 16:59:15 +03:00
void CGTownInstance : : fightOver ( const CGHeroInstance * h , BattleResult * result ) const
{
if ( result - > winner = = 0 )
{
2010-02-28 08:36:51 +02:00
removeCapitols ( h - > getOwner ( ) ) ;
2009-09-23 15:42:14 +03:00
cb - > setOwner ( id , h - > tempOwner ) ; //give control after checkout is done
2010-06-03 09:18:12 +03:00
FoWChange fw ;
fw . player = h - > tempOwner ;
fw . mode = 1 ;
getSightTiles ( fw . tiles ) ; //update visibility for castle structures
cb - > sendAndApply ( & fw ) ;
2009-08-22 16:59:15 +03:00
}
}
2009-09-23 15:05:33 +03:00
2012-09-24 21:52:30 +03:00
void CGTownInstance : : removeCapitols ( ui8 owner ) const
2012-07-19 21:52:44 +03:00
{
2010-02-28 08:36:51 +02:00
if ( hasCapitol ( ) ) // search if there's an older capitol
2012-07-19 21:52:44 +03:00
{
2009-10-06 09:15:56 +03:00
PlayerState * state = cb - > gameState ( ) - > getPlayer ( owner ) ; //get all towns owned by player
2012-09-24 21:52:30 +03:00
for ( std : : vector < ConstTransitivePtr < CGTownInstance > > : : const_iterator i = state - > towns . begin ( ) ; i < state - > towns . end ( ) ; + + i )
2012-07-19 21:52:44 +03:00
{
if ( * i ! = this & & ( * i ) - > hasCapitol ( ) )
{
RazeStructures rs ;
rs . tid = id ;
rs . bid . insert ( 13 ) ;
rs . destroyed = destroyed ;
cb - > sendAndApply ( & rs ) ;
return ;
}
}
}
}
2009-09-23 15:42:14 +03:00
2010-02-13 06:47:31 +02:00
int CGTownInstance : : getBoatType ( ) const
2010-02-01 19:07:46 +02:00
{
2012-08-08 00:46:24 +03:00
const CCreature * c = VLC - > creh - > creatures [ town - > creatures . front ( ) . front ( ) ] ;
2010-02-13 06:47:31 +02:00
if ( c - > isGood ( ) )
return 1 ;
else if ( c - > isEvil ( ) )
return 0 ;
else //neutral
return 2 ;
2010-02-01 19:07:46 +02:00
}
2010-05-18 10:01:54 +03:00
int CGTownInstance : : getMarketEfficiency ( ) const
{
2012-09-05 15:49:23 +03:00
if ( ! hasBuilt ( EBuilding : : MARKETPLACE ) )
2010-05-18 10:01:54 +03:00
return 0 ;
2011-05-10 01:20:47 +03:00
const PlayerState * p = cb - > getPlayer ( tempOwner ) ;
2010-05-18 10:01:54 +03:00
assert ( p ) ;
int marketCount = 0 ;
BOOST_FOREACH ( const CGTownInstance * t , p - > towns )
2012-09-05 15:49:23 +03:00
if ( t - > hasBuilt ( EBuilding : : MARKETPLACE ) )
2010-05-18 10:01:54 +03:00
marketCount + + ;
return marketCount ;
}
2011-12-14 00:23:17 +03:00
bool CGTownInstance : : allowsTrade ( EMarketMode : : EMarketMode mode ) const
2010-05-18 10:01:54 +03:00
{
switch ( mode )
{
2011-12-14 00:23:17 +03:00
case EMarketMode : : RESOURCE_RESOURCE :
case EMarketMode : : RESOURCE_PLAYER :
2012-09-05 15:49:23 +03:00
return hasBuilt ( EBuilding : : MARKETPLACE ) ;
2011-12-14 00:23:17 +03:00
case EMarketMode : : ARTIFACT_RESOURCE :
case EMarketMode : : RESOURCE_ARTIFACT :
2012-09-05 15:49:23 +03:00
return hasBuilt ( EBuilding : : ARTIFACT_MERCHANT , ETownType : : TOWER )
| | hasBuilt ( EBuilding : : ARTIFACT_MERCHANT , ETownType : : DUNGEON )
| | hasBuilt ( EBuilding : : ARTIFACT_MERCHANT , ETownType : : CONFLUX ) ;
2011-12-14 00:23:17 +03:00
case EMarketMode : : CREATURE_RESOURCE :
2012-09-05 15:49:23 +03:00
return hasBuilt ( EBuilding : : FREELANCERS_GUILD , ETownType : : STRONGHOLD ) ;
2011-12-14 00:23:17 +03:00
case EMarketMode : : CREATURE_UNDEAD :
2012-09-05 15:49:23 +03:00
return hasBuilt ( EBuilding : : SKELETON_TRANSFORMER , ETownType : : NECROPOLIS ) ;
2011-12-14 00:23:17 +03:00
case EMarketMode : : RESOURCE_SKILL :
2012-09-05 15:49:23 +03:00
return hasBuilt ( EBuilding : : MAGIC_UNIVERSITY , ETownType : : CONFLUX ) ;
2010-05-18 10:01:54 +03:00
default :
assert ( 0 ) ;
return false ;
}
}
2011-12-14 00:23:17 +03:00
std : : vector < int > CGTownInstance : : availableItemsIds ( EMarketMode : : EMarketMode mode ) const
2010-06-27 19:03:01 +03:00
{
2011-12-14 00:23:17 +03:00
if ( mode = = EMarketMode : : RESOURCE_ARTIFACT )
2010-06-27 19:03:01 +03:00
{
std : : vector < int > ret ;
BOOST_FOREACH ( const CArtifact * a , merchantArtifacts )
if ( a )
ret . push_back ( a - > id ) ;
else
ret . push_back ( - 1 ) ;
return ret ;
}
2011-12-14 00:23:17 +03:00
else if ( mode = = EMarketMode : : RESOURCE_SKILL )
2010-07-20 17:08:13 +03:00
{
return universitySkills ;
}
2010-06-27 19:03:01 +03:00
else
return IMarket : : availableItemsIds ( mode ) ;
}
2011-02-04 16:58:14 +02:00
std : : string CGTownInstance : : nodeName ( ) const
{
2012-10-05 16:11:26 +03:00
return " Town ( " + ( town ? VLC - > townh - > factions [ town - > typeID ] . name : " unknown " ) + " ) of " + name ;
2011-02-04 16:58:14 +02:00
}
void CGTownInstance : : deserializationFix ( )
{
attachTo ( & townAndVis ) ;
if ( visitingHero )
visitingHero - > attachTo ( & townAndVis ) ;
if ( garrisonHero )
garrisonHero - > attachTo ( this ) ;
}
void CGTownInstance : : recreateBuildingsBonuses ( )
{
2012-03-06 19:59:55 +03:00
BonusList bl ;
2011-07-13 21:39:02 +03:00
getExportedBonusList ( ) . getBonuses ( bl , Selector : : sourceType ( Bonus : : TOWN_STRUCTURE ) ) ;
2012-03-06 19:59:55 +03:00
BOOST_FOREACH ( Bonus * b , bl )
2011-02-21 06:13:00 +02:00
removeBonus ( b ) ;
2011-02-04 16:58:14 +02:00
2012-09-05 15:49:23 +03:00
//tricky! -> checks tavern only if no bratherhood of sword or not a castle
if ( subID ! = ETownType : : CASTLE | | ! addBonusIfBuilt ( EBuilding : : BROTHERHOOD , Bonus : : MORALE , + 2 ) )
addBonusIfBuilt ( EBuilding : : TAVERN , Bonus : : MORALE , + 1 ) ;
2011-02-04 16:58:14 +02:00
2012-09-05 15:49:23 +03:00
if ( subID = = ETownType : : CASTLE ) //castle
2011-02-21 06:13:00 +02:00
{
2012-09-05 15:49:23 +03:00
addBonusIfBuilt ( EBuilding : : LIGHTHOUSE , Bonus : : SEA_MOVEMENT , + 500 , make_shared < CPropagatorNodeType > ( PLAYER ) ) ;
addBonusIfBuilt ( EBuilding : : GRAIL , Bonus : : MORALE , + 2 , make_shared < CPropagatorNodeType > ( PLAYER ) ) ; //colossus
2011-02-21 06:13:00 +02:00
}
2012-09-05 15:49:23 +03:00
else if ( subID = = ETownType : : RAMPART ) //rampart
2011-02-21 06:13:00 +02:00
{
2012-09-05 15:49:23 +03:00
addBonusIfBuilt ( EBuilding : : FOUNTAIN_OF_FORTUNE , Bonus : : LUCK , + 2 ) ; //fountain of fortune
addBonusIfBuilt ( EBuilding : : GRAIL , Bonus : : LUCK , + 2 , make_shared < CPropagatorNodeType > ( PLAYER ) ) ; //guardian spirit
2011-02-21 06:13:00 +02:00
}
2012-09-26 16:13:39 +03:00
else if ( subID = = ETownType : : TOWER ) //tower
2011-02-21 06:13:00 +02:00
{
2012-09-05 15:49:23 +03:00
addBonusIfBuilt ( EBuilding : : GRAIL , Bonus : : PRIMARY_SKILL , + 15 , PrimarySkill : : KNOWLEDGE ) ; //grail
2011-02-21 06:13:00 +02:00
}
2012-09-26 16:13:39 +03:00
else if ( subID = = ETownType : : INFERNO ) //Inferno
2011-02-21 06:13:00 +02:00
{
2012-09-05 15:49:23 +03:00
addBonusIfBuilt ( EBuilding : : STORMCLOUDS , Bonus : : PRIMARY_SKILL , + 2 , PrimarySkill : : SPELL_POWER ) ; //Brimstone Clouds
2011-02-21 06:13:00 +02:00
}
2012-09-05 15:49:23 +03:00
else if ( subID = = ETownType : : NECROPOLIS ) //necropolis
2011-02-21 06:13:00 +02:00
{
2012-09-05 15:49:23 +03:00
addBonusIfBuilt ( EBuilding : : COVER_OF_DARKNESS , Bonus : : DARKNESS , + 20 ) ;
addBonusIfBuilt ( EBuilding : : NECROMANCY_AMPLIFIER , Bonus : : SECONDARY_SKILL_PREMY , + 10 , make_shared < CPropagatorNodeType > ( PLAYER ) , CGHeroInstance : : NECROMANCY ) ; //necromancy amplifier
addBonusIfBuilt ( EBuilding : : GRAIL , Bonus : : SECONDARY_SKILL_PREMY , + 20 , make_shared < CPropagatorNodeType > ( PLAYER ) , CGHeroInstance : : NECROMANCY ) ; //Soul prison
2011-02-21 06:13:00 +02:00
}
2012-09-05 15:49:23 +03:00
else if ( subID = = ETownType : : DUNGEON ) //Dungeon
2011-02-21 06:13:00 +02:00
{
2012-09-05 15:49:23 +03:00
addBonusIfBuilt ( EBuilding : : GRAIL , Bonus : : PRIMARY_SKILL , + 12 , PrimarySkill : : SPELL_POWER ) ; //grail
2011-02-21 06:13:00 +02:00
}
2012-09-05 15:49:23 +03:00
else if ( subID = = ETownType : : STRONGHOLD ) //Stronghold
2011-02-21 06:13:00 +02:00
{
2012-09-05 15:49:23 +03:00
addBonusIfBuilt ( EBuilding : : GRAIL , Bonus : : PRIMARY_SKILL , + 20 , PrimarySkill : : ATTACK ) ; //grail
2011-02-21 06:13:00 +02:00
}
2012-09-05 15:49:23 +03:00
else if ( subID = = ETownType : : FORTRESS ) //Fortress
2011-02-21 06:13:00 +02:00
{
2012-09-05 15:49:23 +03:00
addBonusIfBuilt ( EBuilding : : GLYPHS_OF_FEAR , Bonus : : PRIMARY_SKILL , + 2 , PrimarySkill : : DEFENSE ) ; //Glyphs of Fear
addBonusIfBuilt ( EBuilding : : BLOOD_OBELISK , Bonus : : PRIMARY_SKILL , + 2 , PrimarySkill : : ATTACK ) ; //Blood Obelisk
addBonusIfBuilt ( EBuilding : : GRAIL , Bonus : : PRIMARY_SKILL , + 10 , PrimarySkill : : ATTACK ) ; //grail
addBonusIfBuilt ( EBuilding : : GRAIL , Bonus : : PRIMARY_SKILL , + 10 , PrimarySkill : : DEFENSE ) ; //grail
2011-02-21 06:13:00 +02:00
}
2012-09-05 15:49:23 +03:00
else if ( subID = = ETownType : : CONFLUX )
2011-02-21 06:13:00 +02:00
{
2011-02-04 16:58:14 +02:00
2011-02-21 06:13:00 +02:00
}
}
bool CGTownInstance : : addBonusIfBuilt ( int building , int type , int val , int subtype /*= -1*/ )
{
2012-06-23 20:19:50 +03:00
return addBonusIfBuilt ( building , type , val , TPropagatorPtr ( ) , subtype ) ;
2011-02-21 06:13:00 +02:00
}
2012-03-06 19:59:55 +03:00
bool CGTownInstance : : addBonusIfBuilt ( int building , int type , int val , TPropagatorPtr prop , int subtype /*= -1*/ )
2011-02-21 06:13:00 +02:00
{
2012-09-05 15:49:23 +03:00
if ( hasBuilt ( building ) )
2011-02-21 06:13:00 +02:00
{
std : : ostringstream descr ;
2012-10-05 16:11:26 +03:00
descr < < town - > buildings [ building ] - > Name ( ) < < " " ;
2011-02-21 06:13:00 +02:00
if ( val > 0 )
descr < < " + " ;
else if ( val < 0 )
descr < < " - " ;
descr < < val ;
Bonus * b = new Bonus ( Bonus : : PERMANENT , type , Bonus : : TOWN_STRUCTURE , val , building , descr . str ( ) , subtype ) ;
if ( prop )
b - > addPropagator ( prop ) ;
addNewBonus ( b ) ;
return true ;
}
return false ;
2011-02-04 16:58:14 +02:00
}
void CGTownInstance : : setVisitingHero ( CGHeroInstance * h )
{
assert ( ! ! visitingHero = = ! h ) ;
if ( h )
{
PlayerState * p = cb - > gameState ( ) - > getPlayer ( h - > tempOwner ) ;
assert ( p ) ;
h - > detachFrom ( p ) ;
h - > attachTo ( & townAndVis ) ;
visitingHero = h ;
h - > visitedTown = this ;
h - > inTownGarrison = false ;
}
else
{
PlayerState * p = cb - > gameState ( ) - > getPlayer ( visitingHero - > tempOwner ) ;
visitingHero - > visitedTown = NULL ;
visitingHero - > detachFrom ( & townAndVis ) ;
visitingHero - > attachTo ( p ) ;
visitingHero = NULL ;
}
}
void CGTownInstance : : setGarrisonedHero ( CGHeroInstance * h )
{
assert ( ! ! garrisonHero = = ! h ) ;
if ( h )
{
PlayerState * p = cb - > gameState ( ) - > getPlayer ( h - > tempOwner ) ;
assert ( p ) ;
h - > detachFrom ( p ) ;
h - > attachTo ( this ) ;
garrisonHero = h ;
h - > visitedTown = this ;
h - > inTownGarrison = true ;
}
else
{
PlayerState * p = cb - > gameState ( ) - > getPlayer ( garrisonHero - > tempOwner ) ;
garrisonHero - > visitedTown = NULL ;
garrisonHero - > inTownGarrison = false ;
garrisonHero - > detachFrom ( this ) ;
garrisonHero - > attachTo ( p ) ;
garrisonHero = NULL ;
}
}
2011-02-21 06:13:00 +02:00
bool CGTownInstance : : armedGarrison ( ) const
{
return stacksCount ( ) | | garrisonHero ;
}
2011-02-22 11:47:25 +02:00
CBonusSystemNode * CGTownInstance : : whatShouldBeAttached ( )
{
return & townAndVis ;
}
2011-02-25 19:57:30 +02:00
const CArmedInstance * CGTownInstance : : getUpperArmy ( ) const
{
if ( garrisonHero )
return garrisonHero ;
return this ;
}
2012-09-05 15:49:23 +03:00
bool CGTownInstance : : hasBuilt ( int buildingID , int townID ) const
{
if ( townID = = town - > typeID | | townID = = ETownType : : ANY )
return hasBuilt ( buildingID ) ;
return false ;
}
2012-01-03 04:55:26 +03:00
bool CGTownInstance : : hasBuilt ( int buildingID ) const
{
return vstd : : contains ( builtBuildings , buildingID ) ;
}
2012-09-05 15:49:23 +03:00
2012-03-14 16:02:38 +03:00
bool CGVisitableOPH : : wasVisited ( const CGHeroInstance * h ) const
{
return vstd : : contains ( visitors , h - > id ) ;
}
2008-12-27 03:01:59 +02:00
void CGVisitableOPH : : onHeroVisit ( const CGHeroInstance * h ) const
2008-12-22 19:48:41 +02:00
{
if ( visitors . find ( h - > id ) = = visitors . end ( ) )
2008-10-26 22:58:34 +02:00
{
2008-12-22 19:48:41 +02:00
onNAHeroVisit ( h - > id , false ) ;
2009-04-11 04:32:50 +03:00
switch ( ID )
{
2013-01-03 15:19:20 +03:00
case Obj : : TREE_OF_KNOWLEDGE :
case Obj : : ARENA :
case Obj : : LIBRARY_OF_ENLIGHTENMENT :
case Obj : : SCHOOL_OF_MAGIC :
case Obj : : SCHOOL_OF_WAR :
2009-04-11 04:32:50 +03:00
break ;
default :
2010-05-02 21:20:26 +03:00
cb - > setObjProperty ( id , ObjProperty : : VISITORS , h - > id ) ; //add to the visitors
2009-04-11 04:32:50 +03:00
break ;
}
2008-10-26 22:58:34 +02:00
}
2008-12-22 19:48:41 +02:00
else
2008-10-26 22:58:34 +02:00
{
2008-12-22 19:48:41 +02:00
onNAHeroVisit ( h - > id , true ) ;
2008-10-26 22:58:34 +02:00
}
2008-12-22 19:48:41 +02:00
}
2008-10-26 22:58:34 +02:00
2008-12-22 19:48:41 +02:00
void CGVisitableOPH : : initObj ( )
{
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : TREE_OF_KNOWLEDGE )
2008-12-22 19:48:41 +02:00
ttype = ran ( ) % 3 ;
else
ttype = - 1 ;
}
2008-10-26 22:58:34 +02:00
2012-09-23 21:01:04 +03:00
void CGVisitableOPH : : treeSelected ( int heroID , int resType , int resVal , TExpType expVal , ui32 result ) const
2008-12-22 19:48:41 +02:00
{
2009-04-11 04:32:50 +03:00
if ( result ) //player agreed to give res for exp
2008-10-26 22:58:34 +02:00
{
2008-12-22 19:48:41 +02:00
cb - > giveResource ( cb - > getOwner ( heroID ) , resType , - resVal ) ; //take resource
cb - > changePrimSkill ( heroID , 4 , expVal ) ; //give exp
2010-05-02 21:20:26 +03:00
cb - > setObjProperty ( id , ObjProperty : : VISITORS , heroID ) ; //add to the visitors
2008-10-26 22:58:34 +02:00
}
2008-12-22 19:48:41 +02:00
}
2008-12-27 03:01:59 +02:00
void CGVisitableOPH : : onNAHeroVisit ( int heroID , bool alreadyVisited ) const
2008-12-22 19:48:41 +02:00
{
2012-08-31 20:39:11 +03:00
Component : : EComponentType id = ( Component : : EComponentType ) 0 ;
int subid = 0 , ot = 0 , sound = 0 ;
2012-09-23 21:01:04 +03:00
TExpType val = 1 ;
2008-12-22 19:48:41 +02:00
switch ( ID )
2008-11-14 20:18:13 +02:00
{
2012-09-23 21:01:04 +03:00
case Obj : : ARENA :
2009-04-24 00:09:10 +03:00
sound = soundBase : : NOMAD ;
2009-01-11 00:08:18 +02:00
ot = 0 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : MERCENARY_CAMP :
2009-04-24 00:09:10 +03:00
sound = soundBase : : NOMAD ;
2008-12-22 19:48:41 +02:00
subid = 0 ;
ot = 80 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : MARLETTO_TOWER :
2009-04-24 00:09:10 +03:00
sound = soundBase : : NOMAD ;
2008-12-22 19:48:41 +02:00
subid = 1 ;
ot = 39 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : STAR_AXIS :
2009-04-24 00:09:10 +03:00
sound = soundBase : : gazebo ;
2008-12-22 19:48:41 +02:00
subid = 2 ;
ot = 100 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : GARDEN_OF_REVELATION :
2009-04-24 00:09:10 +03:00
sound = soundBase : : GETPROTECTION ;
2008-12-22 19:48:41 +02:00
subid = 3 ;
ot = 59 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : LEARNING_STONE :
2009-04-24 00:09:10 +03:00
sound = soundBase : : gazebo ;
2012-08-31 20:39:11 +03:00
id = Component : : EXPERIENCE ;
2008-12-22 19:48:41 +02:00
ot = 143 ;
val = 1000 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : TREE_OF_KNOWLEDGE :
2009-04-24 00:09:10 +03:00
sound = soundBase : : gazebo ;
2012-08-31 20:39:11 +03:00
id = Component : : EXPERIENCE ;
2008-12-22 19:48:41 +02:00
subid = 1 ;
ot = 146 ;
val = 1 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : LIBRARY_OF_ENLIGHTENMENT :
2009-04-24 00:09:10 +03:00
sound = soundBase : : gazebo ;
2009-02-01 16:11:41 +02:00
ot = 66 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : SCHOOL_OF_MAGIC :
2009-04-24 00:09:10 +03:00
sound = soundBase : : faerie ;
2009-04-11 04:32:50 +03:00
ot = 71 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : SCHOOL_OF_WAR :
2009-04-24 00:09:10 +03:00
sound = soundBase : : MILITARY ;
2009-04-11 04:32:50 +03:00
ot = 158 ;
break ;
2008-11-14 20:18:13 +02:00
}
2008-12-22 19:48:41 +02:00
if ( ! alreadyVisited )
2008-10-26 22:58:34 +02:00
{
2008-12-22 19:48:41 +02:00
switch ( ID )
2008-10-26 22:58:34 +02:00
{
2012-09-23 21:01:04 +03:00
case Obj : : ARENA :
2009-01-11 00:08:18 +02:00
{
2009-04-11 04:32:50 +03:00
BlockingDialog sd ( false , true ) ;
2009-04-24 00:09:10 +03:00
sd . soundID = sound ;
2009-01-11 00:08:18 +02:00
sd . text < < std : : pair < ui8 , ui32 > ( 11 , ot ) ;
2012-08-31 20:39:11 +03:00
sd . components . push_back ( Component ( Component : : PRIM_SKILL , PrimarySkill : : ATTACK , 2 , 0 ) ) ;
sd . components . push_back ( Component ( Component : : PRIM_SKILL , PrimarySkill : : DEFENSE , 2 , 0 ) ) ;
2009-01-11 00:08:18 +02:00
sd . player = cb - > getOwner ( heroID ) ;
2009-04-11 04:32:50 +03:00
cb - > showBlockingDialog ( & sd , boost : : bind ( & CGVisitableOPH : : arenaSelected , this , heroID , _1 ) ) ;
2009-01-11 00:08:18 +02:00
return ;
}
2012-09-23 21:01:04 +03:00
case Obj : : MERCENARY_CAMP :
case Obj : : MARLETTO_TOWER :
case Obj : : STAR_AXIS :
case Obj : : GARDEN_OF_REVELATION :
2008-10-26 22:58:34 +02:00
{
2008-12-22 19:48:41 +02:00
cb - > changePrimSkill ( heroID , subid , val ) ;
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = sound ;
2012-08-31 20:39:11 +03:00
iw . components . push_back ( Component ( Component : : PRIM_SKILL , subid , val , 0 ) ) ;
2008-12-22 19:48:41 +02:00
iw . text < < std : : pair < ui8 , ui32 > ( 11 , ot ) ;
iw . player = cb - > getOwner ( heroID ) ;
cb - > showInfoDialog ( & iw ) ;
break ;
}
2012-09-23 21:01:04 +03:00
case Obj : : LEARNING_STONE : //give exp
2008-12-22 19:48:41 +02:00
{
2010-07-26 01:47:59 +03:00
const CGHeroInstance * h = cb - > getHero ( heroID ) ;
2011-02-12 20:48:11 +02:00
val = h - > calculateXp ( val ) ;
2008-12-22 19:48:41 +02:00
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = sound ;
2008-12-22 19:48:41 +02:00
iw . components . push_back ( Component ( id , subid , val , 0 ) ) ;
iw . player = cb - > getOwner ( heroID ) ;
iw . text < < std : : pair < ui8 , ui32 > ( 11 , ot ) ;
2009-04-22 21:48:56 +03:00
iw . soundID = soundBase : : gazebo ;
2008-12-22 19:48:41 +02:00
cb - > showInfoDialog ( & iw ) ;
cb - > changePrimSkill ( heroID , 4 , val ) ;
break ;
}
2012-09-23 21:01:04 +03:00
case Obj : : TREE_OF_KNOWLEDGE :
2008-12-22 19:48:41 +02:00
{
const CGHeroInstance * h = cb - > getHero ( heroID ) ;
val = VLC - > heroh - > reqExp ( h - > level + val ) - VLC - > heroh - > reqExp ( h - > level ) ;
if ( ! ttype )
2008-10-26 22:58:34 +02:00
{
2010-05-02 21:20:26 +03:00
cb - > setObjProperty ( this - > id , ObjProperty : : VISITORS , heroID ) ; //add to the visitors
2008-12-22 19:48:41 +02:00
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = sound ;
2008-12-22 19:48:41 +02:00
iw . components . push_back ( Component ( id , subid , 1 , 0 ) ) ;
iw . player = cb - > getOwner ( heroID ) ;
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 148 ) ;
cb - > showInfoDialog ( & iw ) ;
cb - > changePrimSkill ( heroID , 4 , val ) ;
2008-10-26 22:58:34 +02:00
break ;
}
2008-12-22 19:48:41 +02:00
else
{
2010-07-13 22:55:13 +03:00
ui32 res ;
2012-09-23 21:01:04 +03:00
TExpType resval ;
2008-12-22 19:48:41 +02:00
if ( ttype = = 1 )
{
res = 6 ;
resval = 2000 ;
ot = 149 ;
}
else
{
res = 5 ;
resval = 10 ;
ot = 151 ;
}
if ( cb - > getResource ( h - > tempOwner , res ) < resval ) //not enough resources
{
ot + + ;
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = sound ;
2008-12-22 19:48:41 +02:00
iw . player = h - > tempOwner ;
iw . text < < std : : pair < ui8 , ui32 > ( 11 , ot ) ;
cb - > showInfoDialog ( & iw ) ;
return ;
}
2009-10-03 14:16:42 +03:00
BlockingDialog sd ( true , false ) ;
2009-04-24 00:09:10 +03:00
sd . soundID = sound ;
2008-12-22 19:48:41 +02:00
sd . player = cb - > getOwner ( heroID ) ;
sd . text < < std : : pair < ui8 , ui32 > ( 11 , ot ) ;
2009-10-03 14:16:42 +03:00
sd . components . push_back ( Component ( Component : : RESOURCE , res , resval , 0 ) ) ;
2009-04-11 04:32:50 +03:00
cb - > showBlockingDialog ( & sd , boost : : bind ( & CGVisitableOPH : : treeSelected , this , heroID , res , resval , val , _1 ) ) ;
2008-12-22 19:48:41 +02:00
}
break ;
2008-10-26 22:58:34 +02:00
}
2012-09-23 21:01:04 +03:00
case Obj : : LIBRARY_OF_ENLIGHTENMENT :
2009-02-01 16:11:41 +02:00
{
const CGHeroInstance * h = cb - > getHero ( heroID ) ;
2010-12-23 22:18:10 +02:00
if ( h - > level < 10 - 2 * h - > getSecSkillLevel ( CGHeroInstance : : DIPLOMACY ) ) //not enough level
2009-02-01 16:11:41 +02:00
{
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = sound ;
2009-02-01 16:11:41 +02:00
iw . player = cb - > getOwner ( heroID ) ;
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 68 ) ;
cb - > showInfoDialog ( & iw ) ;
}
else
{
2010-05-02 21:20:26 +03:00
cb - > setObjProperty ( this - > id , ObjProperty : : VISITORS , heroID ) ; //add to the visitors
2009-02-01 16:11:41 +02:00
cb - > changePrimSkill ( heroID , 0 , 2 ) ;
cb - > changePrimSkill ( heroID , 1 , 2 ) ;
cb - > changePrimSkill ( heroID , 2 , 2 ) ;
cb - > changePrimSkill ( heroID , 3 , 2 ) ;
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = sound ;
2009-02-01 16:11:41 +02:00
iw . player = cb - > getOwner ( heroID ) ;
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 66 ) ;
cb - > showInfoDialog ( & iw ) ;
}
break ;
}
2012-09-23 21:01:04 +03:00
case Obj : : SCHOOL_OF_MAGIC :
case Obj : : SCHOOL_OF_WAR :
2009-04-11 04:32:50 +03:00
{
2012-09-23 21:01:04 +03:00
int skill = ( ID = = Obj : : SCHOOL_OF_MAGIC ? 2 : 0 ) ;
2009-04-11 04:32:50 +03:00
if ( cb - > getResource ( cb - > getOwner ( heroID ) , 6 ) < 1000 ) //not enough resources
{
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = sound ;
2009-04-11 04:32:50 +03:00
iw . player = cb - > getOwner ( heroID ) ;
iw . text < < std : : pair < ui8 , ui32 > ( MetaString : : ADVOB_TXT , ot + 2 ) ;
cb - > showInfoDialog ( & iw ) ;
}
else
{
BlockingDialog sd ( true , true ) ;
2009-04-24 00:09:10 +03:00
sd . soundID = sound ;
2009-04-11 04:32:50 +03:00
sd . player = cb - > getOwner ( heroID ) ;
sd . text < < std : : pair < ui8 , ui32 > ( 11 , ot ) ;
sd . components . push_back ( Component ( Component : : PRIM_SKILL , skill , + 1 , 0 ) ) ;
sd . components . push_back ( Component ( Component : : PRIM_SKILL , skill + 1 , + 1 , 0 ) ) ;
cb - > showBlockingDialog ( & sd , boost : : bind ( & CGVisitableOPH : : schoolSelected , this , heroID , _1 ) ) ;
}
}
break ;
2008-10-26 22:58:34 +02:00
}
}
2008-12-22 19:48:41 +02:00
else
{
ot + + ;
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = sound ;
2008-12-22 19:48:41 +02:00
iw . player = cb - > getOwner ( heroID ) ;
iw . text < < std : : pair < ui8 , ui32 > ( 11 , ot ) ;
cb - > showInfoDialog ( & iw ) ;
}
2008-02-07 20:45:22 +02:00
}
2007-10-27 22:38:48 +03:00
2008-12-22 19:48:41 +02:00
const std : : string & CGVisitableOPH : : getHoverText ( ) const
2007-10-27 22:38:48 +03:00
{
2009-01-11 00:08:18 +02:00
int pom = - 1 ;
2008-12-22 19:48:41 +02:00
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : ARENA :
2009-01-11 00:08:18 +02:00
pom = - 1 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : MERCENARY_CAMP :
2012-07-19 21:52:44 +03:00
pom = 8 ;
2008-12-22 19:48:41 +02:00
break ;
2012-09-23 21:01:04 +03:00
case Obj : : MARLETTO_TOWER :
2008-12-22 19:48:41 +02:00
pom = 7 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : STAR_AXIS :
2012-07-19 21:52:44 +03:00
pom = 11 ;
2008-12-22 19:48:41 +02:00
break ;
2012-09-23 21:01:04 +03:00
case Obj : : GARDEN_OF_REVELATION :
2012-07-19 21:52:44 +03:00
pom = 4 ;
2008-12-22 19:48:41 +02:00
break ;
2012-09-23 21:01:04 +03:00
case Obj : : LEARNING_STONE :
2012-07-19 21:52:44 +03:00
pom = 5 ;
2008-12-22 19:48:41 +02:00
break ;
2012-09-23 21:01:04 +03:00
case Obj : : TREE_OF_KNOWLEDGE :
2008-12-22 19:48:41 +02:00
pom = 18 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : LIBRARY_OF_ENLIGHTENMENT :
2009-02-01 16:11:41 +02:00
break ;
2012-09-23 21:01:04 +03:00
case Obj : : SCHOOL_OF_MAGIC :
2009-04-11 04:32:50 +03:00
pom = 9 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : SCHOOL_OF_WAR :
2009-04-11 04:32:50 +03:00
pom = 10 ;
break ;
2008-12-22 19:48:41 +02:00
default :
2012-04-22 10:32:45 +03:00
throw std : : runtime_error ( " Wrong CGVisitableOPH object ID! \n " ) ;
2008-12-22 19:48:41 +02:00
}
2009-01-11 00:08:18 +02:00
hoverName = VLC - > generaltexth - > names [ ID ] ;
if ( pom > = 0 )
2010-07-31 03:26:34 +03:00
hoverName + = ( " \n " + VLC - > generaltexth - > xtrainfo [ pom ] ) ;
2008-12-22 19:48:41 +02:00
const CGHeroInstance * h = cb - > getSelectedHero ( cb - > getCurrentPlayer ( ) ) ;
if ( h )
{
2010-07-31 03:26:34 +03:00
hoverName + = " \n \n " ;
2012-07-19 21:52:44 +03:00
hoverName + = ( vstd : : contains ( visitors , h - > id ) )
2008-12-27 03:01:59 +02:00
? ( VLC - > generaltexth - > allTexts [ 352 ] ) //visited
: ( VLC - > generaltexth - > allTexts [ 353 ] ) ; //not visited
2008-12-22 19:48:41 +02:00
}
return hoverName ;
2007-10-27 22:38:48 +03:00
}
2009-01-11 00:08:18 +02:00
void CGVisitableOPH : : arenaSelected ( int heroID , int primSkill ) const
{
2010-05-02 21:20:26 +03:00
cb - > setObjProperty ( id , ObjProperty : : VISITORS , heroID ) ; //add to the visitors
2009-04-11 04:32:50 +03:00
cb - > changePrimSkill ( heroID , primSkill - 1 , 2 ) ;
2009-01-11 00:08:18 +02:00
}
2009-02-20 14:39:27 +02:00
void CGVisitableOPH : : setPropertyDer ( ui8 what , ui32 val )
{
2010-05-02 21:20:26 +03:00
if ( what = = ObjProperty : : VISITORS )
2009-02-20 14:39:27 +02:00
visitors . insert ( val ) ;
}
2009-04-11 04:32:50 +03:00
void CGVisitableOPH : : schoolSelected ( int heroID , ui32 which ) const
{
if ( ! which ) //player refused to pay
return ;
2012-09-23 21:01:04 +03:00
int base = ( ID = = Obj : : SCHOOL_OF_MAGIC ? 2 : 0 ) ;
2010-05-02 21:20:26 +03:00
cb - > setObjProperty ( id , ObjProperty : : VISITORS , heroID ) ; //add to the visitors
2009-04-11 04:32:50 +03:00
cb - > giveResource ( cb - > getOwner ( heroID ) , 6 , - 1000 ) ; //take 1000 gold
cb - > changePrimSkill ( heroID , base + which - 1 , + 1 ) ; //give appropriate skill
}
2009-10-05 19:38:54 +03:00
COPWBonus : : COPWBonus ( int index , CGTownInstance * TOWN )
{
ID = index ;
town = TOWN ;
id = town - > bonusingBuildings . size ( ) ;
}
void COPWBonus : : setProperty ( ui8 what , ui32 val )
{
switch ( what )
{
case 4 :
visitors . insert ( val ) ;
break ;
2009-10-06 10:55:39 +03:00
case 12 :
2009-10-05 19:38:54 +03:00
visitors . clear ( ) ;
break ;
}
}
void COPWBonus : : onHeroVisit ( const CGHeroInstance * h ) const
{
int heroID = h - > id ;
2012-09-05 15:49:23 +03:00
if ( town - > hasBuilt ( ID ) )
2009-10-05 19:38:54 +03:00
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
2009-10-06 10:55:39 +03:00
switch ( town - > subID )
2009-10-05 19:38:54 +03:00
{
2012-09-05 15:49:23 +03:00
case ETownType : : CASTLE : //Stables
2010-05-02 21:20:26 +03:00
if ( ! h - > hasBonusFrom ( Bonus : : OBJECT , 94 ) ) //does not stack with advMap Stables
2009-10-05 19:38:54 +03:00
{
GiveBonus gb ;
2010-05-02 21:20:26 +03:00
gb . bonus = Bonus ( Bonus : : ONE_WEEK , Bonus : : LAND_MOVEMENT , Bonus : : OBJECT , 600 , 94 , VLC - > generaltexth - > arraytxt [ 100 ] ) ;
2010-02-10 04:56:00 +02:00
gb . id = heroID ;
2009-10-05 19:38:54 +03:00
cb - > giveHeroBonus ( & gb ) ;
2009-10-06 10:55:39 +03:00
iw . text < < VLC - > generaltexth - > allTexts [ 580 ] ;
cb - > showInfoDialog ( & iw ) ;
}
break ;
2012-09-05 15:49:23 +03:00
case ETownType : : DUNGEON : //Mana Vortex
2011-01-15 19:30:46 +02:00
if ( visitors . empty ( ) & & h - > mana < = h - > manaLimit ( ) * 2 )
2009-10-06 10:55:39 +03:00
{
2012-07-19 21:52:44 +03:00
cb - > setManaPoints ( heroID , 2 * h - > manaLimit ( ) ) ;
2010-05-02 21:20:26 +03:00
cb - > setObjProperty ( id , ObjProperty : : VISITED , true ) ;
2009-10-06 10:55:39 +03:00
iw . text < < VLC - > generaltexth - > allTexts [ 579 ] ;
2009-10-05 19:38:54 +03:00
cb - > showInfoDialog ( & iw ) ;
2009-10-06 10:55:39 +03:00
cb - > setObjProperty ( town - > id , 11 , id ) ; //add to visitors
2009-10-05 19:38:54 +03:00
}
break ;
}
}
}
2009-08-23 18:02:21 +03:00
CTownBonus : : CTownBonus ( int index , CGTownInstance * TOWN )
{
ID = index ;
town = TOWN ;
2009-09-21 12:00:33 +03:00
id = town - > bonusingBuildings . size ( ) ;
}
void CTownBonus : : setProperty ( ui8 what , ui32 val )
{
if ( what = = 4 )
visitors . insert ( val ) ;
2009-08-23 18:02:21 +03:00
}
void CTownBonus : : onHeroVisit ( const CGHeroInstance * h ) const
{
int heroID = h - > id ;
2012-09-05 15:49:23 +03:00
if ( town - > hasBuilt ( ID ) & & visitors . find ( heroID ) = = visitors . end ( ) )
2009-08-23 18:02:21 +03:00
{
InfoWindow iw ;
2012-07-06 23:19:54 +03:00
int what = 0 , val = 0 , mid = 0 ;
2009-08-23 18:02:21 +03:00
switch ( ID )
{
2012-09-05 15:49:23 +03:00
case EBuilding : : SPECIAL_4 :
2009-09-21 12:00:33 +03:00
switch ( town - > subID )
2009-08-23 18:02:21 +03:00
{
2012-09-05 15:49:23 +03:00
case ETownType : : TOWER : //wall
2009-10-05 14:28:49 +03:00
what = 3 ;
val = 1 ;
2009-10-06 10:55:39 +03:00
mid = 581 ;
2009-08-23 18:02:21 +03:00
iw . components . push_back ( Component ( Component : : PRIM_SKILL , 3 , 1 , 0 ) ) ;
break ;
2012-09-05 15:49:23 +03:00
case ETownType : : INFERNO : //order of fire
2009-10-05 14:28:49 +03:00
what = 2 ;
val = 1 ;
2009-10-06 10:55:39 +03:00
mid = 582 ;
2009-08-23 18:02:21 +03:00
iw . components . push_back ( Component ( Component : : PRIM_SKILL , 2 , 1 , 0 ) ) ;
break ;
2012-09-05 15:49:23 +03:00
case ETownType : : STRONGHOLD : //hall of valhalla
2009-10-05 14:28:49 +03:00
what = 0 ;
val = 1 ;
2009-10-06 10:55:39 +03:00
mid = 584 ;
2009-08-23 18:02:21 +03:00
iw . components . push_back ( Component ( Component : : PRIM_SKILL , 0 , 1 , 0 ) ) ;
break ;
2012-09-05 15:49:23 +03:00
case ETownType : : DUNGEON : //academy of battle scholars
2009-10-05 14:28:49 +03:00
what = 4 ;
2011-12-14 00:23:17 +03:00
val = 1000 * ( 100 + h - > getSecSkillLevel ( CGHeroInstance : : LEARNING ) * 5 ) / 100.0 ;
2009-10-06 10:55:39 +03:00
mid = 583 ;
2010-07-26 01:47:59 +03:00
iw . components . push_back ( Component ( Component : : EXPERIENCE , 0 , val , 0 ) ) ;
2009-08-23 18:02:21 +03:00
break ;
}
break ;
2012-09-05 15:49:23 +03:00
case EBuilding : : SPECIAL_1 :
2009-09-21 12:00:33 +03:00
switch ( town - > subID )
2009-08-23 18:02:21 +03:00
{
2012-09-05 15:49:23 +03:00
case ETownType : : FORTRESS : //cage of warlords
2009-10-05 14:28:49 +03:00
what = 1 ;
val = 1 ;
2009-10-06 10:55:39 +03:00
mid = 585 ;
2009-08-23 18:02:21 +03:00
iw . components . push_back ( Component ( Component : : PRIM_SKILL , 1 , 1 , 0 ) ) ;
break ;
}
break ;
}
2012-07-06 23:19:54 +03:00
assert ( mid ) ;
2009-08-23 18:02:21 +03:00
iw . player = cb - > getOwner ( heroID ) ;
2009-10-06 10:55:39 +03:00
iw . text < < VLC - > generaltexth - > allTexts [ mid ] ;
2009-08-23 18:02:21 +03:00
cb - > showInfoDialog ( & iw ) ;
2009-10-05 14:28:49 +03:00
cb - > changePrimSkill ( heroID , what , val ) ;
2009-12-30 22:49:10 +02:00
if ( town - > visitingHero = = h )
cb - > setObjProperty ( town - > id , 11 , id ) ; //add to visitors
else
cb - > setObjProperty ( town - > id , 13 , id ) ; //then it must be garrisoned hero
2009-08-23 18:02:21 +03:00
}
}
2010-06-28 16:33:05 +03:00
const std : : string & CGCreature : : getHoverText ( ) const
2009-08-11 10:50:29 +03:00
{
2010-06-28 16:33:05 +03:00
MetaString ms ;
2011-04-10 11:33:53 +03:00
int pom = stacks . begin ( ) - > second - > getQuantityID ( ) ;
2012-08-08 11:27:52 +03:00
pom = 172 + 3 * pom ;
2010-06-28 16:33:05 +03:00
ms < < std : : pair < ui8 , ui32 > ( 6 , pom ) < < " " < < std : : pair < ui8 , ui32 > ( 7 , subID ) ;
ms . toString ( hoverName ) ;
2010-07-07 05:29:07 +03:00
if ( const CGHeroInstance * selHero = cb - > getSelectedHero ( cb - > getCurrentPlayer ( ) ) )
{
2011-02-07 16:39:17 +02:00
std : : vector < std : : string > * texts = & VLC - > generaltexth - > threat ;
hoverName + = " \n \n " ;
hoverName + = ( * texts ) [ 0 ] ;
int choice ;
2011-06-21 12:31:08 +03:00
double ratio = ( ( double ) getArmyStrength ( ) / selHero - > getTotalStrength ( ) ) ;
2011-02-07 16:39:17 +02:00
if ( ratio < 0.1 ) choice = 1 ;
else if ( ratio < 0.25 ) choice = 2 ;
else if ( ratio < 0.6 ) choice = 3 ;
else if ( ratio < 0.9 ) choice = 4 ;
else if ( ratio < 1.1 ) choice = 5 ;
else if ( ratio < 1.3 ) choice = 6 ;
else if ( ratio < 1.8 ) choice = 7 ;
else if ( ratio < 2.5 ) choice = 8 ;
else if ( ratio < 4 ) choice = 9 ;
else if ( ratio < 8 ) choice = 10 ;
else if ( ratio < 20 ) choice = 11 ;
else choice = 12 ;
hoverName + = ( * texts ) [ choice ] ;
2010-07-07 05:29:07 +03:00
}
2009-08-11 10:50:29 +03:00
return hoverName ;
2010-06-28 16:33:05 +03:00
}
2008-12-27 03:01:59 +02:00
void CGCreature : : onHeroVisit ( const CGHeroInstance * h ) const
2008-11-28 03:36:34 +02:00
{
2009-04-12 04:48:50 +03:00
int action = takenAction ( h ) ;
switch ( action ) //decide what we do...
{
case - 2 : //fight
fight ( h ) ;
break ;
case - 1 : //flee
{
flee ( h ) ;
break ;
}
case 0 : //join for free
{
BlockingDialog ynd ( true , false ) ;
ynd . player = h - > tempOwner ;
2012-07-19 21:52:44 +03:00
ynd . text < < std : : pair < ui8 , ui32 > ( MetaString : : ADVOB_TXT , 86 ) ;
2009-07-09 22:15:22 +03:00
ynd . text . addReplacement ( MetaString : : CRE_PL_NAMES , subID ) ;
2009-04-12 04:48:50 +03:00
cb - > showBlockingDialog ( & ynd , boost : : bind ( & CGCreature : : joinDecision , this , h , 0 , _1 ) ) ;
break ;
}
default : //join for gold
{
assert ( action > 0 ) ;
//ask if player agrees to pay gold
BlockingDialog ynd ( true , false ) ;
ynd . player = h - > tempOwner ;
std : : string tmp = VLC - > generaltexth - > advobtxt [ 90 ] ;
2010-11-27 03:46:19 +02:00
boost : : algorithm : : replace_first ( tmp , " %d " , boost : : lexical_cast < std : : string > ( getStackCount ( 0 ) ) ) ;
2009-04-12 04:48:50 +03:00
boost : : algorithm : : replace_first ( tmp , " %d " , boost : : lexical_cast < std : : string > ( action ) ) ;
2010-05-02 21:20:26 +03:00
boost : : algorithm : : replace_first ( tmp , " %s " , VLC - > creh - > creatures [ subID ] - > namePl ) ;
2009-04-12 04:48:50 +03:00
ynd . text < < tmp ;
cb - > showBlockingDialog ( & ynd , boost : : bind ( & CGCreature : : joinDecision , this , h , action , _1 ) ) ;
break ;
}
}
2008-11-28 03:36:34 +02:00
}
2008-09-12 11:51:46 +03:00
2008-12-27 03:01:59 +02:00
void CGCreature : : endBattle ( BattleResult * result ) const
2008-09-12 11:51:46 +03:00
{
2008-12-22 19:48:41 +02:00
if ( result - > winner = = 0 )
{
cb - > removeObject ( id ) ;
}
else
{
2009-07-21 02:34:06 +03:00
//int killedAmount=0;
//for(std::set<std::pair<ui32,si32> >::iterator i=result->casualties[1].begin(); i!=result->casualties[1].end(); i++)
// if(i->first == subID)
// killedAmount += i->second;
2012-07-19 21:52:44 +03:00
//cb->setAmount(id, slots.find(0)->second.second - killedAmount);
2009-07-21 02:34:06 +03:00
2010-06-28 16:33:05 +03:00
/*
2008-12-27 03:01:59 +02:00
MetaString ms ;
2010-05-02 21:20:26 +03:00
int pom = slots . find ( 0 ) - > second . getQuantityID ( ) ;
2008-12-27 03:01:59 +02:00
pom = 174 + 3 * pom + 1 ;
ms < < std : : pair < ui8 , ui32 > ( 6 , pom ) < < " " < < std : : pair < ui8 , ui32 > ( 7 , subID ) ;
cb - > setHoverName ( id , & ms ) ;
2010-06-26 15:57:16 +03:00
cb - > setObjProperty ( id , 11 , slots . begin ( ) - > second . count * 1000 ) ;
2010-06-28 16:33:05 +03:00
*/
2011-04-14 10:58:19 +03:00
//merge stacks into one
TSlots : : const_iterator i ;
CCreature * cre = VLC - > creh - > creatures [ restore . basicType ] ;
for ( i = stacks . begin ( ) ; i ! = stacks . end ( ) ; i + + )
{
if ( cre - > isMyUpgrade ( i - > second - > type ) )
{
cb - > changeStackType ( StackLocation ( this , i - > first ) , cre ) ; //un-upgrade creatures
}
}
2011-09-06 09:00:32 +03:00
//first stack has to be at slot 0 -> if original one got killed, move there first remaining stack
if ( ! hasStackAtSlot ( 0 ) )
cb - > moveStack ( StackLocation ( this , stacks . begin ( ) - > first ) , StackLocation ( this , 0 ) , stacks . begin ( ) - > second - > count ) ;
2011-04-14 10:58:19 +03:00
while ( stacks . size ( ) > 1 ) //hopefully that's enough
{
2011-09-06 09:00:32 +03:00
// TODO it's either overcomplicated (if we assume there'll be only one stack) or buggy (if we allow multiple stacks... but that'll also cause troubles elsewhere)
2011-04-14 10:58:19 +03:00
i = stacks . end ( ) ;
i - - ;
TSlot slot = getSlotFor ( i - > second - > type ) ;
if ( slot = = i - > first ) //no reason to move stack to its own slot
break ;
else
cb - > moveStack ( StackLocation ( this , i - > first ) , StackLocation ( this , slot ) , i - > second - > count ) ;
}
2012-03-06 11:39:05 +03:00
cb - > setObjProperty ( id , ObjProperty : : MONSTER_POWER , stacks . begin ( ) - > second - > count * 1000 ) ; //remember casualties
2008-12-22 19:48:41 +02:00
}
2008-09-12 11:51:46 +03:00
}
2008-11-15 02:55:19 +02:00
2008-12-22 19:48:41 +02:00
void CGCreature : : initObj ( )
2007-10-27 22:38:48 +03:00
{
2009-04-08 22:55:50 +03:00
blockVisit = true ;
2009-04-12 04:48:50 +03:00
switch ( character )
{
case 0 :
2012-01-19 17:33:22 +03:00
character = - 4 ;
2009-04-12 04:48:50 +03:00
break ;
case 1 :
character = 1 + ran ( ) % 7 ;
break ;
case 2 :
character = 1 + ran ( ) % 10 ;
break ;
case 3 :
character = 4 + ran ( ) % 7 ;
break ;
case 4 :
character = 10 ;
break ;
}
2011-01-21 04:36:30 +02:00
stacks [ 0 ] - > setType ( subID ) ;
TQuantity & amount = stacks [ 0 ] - > count ;
2010-05-02 21:20:26 +03:00
CCreature & c = * VLC - > creh - > creatures [ subID ] ;
2008-12-22 19:48:41 +02:00
if ( ! amount )
2010-08-30 02:12:34 +03:00
{
2008-12-22 19:48:41 +02:00
if ( c . ammMax = = c . ammMin )
amount = c . ammMax ;
else
amount = c . ammMin + ( ran ( ) % ( c . ammMax - c . ammMin ) ) ;
2010-08-30 02:12:34 +03:00
}
2010-07-07 15:51:28 +03:00
2011-01-21 04:36:30 +02:00
temppower = stacks [ 0 ] - > count * 1000 ;
2010-06-26 15:57:16 +03:00
}
void CGCreature : : newTurn ( ) const
{ //Works only for stacks of single type of size up to 2 millions
2012-08-11 12:06:23 +03:00
if ( stacks . begin ( ) - > second - > count < VLC - > modh - > settings . CREEP_SIZE & & cb - > getDate ( 1 ) = = 1 & & cb - > getDate ( 0 ) > 1 )
2010-06-26 15:57:16 +03:00
{
2012-08-11 12:06:23 +03:00
ui32 power = temppower * ( 100 + VLC - > modh - > settings . WEEKLY_GROWTH ) / 100 ;
cb - > setObjProperty ( id , ObjProperty : : MONSTER_COUNT , std : : min ( power / 1000 , ( ui32 ) VLC - > modh - > settings . CREEP_SIZE ) ) ; //set new amount
2011-05-28 04:02:28 +03:00
cb - > setObjProperty ( id , ObjProperty : : MONSTER_POWER , power ) ; //increase temppower
2010-06-26 15:57:16 +03:00
}
2012-08-24 12:37:52 +03:00
if ( VLC - > modh - > modules . STACK_EXP )
2012-08-11 12:06:23 +03:00
cb - > setObjProperty ( id , ObjProperty : : MONSTER_EXP , VLC - > modh - > settings . NEUTRAL_STACK_EXP ) ; //for testing purpose
2010-06-26 15:57:16 +03:00
}
void CGCreature : : setPropertyDer ( ui8 what , ui32 val )
{
switch ( what )
{
2011-05-28 04:02:28 +03:00
case ObjProperty : : MONSTER_COUNT :
2011-01-21 04:36:30 +02:00
stacks [ 0 ] - > count = val ;
2010-06-26 15:57:16 +03:00
break ;
2011-05-28 04:02:28 +03:00
case ObjProperty : : MONSTER_POWER :
2010-06-26 15:57:16 +03:00
temppower = val ;
break ;
2011-05-28 04:02:28 +03:00
case ObjProperty : : MONSTER_EXP :
2011-02-13 15:11:09 +02:00
giveStackExp ( val ) ;
break ;
2011-09-06 09:00:32 +03:00
case ObjProperty : : MONSTER_RESTORE_TYPE :
2011-04-14 10:58:19 +03:00
restore . basicType = val ;
break ;
2010-06-26 15:57:16 +03:00
}
2008-12-27 03:01:59 +02:00
}
2009-04-12 04:48:50 +03:00
int CGCreature : : takenAction ( const CGHeroInstance * h , bool allowJoin ) const
{
2012-01-19 17:33:22 +03:00
//calculate relative strength of hero and creatures armies
double relStrength = double ( h - > getTotalStrength ( ) ) / getArmyStrength ( ) ;
2009-04-12 04:48:50 +03:00
2012-01-19 17:33:22 +03:00
int powerFactor ;
if ( relStrength > = 7 )
powerFactor = 11 ;
2009-04-12 04:48:50 +03:00
2012-01-19 17:33:22 +03:00
else if ( relStrength > = 1 )
powerFactor = ( int ) ( 2 * ( relStrength - 1 ) ) ;
2009-04-12 04:48:50 +03:00
2012-01-19 17:33:22 +03:00
else if ( relStrength > = 0.5 )
powerFactor = - 1 ;
2009-04-12 04:48:50 +03:00
2012-01-19 17:33:22 +03:00
else if ( relStrength > = 0.333 )
powerFactor = - 2 ;
else
powerFactor = - 3 ;
2010-04-02 05:07:40 +03:00
2012-01-19 17:33:22 +03:00
std : : set < ui32 > myKindCres ; //what creatures are the same kind as we
myKindCres . insert ( subID ) ; //we
myKindCres . insert ( VLC - > creh - > creatures [ subID ] - > upgrades . begin ( ) , VLC - > creh - > creatures [ subID ] - > upgrades . end ( ) ) ; //our upgrades
2012-07-19 21:52:44 +03:00
2012-01-19 17:33:22 +03:00
BOOST_FOREACH ( ConstTransitivePtr < CCreature > & crea , VLC - > creh - > creatures )
{
if ( vstd : : contains ( crea - > upgrades , ( ui32 ) id ) ) //it's our base creatures
myKindCres . insert ( crea - > idNumber ) ;
}
2009-04-12 04:48:50 +03:00
2012-01-19 17:33:22 +03:00
int count = 0 , //how many creatures of similar kind has hero
totalCount = 0 ;
2009-04-12 04:48:50 +03:00
2012-01-19 17:33:22 +03:00
for ( TSlots : : const_iterator i = h - > Slots ( ) . begin ( ) ; i ! = h - > Slots ( ) . end ( ) ; i + + )
{
if ( vstd : : contains ( myKindCres , i - > second - > type - > idNumber ) )
count + = i - > second - > count ;
totalCount + = i - > second - > count ;
2009-04-12 04:48:50 +03:00
}
2012-01-19 17:33:22 +03:00
int sympathy = 0 ; // 0 if hero have no similar creatures
if ( count )
sympathy + + ; // 1 - if hero have at least 1 similar creature
if ( count * 2 > totalCount )
sympathy + + ; // 2 - hero have similar creatures more that 50%
2009-04-12 04:48:50 +03:00
2012-01-19 17:33:22 +03:00
int charisma = powerFactor + h - > getSecSkillLevel ( CGHeroInstance : : DIPLOMACY ) + sympathy ;
if ( charisma < character ) //creatures will fight
return - 2 ;
if ( allowJoin )
{
if ( h - > getSecSkillLevel ( CGHeroInstance : : DIPLOMACY ) + sympathy + 1 > = character )
return 0 ; //join for free
else if ( h - > getSecSkillLevel ( CGHeroInstance : : DIPLOMACY ) * 2 + sympathy + 1 > = character )
return VLC - > creh - > creatures [ subID ] - > cost [ 6 ] * getStackCount ( 0 ) ; //join for gold
}
//we are still here - creatures have not joined hero, flee or fight
if ( charisma > character )
2009-04-16 03:28:54 +03:00
return - 1 ; //flee
2009-04-12 04:48:50 +03:00
else
2009-04-16 03:28:54 +03:00
return - 2 ; //fight
2009-04-12 04:48:50 +03:00
}
void CGCreature : : fleeDecision ( const CGHeroInstance * h , ui32 pursue ) const
{
if ( pursue )
{
fight ( h ) ;
}
else
{
cb - > removeObject ( id ) ;
}
}
void CGCreature : : joinDecision ( const CGHeroInstance * h , int cost , ui32 accept ) const
{
if ( ! accept )
{
if ( takenAction ( h , false ) = = - 1 ) //they flee
{
flee ( h ) ;
}
else //they fight
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 87 ) ; //Insulted by your refusal of their offer, the monsters attack!
cb - > showInfoDialog ( & iw ) ;
fight ( h ) ;
}
}
else //accepted
{
if ( cb - > getResource ( h - > tempOwner , 6 ) < cost ) //player don't have enough gold!
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text < < std : : pair < ui8 , ui32 > ( 1 , 29 ) ; //You don't have enough gold
cb - > showInfoDialog ( & iw ) ;
//act as if player refused
2009-08-04 02:53:18 +03:00
joinDecision ( h , cost , false ) ;
2009-04-12 04:48:50 +03:00
return ;
}
//take gold
if ( cost )
cb - > giveResource ( h - > tempOwner , 6 , - cost ) ;
2011-06-23 13:12:35 +03:00
cb - > tryJoiningArmy ( this , h , true , true ) ;
2009-04-12 04:48:50 +03:00
}
}
void CGCreature : : fight ( const CGHeroInstance * h ) const
{
2011-04-10 11:33:53 +03:00
//split stacks
2011-11-01 15:58:01 +03:00
//TODO: multiple creature types in a stack?
2011-04-10 11:33:53 +03:00
int basicType = stacks . begin ( ) - > second - > type - > idNumber ;
2011-09-06 09:00:32 +03:00
cb - > setObjProperty ( id , ObjProperty : : MONSTER_RESTORE_TYPE , basicType ) ; //store info about creature stack
2011-04-10 11:33:53 +03:00
2011-12-14 00:23:17 +03:00
double relativePower = static_cast < double > ( h - > getTotalStrength ( ) ) / getArmyStrength ( ) ;
2011-04-10 11:33:53 +03:00
int stacksCount ;
//TODO: number depends on tile type
if ( relativePower < 0.5 )
{
stacksCount = 7 ;
}
else if ( relativePower < 0.67 )
{
stacksCount = 7 ;
}
else if ( relativePower < 1 )
{
stacksCount = 6 ;
}
else if ( relativePower < 1.5 )
{
stacksCount = 5 ;
}
else if ( relativePower < 2 )
{
stacksCount = 4 ;
}
else
{
stacksCount = 3 ;
}
int stackSize ;
TSlot sourceSlot = stacks . begin ( ) - > first ;
TSlot destSlot ;
for ( int stacksLeft = stacksCount ; stacksLeft > 1 ; - - stacksLeft )
{
stackSize = stacks . begin ( ) - > second - > count / stacksLeft ;
if ( stackSize )
{
if ( ( destSlot = getFreeSlot ( ) ) > - 1 )
cb - > moveStack ( StackLocation ( this , sourceSlot ) , StackLocation ( this , destSlot ) , stackSize ) ;
else
{
tlog2 < < " Warning! Not enough empty slots to split stack! " ;
break ;
}
}
else break ;
}
if ( stacksCount > 1 )
{
2011-04-14 10:58:19 +03:00
if ( rand ( ) % 100 < 50 ) //upgrade
2011-04-10 11:33:53 +03:00
{
TSlot slotId = ( stacks . size ( ) / 2 ) ;
if ( ui32 upgradesSize = getStack ( slotId ) . type - > upgrades . size ( ) )
{
std : : set < TCreature > : : const_iterator it = getStack ( slotId ) . type - > upgrades . begin ( ) ; //pick random in case there are more
std : : advance ( it , rand ( ) % upgradesSize ) ;
cb - > changeStackType ( StackLocation ( this , slotId ) , VLC - > creh - > creatures [ * it ] ) ;
}
}
}
2009-09-13 01:17:23 +03:00
cb - > startBattleI ( h , this , boost : : bind ( & CGCreature : : endBattle , this , _1 ) ) ;
2011-04-10 11:33:53 +03:00
2009-04-12 04:48:50 +03:00
}
void CGCreature : : flee ( const CGHeroInstance * h ) const
{
BlockingDialog ynd ( true , false ) ;
ynd . player = h - > tempOwner ;
2012-07-19 21:52:44 +03:00
ynd . text < < std : : pair < ui8 , ui32 > ( 11 , 91 ) ;
2009-07-09 22:15:22 +03:00
ynd . text . addReplacement ( MetaString : : CRE_PL_NAMES , subID ) ;
2009-04-12 04:48:50 +03:00
cb - > showBlockingDialog ( & ynd , boost : : bind ( & CGCreature : : fleeDecision , this , h , _1 ) ) ;
}
2008-12-27 03:01:59 +02:00
void CGMine : : onHeroVisit ( const CGHeroInstance * h ) const
{
2010-08-13 13:46:08 +03:00
int relations = cb - > gameState ( ) - > getPlayerRelations ( h - > tempOwner , tempOwner ) ;
2012-07-19 21:52:44 +03:00
2010-08-06 16:14:10 +03:00
if ( relations = = 2 ) //we're visiting our mine
2009-04-12 03:58:41 +03:00
{
2009-09-09 20:49:03 +03:00
cb - > showGarrisonDialog ( id , h - > id , true , 0 ) ;
2012-07-19 21:52:44 +03:00
return ;
2009-04-12 03:58:41 +03:00
}
2010-08-06 16:14:10 +03:00
else if ( relations = = 1 ) //ally
return ;
2008-12-27 03:01:59 +02:00
2010-08-06 16:14:10 +03:00
if ( stacksCount ( ) ) //Mine is guarded
2010-07-31 03:26:34 +03:00
{
BlockingDialog ynd ( true , false ) ;
ynd . player = h - > tempOwner ;
2012-07-19 21:52:44 +03:00
ynd . text < < std : : pair < ui8 , ui32 > ( MetaString : : ADVOB_TXT , subID = = 7 ? 84 : 187 ) ;
2010-07-31 03:26:34 +03:00
cb - > showBlockingDialog ( & ynd , boost : : bind ( & CGMine : : fight , this , _1 , h ) ) ;
return ;
}
2008-12-27 03:01:59 +02:00
2010-07-31 03:26:34 +03:00
flagMine ( h - > tempOwner ) ;
2008-12-27 03:01:59 +02:00
}
void CGMine : : newTurn ( ) const
{
2009-05-25 02:21:55 +03:00
if ( cb - > getDate ( ) = = 1 )
return ;
2011-12-14 00:23:17 +03:00
if ( tempOwner = = GameConstants : : NEUTRAL_PLAYER )
2008-12-27 03:01:59 +02:00
return ;
2010-07-31 04:38:12 +03:00
cb - > giveResource ( tempOwner , producedResource , producedQuantity ) ;
2008-12-27 03:01:59 +02:00
}
void CGMine : : initObj ( )
{
2010-07-31 03:26:34 +03:00
if ( subID > = 7 ) //Abandoned Mine
{
//set guardians
int howManyTroglodytes = 100 + ran ( ) % 100 ;
2010-11-22 02:34:46 +02:00
CStackInstance * troglodytes = new CStackInstance ( 70 , howManyTroglodytes ) ;
2010-11-27 03:46:19 +02:00
putStack ( 0 , troglodytes ) ;
2010-07-31 03:26:34 +03:00
//after map reading tempOwner placeholds bitmask for allowed resources
std : : vector < int > possibleResources ;
for ( int i = 0 ; i < 8 ; i + + )
if ( tempOwner & 1 < < i )
possibleResources . push_back ( i ) ;
assert ( possibleResources . size ( ) ) ;
producedResource = possibleResources [ ran ( ) % possibleResources . size ( ) ] ;
2011-12-14 00:23:17 +03:00
tempOwner = GameConstants : : NEUTRAL_PLAYER ;
2010-11-22 02:34:46 +02:00
hoverName = VLC - > generaltexth - > mines [ 7 ] . first + " \n " + VLC - > generaltexth - > allTexts [ 202 ] + " " + troglodytes - > getQuantityTXT ( false ) + " " + troglodytes - > type - > namePl ;
2010-07-31 03:26:34 +03:00
}
2008-12-27 03:01:59 +02:00
else
2010-07-31 03:26:34 +03:00
{
producedResource = subID ;
MetaString ms ;
ms < < std : : pair < ui8 , ui32 > ( 9 , producedResource ) ;
2011-12-14 00:23:17 +03:00
if ( tempOwner > = GameConstants : : PLAYER_LIMIT )
2012-07-19 21:52:44 +03:00
tempOwner = GameConstants : : NEUTRAL_PLAYER ;
2010-07-31 03:26:34 +03:00
else
ms < < " ( " < < std : : pair < ui8 , ui32 > ( 6 , 23 + tempOwner ) < < " ) " ;
ms . toString ( hoverName ) ;
}
producedQuantity = defaultResProduction ( ) ;
}
void CGMine : : fight ( ui32 agreed , const CGHeroInstance * h ) const
{
cb - > startBattleI ( h , this , boost : : bind ( & CGMine : : endBattle , this , _1 , h - > tempOwner ) ) ;
}
void CGMine : : endBattle ( BattleResult * result , ui8 attackingPlayer ) const
{
if ( result - > winner = = 0 ) //attacker won
{
if ( subID = = 7 )
{
InfoWindow iw ;
iw . player = attackingPlayer ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , 85 ) ;
cb - > showInfoDialog ( & iw ) ;
}
flagMine ( attackingPlayer ) ;
}
}
void CGMine : : flagMine ( ui8 player ) const
{
assert ( tempOwner ! = player ) ;
cb - > setOwner ( id , player ) ; //not ours? flag it!
MetaString ms ;
ms < < std : : pair < ui8 , ui32 > ( 9 , subID ) < < " \n ( " < < std : : pair < ui8 , ui32 > ( 6 , 23 + player ) < < " ) " ;
if ( subID = = 7 )
{
ms < < " (%s) " ;
ms . addReplacement ( MetaString : : RES_NAMES , producedResource ) ;
}
cb - > setHoverName ( id , & ms ) ;
InfoWindow iw ;
iw . soundID = soundBase : : FLAGMINE ;
iw . text . addTxt ( MetaString : : MINE_EVNTS , producedResource ) ; //not use subID, abandoned mines uses default mine texts
iw . player = player ;
2012-08-31 20:39:11 +03:00
iw . components . push_back ( Component ( Component : : RESOURCE , producedResource , producedQuantity , - 1 ) ) ;
2010-07-31 03:26:34 +03:00
cb - > showInfoDialog ( & iw ) ;
}
ui32 CGMine : : defaultResProduction ( )
{
switch ( producedResource )
{
2013-01-03 15:19:20 +03:00
case Res : : WOOD :
case Res : : ORE :
2010-07-31 03:26:34 +03:00
return 2 ;
2013-01-03 15:19:20 +03:00
case Res : : GOLD :
2010-07-31 03:26:34 +03:00
return 1000 ;
default :
return 1 ;
}
2008-12-27 03:01:59 +02:00
}
void CGResource : : initObj ( )
{
blockVisit = true ;
2009-01-06 20:42:20 +02:00
hoverName = VLC - > generaltexth - > restypes [ subID ] ;
2008-12-27 03:01:59 +02:00
if ( ! amount )
{
switch ( subID )
{
case 6 :
amount = 500 + ( rand ( ) % 6 ) * 100 ;
break ;
case 0 : case 2 :
amount = 6 + ( rand ( ) % 5 ) ;
break ;
default :
amount = 3 + ( rand ( ) % 3 ) ;
break ;
}
}
}
void CGResource : : onHeroVisit ( const CGHeroInstance * h ) const
{
2010-05-02 21:20:26 +03:00
if ( stacksCount ( ) )
2008-12-27 03:01:59 +02:00
{
2009-01-11 00:08:18 +02:00
if ( message . size ( ) )
{
2009-04-11 04:32:50 +03:00
BlockingDialog ynd ( true , false ) ;
2009-01-11 00:08:18 +02:00
ynd . player = h - > getOwner ( ) ;
ynd . text < < message ;
2009-04-11 04:32:50 +03:00
cb - > showBlockingDialog ( & ynd , boost : : bind ( & CGResource : : fightForRes , this , _1 , h ) ) ;
2009-01-11 00:08:18 +02:00
}
else
{
2009-05-02 01:08:03 +03:00
fightForRes ( 1 , h ) ;
2009-01-11 00:08:18 +02:00
}
2008-12-27 03:01:59 +02:00
}
2009-01-11 00:08:18 +02:00
else
{
if ( message . length ( ) )
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text < < message ;
cb - > showInfoDialog ( & iw ) ;
}
collectRes ( h - > getOwner ( ) ) ;
}
}
2008-12-27 03:01:59 +02:00
2009-01-11 00:08:18 +02:00
void CGResource : : collectRes ( int player ) const
{
cb - > giveResource ( player , subID , amount ) ;
2008-12-27 03:01:59 +02:00
ShowInInfobox sii ;
2009-01-11 00:08:18 +02:00
sii . player = player ;
2012-08-31 20:39:11 +03:00
sii . c = Component ( Component : : RESOURCE , subID , amount , 0 ) ;
2008-12-27 03:01:59 +02:00
sii . text < < std : : pair < ui8 , ui32 > ( 11 , 113 ) ;
2009-07-09 22:15:22 +03:00
sii . text . addReplacement ( MetaString : : RES_NAMES , subID ) ;
2008-12-27 03:01:59 +02:00
cb - > showCompInfo ( & sii ) ;
cb - > removeObject ( id ) ;
}
2009-04-11 04:32:50 +03:00
void CGResource : : fightForRes ( ui32 agreed , const CGHeroInstance * h ) const
2009-01-11 00:08:18 +02:00
{
2009-04-11 04:32:50 +03:00
if ( agreed )
2009-09-13 01:17:23 +03:00
cb - > startBattleI ( h , this , boost : : bind ( & CGResource : : endBattle , this , _1 , h ) ) ;
2009-01-11 00:08:18 +02:00
}
void CGResource : : endBattle ( BattleResult * result , const CGHeroInstance * h ) const
{
if ( result - > winner = = 0 ) //attacker won
collectRes ( h - > getOwner ( ) ) ;
}
2008-12-27 03:01:59 +02:00
void CGVisitableOPW : : newTurn ( ) const
{
2010-08-15 19:24:58 +03:00
if ( cb - > getDate ( 1 ) = = 1 ) //first day of week = 1
2008-12-27 03:01:59 +02:00
{
2010-05-02 21:20:26 +03:00
cb - > setObjProperty ( id , ObjProperty : : VISITED , false ) ;
2008-12-27 03:01:59 +02:00
MetaString ms ; //set text to "not visited"
ms < < std : : pair < ui8 , ui32 > ( 3 , ID ) < < " " < < std : : pair < ui8 , ui32 > ( 1 , 353 ) ;
cb - > setHoverName ( id , & ms ) ;
}
}
2012-03-14 16:02:38 +03:00
bool CGVisitableOPW : : wasVisited ( ui8 player ) const
{
2012-07-19 19:29:29 +03:00
return visited ; //TODO: other players should see object as unvisited
2012-03-14 16:02:38 +03:00
}
2008-12-27 03:01:59 +02:00
void CGVisitableOPW : : onHeroVisit ( const CGHeroInstance * h ) const
{
2012-07-06 23:19:54 +03:00
int mid = 0 , sound = 0 ;
2008-12-27 03:01:59 +02:00
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : MYSTICAL_GARDEN :
2009-04-24 00:09:10 +03:00
sound = soundBase : : experience ;
2008-12-27 03:01:59 +02:00
mid = 92 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : WINDMILL :
2009-04-24 00:09:10 +03:00
sound = soundBase : : GENIE ;
2008-12-27 03:01:59 +02:00
mid = 170 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : WATER_WHEEL :
2009-04-24 00:09:10 +03:00
sound = soundBase : : GENIE ;
2008-12-27 03:01:59 +02:00
mid = 164 ;
break ;
2012-07-06 23:19:54 +03:00
default :
assert ( 0 ) ;
2008-12-27 03:01:59 +02:00
}
if ( visited )
{
2012-09-23 21:01:04 +03:00
if ( ID ! = Obj : : WINDMILL )
2008-12-27 03:01:59 +02:00
mid + + ;
2012-07-19 21:52:44 +03:00
else
2008-12-27 03:01:59 +02:00
mid - - ;
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = sound ;
2008-12-27 03:01:59 +02:00
iw . player = h - > tempOwner ;
iw . text < < std : : pair < ui8 , ui32 > ( 11 , mid ) ;
cb - > showInfoDialog ( & iw ) ;
}
else
{
2012-08-31 20:39:11 +03:00
Component : : EComponentType type = Component : : RESOURCE ;
int sub = 0 , val = 0 ;
2008-12-27 03:01:59 +02:00
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : MYSTICAL_GARDEN :
2008-12-27 03:01:59 +02:00
if ( rand ( ) % 2 )
{
sub = 5 ;
val = 5 ;
}
else
{
sub = 6 ;
val = 500 ;
}
break ;
2012-09-23 21:01:04 +03:00
case Obj : : WINDMILL :
2008-12-27 03:01:59 +02:00
mid = 170 ;
sub = ( rand ( ) % 5 ) + 1 ;
val = ( rand ( ) % 4 ) + 3 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : WATER_WHEEL :
2008-12-27 03:01:59 +02:00
mid = 164 ;
sub = 6 ;
2009-02-15 21:49:23 +02:00
if ( cb - > getDate ( 0 ) < 8 )
2008-12-27 03:01:59 +02:00
val = 500 ;
else
val = 1000 ;
}
cb - > giveResource ( h - > tempOwner , sub , val ) ;
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = sound ;
2008-12-27 03:01:59 +02:00
iw . player = h - > tempOwner ;
iw . components . push_back ( Component ( type , sub , val , 0 ) ) ;
iw . text < < std : : pair < ui8 , ui32 > ( 11 , mid ) ;
cb - > showInfoDialog ( & iw ) ;
2010-05-02 21:20:26 +03:00
cb - > setObjProperty ( id , ObjProperty : : VISITED , true ) ;
2008-12-27 03:01:59 +02:00
MetaString ms ; //set text to "visited"
ms < < std : : pair < ui8 , ui32 > ( 3 , ID ) < < " " < < std : : pair < ui8 , ui32 > ( 1 , 352 ) ;
cb - > setHoverName ( id , & ms ) ;
}
}
2009-02-20 14:39:27 +02:00
void CGVisitableOPW : : setPropertyDer ( ui8 what , ui32 val )
{
2010-05-02 21:20:26 +03:00
if ( what = = ObjProperty : : VISITED )
2009-02-20 14:39:27 +02:00
visited = val ;
}
2008-12-27 03:01:59 +02:00
void CGTeleport : : onHeroVisit ( const CGHeroInstance * h ) const
{
int destinationid = - 1 ;
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : MONOLITH1 : //one way - find corresponding exit monolith
2013-01-03 15:19:20 +03:00
if ( vstd : : contains ( objs , Obj : : MONOLITH2 ) & & vstd : : contains ( objs [ Obj : : MONOLITH2 ] , subID ) & & objs [ Obj : : MONOLITH2 ] [ subID ] . size ( ) )
destinationid = objs [ Obj : : MONOLITH2 ] [ subID ] [ rand ( ) % objs [ Obj : : MONOLITH2 ] [ subID ] . size ( ) ] ;
2008-12-27 03:01:59 +02:00
else
tlog2 < < " Cannot find corresponding exit monolith for " < < id < < std : : endl ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : MONOLITH3 : //two way monolith - pick any other one
case Obj : : WHIRLPOOL : //Whirlpool
2010-06-07 16:11:08 +03:00
if ( vstd : : contains ( objs , ID ) & & vstd : : contains ( objs [ ID ] , subID ) & & objs [ ID ] [ subID ] . size ( ) > 1 )
{
while ( ( destinationid = objs [ ID ] [ subID ] [ rand ( ) % objs [ ID ] [ subID ] . size ( ) ] ) = = id ) ; //choose another exit
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : WHIRLPOOL )
2010-06-07 16:11:08 +03:00
{
if ( ! h - > hasBonusOfType ( Bonus : : WHIRLPOOL_PROTECTION ) )
{
2010-12-06 01:10:02 +02:00
if ( h - > Slots ( ) . size ( ) > 1 | | h - > Slots ( ) . begin ( ) - > second - > count > 1 )
2010-06-25 11:19:39 +03:00
{ //we can't remove last unit
2010-12-06 01:10:02 +02:00
TSlot targetstack = h - > Slots ( ) . begin ( ) - > first ; //slot numbers may vary
for ( TSlots : : const_reverse_iterator i = h - > Slots ( ) . rbegin ( ) ; i ! = h - > Slots ( ) . rend ( ) ; i + + )
2010-06-07 16:11:08 +03:00
{
2010-12-06 01:10:02 +02:00
if ( h - > getPower ( targetstack ) > h - > getPower ( i - > first ) )
2010-06-25 11:19:39 +03:00
{
targetstack = ( i - > first ) ;
}
2010-06-07 16:11:08 +03:00
}
2010-12-06 01:10:02 +02:00
TQuantity countToTake = h - > getStackCount ( targetstack ) * 0.5 ;
2011-12-14 00:23:17 +03:00
vstd : : amax ( countToTake , 1 ) ;
2010-12-06 01:10:02 +02:00
2010-06-25 11:19:39 +03:00
2010-06-07 16:11:08 +03:00
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , 168 ) ;
2010-12-06 01:10:02 +02:00
iw . components . push_back ( Component ( CStackBasicDescriptor ( h - > getCreature ( targetstack ) , countToTake ) ) ) ;
2010-06-07 16:11:08 +03:00
cb - > showInfoDialog ( & iw ) ;
2010-12-06 01:10:02 +02:00
cb - > changeStackCount ( StackLocation ( h , targetstack ) , - countToTake ) ;
2010-06-07 16:11:08 +03:00
}
}
}
}
2008-12-27 03:01:59 +02:00
else
tlog2 < < " Cannot find corresponding exit monolith for " < < id < < std : : endl ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : SUBTERRANEAN_GATE : //find nearest subterranean gate on the other level
2008-12-27 03:01:59 +02:00
{
2011-09-19 23:50:25 +03:00
destinationid = getMatchingGate ( id ) ;
if ( destinationid < 0 ) //no exit
2010-02-10 05:35:48 +02:00
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
2010-06-07 16:11:08 +03:00
iw . text . addTxt ( MetaString : : ADVOB_TXT , 153 ) ; //Just inside the entrance you find a large pile of rubble blocking the tunnel. You leave discouraged.
2010-02-10 05:35:48 +02:00
cb - > sendAndApply ( & iw ) ;
}
2008-12-27 03:01:59 +02:00
break ;
}
}
if ( destinationid < 0 )
{
2009-09-07 05:29:44 +03:00
tlog2 < < " Cannot find exit... (obj at " < < pos < < " ) :( \n " ;
2008-12-27 03:01:59 +02:00
return ;
}
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : WHIRLPOOL )
2010-08-29 13:13:56 +03:00
{
std : : set < int3 > tiles = cb - > getObj ( destinationid ) - > getBlockedPos ( ) ;
std : : set < int3 > : : iterator it = tiles . begin ( ) ;
std : : advance ( it , rand ( ) % tiles . size ( ) ) ; //picking random element of set is tiring
cb - > moveHero ( h - > id , * it + int3 ( 1 , 0 , 0 ) , true ) ;
}
2010-06-25 11:19:39 +03:00
else
cb - > moveHero ( h - > id , CGHeroInstance : : convertPosition ( cb - > getObj ( destinationid ) - > pos , true ) - getVisitableOffset ( ) , true ) ;
2008-12-27 03:01:59 +02:00
}
void CGTeleport : : initObj ( )
{
2009-09-07 05:29:44 +03:00
int si = subID ;
2010-06-07 16:11:08 +03:00
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : SUBTERRANEAN_GATE : //ignore subterranean gates subid
case Obj : : WHIRLPOOL :
2010-06-07 16:11:08 +03:00
{
si = 0 ;
break ;
}
2012-09-23 21:01:04 +03:00
default :
break ;
2010-06-07 16:11:08 +03:00
}
2009-09-07 05:29:44 +03:00
objs [ ID ] [ si ] . push_back ( id ) ;
}
void CGTeleport : : postInit ( ) //matches subterranean gates into pairs
{
//split on underground and surface gates
std : : vector < const CGObjectInstance * > gatesSplit [ 2 ] ; //surface and underground gates
2012-09-23 21:01:04 +03:00
for ( size_t i = 0 ; i < objs [ Obj : : SUBTERRANEAN_GATE ] [ 0 ] . size ( ) ; i + + )
2009-09-07 05:29:44 +03:00
{
2012-09-23 21:01:04 +03:00
const CGObjectInstance * hlp = cb - > getObj ( objs [ Obj : : SUBTERRANEAN_GATE ] [ 0 ] [ i ] ) ;
2009-09-07 05:29:44 +03:00
gatesSplit [ hlp - > pos . z ] . push_back ( hlp ) ;
}
//sort by position
std : : sort ( gatesSplit [ 0 ] . begin ( ) , gatesSplit [ 0 ] . end ( ) , boost : : bind ( & CGObjectInstance : : pos , _1 ) < boost : : bind ( & CGObjectInstance : : pos , _2 ) ) ;
for ( size_t i = 0 ; i < gatesSplit [ 0 ] . size ( ) ; i + + )
{
const CGObjectInstance * cur = gatesSplit [ 0 ] [ i ] ;
//find nearest underground exit
std : : pair < int , double > best ( - 1 , 150000 ) ; //pair<pos_in_vector, distance>
for ( int j = 0 ; j < gatesSplit [ 1 ] . size ( ) ; j + + )
{
const CGObjectInstance * checked = gatesSplit [ 1 ] [ j ] ;
if ( ! checked )
continue ;
double hlp = checked - > pos . dist2d ( cur - > pos ) ;
if ( hlp < best . second )
{
best . first = j ;
best . second = hlp ;
}
}
2010-02-10 05:35:48 +02:00
if ( best . first > = 0 ) //found pair
{
gates . push_back ( std : : pair < int , int > ( cur - > id , gatesSplit [ 1 ] [ best . first ] - > id ) ) ;
gatesSplit [ 1 ] [ best . first ] = NULL ;
}
else
{
gates . push_back ( std : : pair < int , int > ( cur - > id , - 1 ) ) ;
}
2009-09-07 05:29:44 +03:00
}
objs . erase ( 103 ) ;
2008-12-27 03:01:59 +02:00
}
2011-09-19 23:50:25 +03:00
int CGTeleport : : getMatchingGate ( int id )
{
for ( int i = 0 ; i < gates . size ( ) ; i + + )
{
if ( gates [ i ] . first = = id )
return gates [ i ] . second ;
if ( gates [ i ] . second = = id )
return gates [ i ] . first ;
}
return - 1 ;
}
2008-12-27 03:01:59 +02:00
void CGArtifact : : initObj ( )
{
blockVisit = true ;
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : ARTIFACT )
2010-12-26 16:34:11 +02:00
{
2010-06-26 19:02:10 +03:00
hoverName = VLC - > arth - > artifacts [ subID ] - > Name ( ) ;
2010-12-26 16:34:11 +02:00
if ( ! storedArtifact - > artType )
storedArtifact - > setType ( VLC - > arth - > artifacts [ subID ] ) ;
}
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : SPELL_SCROLL )
2010-11-10 02:06:25 +02:00
subID = 1 ;
2010-12-26 16:34:11 +02:00
assert ( storedArtifact - > artType ) ;
2011-07-13 21:39:02 +03:00
assert ( storedArtifact - > getParentNodes ( ) . size ( ) ) ;
2008-12-27 03:01:59 +02:00
}
void CGArtifact : : onHeroVisit ( const CGHeroInstance * h ) const
2009-02-03 07:28:05 +02:00
{
2010-05-02 21:20:26 +03:00
if ( ! stacksCount ( ) )
2009-02-03 07:28:05 +02:00
{
2010-11-10 02:06:25 +02:00
InfoWindow iw ;
iw . player = h - > tempOwner ;
switch ( ID )
2009-03-19 16:17:19 +02:00
{
2012-09-23 21:01:04 +03:00
case Obj : : ARTIFACT :
2010-11-10 02:06:25 +02:00
{
2011-03-12 23:55:31 +02:00
iw . soundID = soundBase : : treasure ; //play sound only for non-scroll arts
2012-08-31 20:39:11 +03:00
iw . components . push_back ( Component ( Component : : ARTIFACT , subID , 0 , 0 ) ) ;
2010-11-10 02:06:25 +02:00
if ( message . length ( ) )
iw . text < < message ;
else
2012-12-06 22:03:47 +03:00
iw . text < < std : : pair < ui8 , ui32 > ( MetaString : : ART_EVNTS , subID ) ;
2010-11-10 02:06:25 +02:00
}
break ;
2012-09-23 21:01:04 +03:00
case Obj : : SPELL_SCROLL :
2011-03-12 23:55:31 +02:00
{
int spellID = storedArtifact - > getGivenSpellID ( ) ;
iw . components . push_back ( Component ( Component : : SPELL , spellID , 0 , 0 ) ) ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , 135 ) ;
iw . text . addReplacement ( MetaString : : SPELL_NAME , spellID ) ;
}
2010-11-10 02:06:25 +02:00
break ;
2009-03-19 16:17:19 +02:00
}
2010-11-10 02:06:25 +02:00
cb - > showInfoDialog ( & iw ) ;
2009-02-03 07:28:05 +02:00
pick ( h ) ;
}
else
{
if ( message . size ( ) )
{
2009-04-11 04:32:50 +03:00
BlockingDialog ynd ( true , false ) ;
2009-02-03 07:28:05 +02:00
ynd . player = h - > getOwner ( ) ;
ynd . text < < message ;
2009-04-11 04:32:50 +03:00
cb - > showBlockingDialog ( & ynd , boost : : bind ( & CGArtifact : : fightForArt , this , _1 , h ) ) ;
2009-02-03 07:28:05 +02:00
}
else
{
fightForArt ( 0 , h ) ;
}
}
}
void CGArtifact : : pick ( const CGHeroInstance * h ) const
2008-12-27 03:01:59 +02:00
{
2010-12-26 16:34:11 +02:00
cb - > giveHeroArtifact ( h , storedArtifact , - 2 ) ;
2009-02-01 16:11:41 +02:00
cb - > removeObject ( id ) ;
2008-12-27 03:01:59 +02:00
}
2009-04-11 04:32:50 +03:00
void CGArtifact : : fightForArt ( ui32 agreed , const CGHeroInstance * h ) const
2009-02-03 07:28:05 +02:00
{
2009-04-11 04:32:50 +03:00
if ( agreed )
2009-09-13 01:17:23 +03:00
cb - > startBattleI ( h , this , boost : : bind ( & CGArtifact : : endBattle , this , _1 , h ) ) ;
2009-02-03 07:28:05 +02:00
}
void CGArtifact : : endBattle ( BattleResult * result , const CGHeroInstance * h ) const
{
if ( result - > winner = = 0 ) //attacker won
pick ( h ) ;
}
2008-12-27 03:01:59 +02:00
void CGPickable : : initObj ( )
{
blockVisit = true ;
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : CAMPFIRE :
2008-12-27 03:01:59 +02:00
val2 = ( ran ( ) % 3 ) + 4 ; //4 - 6
val1 = val2 * 100 ;
type = ran ( ) % 6 ; //given resource
break ;
2012-09-23 21:01:04 +03:00
case Obj : : FLOTSAM :
2009-07-19 06:10:24 +03:00
switch ( type = ran ( ) % 4 )
{
case 0 :
val1 = val2 = 0 ;
break ;
case 1 :
val1 = 5 ;
val2 = 0 ;
break ;
case 2 :
val1 = 5 ;
val2 = 200 ;
break ;
case 3 :
val1 = 10 ;
val2 = 500 ;
break ;
}
break ;
2012-09-23 21:01:04 +03:00
case Obj : : SEA_CHEST :
2009-07-19 06:10:24 +03:00
{
int hlp = ran ( ) % 100 ;
if ( hlp < 20 )
{
val1 = 0 ;
type = 0 ;
}
else if ( hlp < 90 )
{
val1 = 1500 ;
type = 2 ;
}
else
{
val1 = 1000 ;
2009-10-24 22:21:32 +03:00
val2 = cb - > getRandomArt ( CArtifact : : ART_TREASURE ) ;
2009-07-19 06:10:24 +03:00
type = 1 ;
}
}
break ;
2012-09-23 21:01:04 +03:00
case Obj : : SHIPWRECK_SURVIVOR :
2009-07-19 06:10:24 +03:00
{
int hlp = ran ( ) % 100 ;
if ( hlp < 55 )
2009-10-24 22:21:32 +03:00
val1 = cb - > getRandomArt ( CArtifact : : ART_TREASURE ) ;
2009-07-19 06:10:24 +03:00
else if ( hlp < 75 )
2009-10-24 22:21:32 +03:00
val1 = cb - > getRandomArt ( CArtifact : : ART_MINOR ) ;
2009-07-19 06:10:24 +03:00
else if ( hlp < 95 )
2009-10-24 22:21:32 +03:00
val1 = cb - > getRandomArt ( CArtifact : : ART_MAJOR ) ;
2009-07-19 06:10:24 +03:00
else
2009-10-24 22:21:32 +03:00
val1 = cb - > getRandomArt ( CArtifact : : ART_RELIC ) ;
2009-07-19 06:10:24 +03:00
}
break ;
2012-09-23 21:01:04 +03:00
case Obj : : TREASURE_CHEST :
2008-12-27 03:01:59 +02:00
{
int hlp = ran ( ) % 100 ;
if ( hlp > = 95 )
{
type = 1 ;
2009-10-24 22:21:32 +03:00
val1 = cb - > getRandomArt ( CArtifact : : ART_TREASURE ) ;
2008-12-27 03:01:59 +02:00
return ;
}
else if ( hlp > = 65 )
{
val1 = 2000 ;
}
else if ( hlp > = 33 )
{
val1 = 1500 ;
}
else
{
val1 = 1000 ;
}
val2 = val1 - 500 ;
type = 0 ;
break ;
}
}
}
void CGPickable : : onHeroVisit ( const CGHeroInstance * h ) const
{
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : CAMPFIRE :
2008-12-27 03:01:59 +02:00
{
cb - > giveResource ( h - > tempOwner , type , val2 ) ; //non-gold resource
2012-08-31 20:39:11 +03:00
cb - > giveResource ( h - > tempOwner , Res : : GOLD , val1 ) ; //gold
2008-12-27 03:01:59 +02:00
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = soundBase : : experience ;
2008-12-27 03:01:59 +02:00
iw . player = h - > tempOwner ;
2012-08-31 20:39:11 +03:00
iw . components . push_back ( Component ( Component : : RESOURCE , Res : : GOLD , val1 , 0 ) ) ;
iw . components . push_back ( Component ( Component : : RESOURCE , type , val2 , 0 ) ) ;
2008-12-27 03:01:59 +02:00
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 23 ) ;
cb - > showInfoDialog ( & iw ) ;
break ;
}
2012-09-23 21:01:04 +03:00
case Obj : : FLOTSAM :
2009-07-19 06:10:24 +03:00
{
cb - > giveResource ( h - > tempOwner , 0 , val1 ) ; //wood
cb - > giveResource ( h - > tempOwner , 6 , val2 ) ; //gold
InfoWindow iw ;
iw . soundID = soundBase : : GENIE ;
iw . player = h - > tempOwner ;
if ( val1 )
2012-08-31 20:39:11 +03:00
iw . components . push_back ( Component ( Component : : RESOURCE , Res : : WOOD , val1 , 0 ) ) ;
2009-07-19 06:10:24 +03:00
if ( val2 )
2012-08-31 20:39:11 +03:00
iw . components . push_back ( Component ( Component : : RESOURCE , Res : : GOLD , val2 , 0 ) ) ;
2009-07-19 06:10:24 +03:00
iw . text . addTxt ( MetaString : : ADVOB_TXT , 51 + type ) ;
cb - > showInfoDialog ( & iw ) ;
break ;
}
2012-09-23 21:01:04 +03:00
case Obj : : SEA_CHEST :
2009-07-19 06:10:24 +03:00
{
InfoWindow iw ;
iw . soundID = soundBase : : chest ;
iw . player = h - > tempOwner ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , 116 + type ) ;
if ( val1 ) //there is gold
{
2012-08-31 20:39:11 +03:00
iw . components . push_back ( Component ( Component : : RESOURCE , Res : : GOLD , val1 , 0 ) ) ;
2009-07-19 06:10:24 +03:00
cb - > giveResource ( h - > tempOwner , 6 , val1 ) ;
}
if ( type = = 1 ) //art
{
//TODO: what if no space in backpack?
2012-08-31 20:39:11 +03:00
iw . components . push_back ( Component ( Component : : ARTIFACT , val2 , 1 , 0 ) ) ;
2009-07-19 06:10:24 +03:00
iw . text . addReplacement ( MetaString : : ART_NAMES , val2 ) ;
2010-12-26 16:34:11 +02:00
cb - > giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ val2 ] , - 2 ) ;
2009-07-19 06:10:24 +03:00
}
cb - > showInfoDialog ( & iw ) ;
break ;
}
2012-09-23 21:01:04 +03:00
case Obj : : SHIPWRECK_SURVIVOR :
2009-07-19 06:10:24 +03:00
{
//TODO: what if no space in backpack?
InfoWindow iw ;
iw . soundID = soundBase : : experience ;
iw . player = h - > tempOwner ;
2012-08-31 20:39:11 +03:00
iw . components . push_back ( Component ( Component : : ARTIFACT , val1 , 1 , 0 ) ) ;
2009-07-19 06:10:24 +03:00
iw . text . addTxt ( MetaString : : ADVOB_TXT , 125 ) ;
iw . text . addReplacement ( MetaString : : ART_NAMES , val1 ) ;
2010-12-26 16:34:11 +02:00
cb - > giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ val1 ] , - 2 ) ;
2009-07-19 06:10:24 +03:00
cb - > showInfoDialog ( & iw ) ;
break ;
}
2012-09-23 21:01:04 +03:00
case Obj : : TREASURE_CHEST :
2008-12-27 03:01:59 +02:00
{
if ( subID ) //not OH3 treasure chest
{
tlog2 < < " Not supported WoG treasure chest! \n " ;
2012-07-19 21:52:44 +03:00
return ;
2008-12-27 03:01:59 +02:00
}
if ( type ) //there is an artifact
{
2010-12-26 16:34:11 +02:00
cb - > giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ val1 ] , - 2 ) ;
2008-12-27 03:01:59 +02:00
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = soundBase : : treasure ;
2008-12-27 03:01:59 +02:00
iw . player = h - > tempOwner ;
2012-08-31 20:39:11 +03:00
iw . components . push_back ( Component ( Component : : ARTIFACT , val1 , 1 , 0 ) ) ;
2008-12-27 03:01:59 +02:00
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 145 ) ;
2009-07-09 22:15:22 +03:00
iw . text . addReplacement ( MetaString : : ART_NAMES , val1 ) ;
2008-12-27 03:01:59 +02:00
cb - > showInfoDialog ( & iw ) ;
break ;
}
else
{
2009-04-11 04:32:50 +03:00
BlockingDialog sd ( false , true ) ;
2008-12-27 03:01:59 +02:00
sd . player = h - > tempOwner ;
sd . text < < std : : pair < ui8 , ui32 > ( 11 , 146 ) ;
2012-08-31 20:39:11 +03:00
sd . components . push_back ( Component ( Component : : RESOURCE , Res : : GOLD , val1 , 0 ) ) ;
2012-09-23 21:01:04 +03:00
TExpType expVal = h - > calculateXp ( val2 ) ;
2012-08-31 20:39:11 +03:00
sd . components . push_back ( Component ( Component : : EXPERIENCE , 0 , expVal , 0 ) ) ;
2009-04-22 21:48:56 +03:00
sd . soundID = soundBase : : chest ;
2008-12-27 03:01:59 +02:00
boost : : function < void ( ui32 ) > fun = boost : : bind ( & CGPickable : : chosen , this , _1 , h - > id ) ;
2009-04-11 04:32:50 +03:00
cb - > showBlockingDialog ( & sd , fun ) ;
2008-12-27 03:01:59 +02:00
return ;
}
}
}
cb - > removeObject ( id ) ;
}
void CGPickable : : chosen ( int which , int heroID ) const
{
2010-07-26 01:47:59 +03:00
const CGHeroInstance * h = cb - > getHero ( heroID ) ;
2008-12-27 03:01:59 +02:00
switch ( which )
{
2009-04-11 04:32:50 +03:00
case 1 : //player pick gold
2008-12-27 03:01:59 +02:00
cb - > giveResource ( cb - > getOwner ( heroID ) , 6 , val1 ) ;
break ;
2009-04-11 04:32:50 +03:00
case 2 : //player pick exp
2011-02-12 20:48:11 +02:00
cb - > changePrimSkill ( heroID , 4 , h - > calculateXp ( val2 ) ) ;
2008-12-27 03:01:59 +02:00
break ;
default :
2012-04-22 10:32:45 +03:00
throw std : : runtime_error ( " Unhandled treasure choice " ) ;
2008-12-27 03:01:59 +02:00
}
2009-01-11 00:08:18 +02:00
cb - > removeObject ( id ) ;
}
2009-12-31 13:43:37 +02:00
bool CQuest : : checkQuest ( const CGHeroInstance * h ) const
{
switch ( missionType )
{
case MISSION_NONE :
return true ;
case MISSION_LEVEL :
if ( m13489val < = h - > level )
return true ;
return false ;
case MISSION_PRIMARY_STAT :
for ( int i = 0 ; i < 4 ; + + i )
{
2010-05-02 21:20:26 +03:00
if ( h - > getPrimSkillLevel ( i ) < m2stats [ i ] )
2009-12-31 13:43:37 +02:00
return false ;
}
return true ;
case MISSION_KILL_HERO :
case MISSION_KILL_CREATURE :
2011-02-11 14:27:38 +02:00
if ( ! h - > cb - > getObjByQuestIdentifier ( m13489val ) )
2010-02-07 18:05:27 +02:00
return true ;
return false ;
2009-12-31 13:43:37 +02:00
case MISSION_ART :
for ( int i = 0 ; i < m5arts . size ( ) ; + + i )
{
2010-02-04 22:34:20 +02:00
if ( h - > hasArt ( m5arts [ i ] ) )
2009-12-31 13:43:37 +02:00
continue ;
return false ; //if the artifact was not found
}
return true ;
case MISSION_ARMY :
2010-01-01 14:15:20 +02:00
{
2010-11-22 02:34:46 +02:00
std : : vector < CStackBasicDescriptor > : : const_iterator cre ;
TSlots : : const_iterator it ;
2010-01-01 14:15:20 +02:00
ui32 count ;
for ( cre = m6creatures . begin ( ) ; cre ! = m6creatures . end ( ) ; + + cre )
{
2010-05-02 21:20:26 +03:00
for ( count = 0 , it = h - > Slots ( ) . begin ( ) ; it ! = h - > Slots ( ) . end ( ) ; + + it )
2010-01-01 14:15:20 +02:00
{
2010-11-22 02:34:46 +02:00
if ( it - > second - > type = = cre - > type )
count + = it - > second - > count ;
2010-01-01 14:15:20 +02:00
}
2010-11-22 02:34:46 +02:00
if ( count < cre - > count ) //not enough creatures of this kind
2010-01-01 14:15:20 +02:00
return false ;
}
}
return true ;
2009-12-31 13:43:37 +02:00
case MISSION_RESOURCES :
for ( int i = 0 ; i < 7 ; + + i ) //including Mithril ?
{ //Quest has no direct access to callback
2012-07-19 21:52:44 +03:00
if ( h - > cb - > getResource ( h - > tempOwner , i ) < m7resources [ i ] )
2009-12-31 13:43:37 +02:00
return false ;
}
return true ;
case MISSION_HERO :
2010-02-04 22:34:20 +02:00
if ( m13489val = = h - > type - > ID )
2009-12-31 13:43:37 +02:00
return true ;
return false ;
case MISSION_PLAYER :
if ( m13489val = = h - > getOwner ( ) )
return true ;
return false ;
default :
return false ;
}
}
2012-07-08 09:33:41 +03:00
void CQuest : : getVisitText ( MetaString & iwText , std : : vector < Component > & components , bool isCustom , bool firstVisit , const CGHeroInstance * h ) const
{
std : : string text ;
bool failRequirements = ( h ? ! checkQuest ( h ) : true ) ;
2012-07-19 21:52:44 +03:00
if ( firstVisit )
2012-07-08 09:33:41 +03:00
{
isCustom = isCustomFirst ;
iwText < < firstVisitText ;
}
else if ( failRequirements )
{
isCustom = isCustomNext ;
iwText < < nextVisitText ;
}
switch ( missionType )
{
case MISSION_LEVEL :
components . push_back ( Component ( Component : : EXPERIENCE , 1 , m13489val , 0 ) ) ;
if ( ! isCustom )
iwText . addReplacement ( m13489val ) ;
break ;
case MISSION_PRIMARY_STAT :
{
MetaString loot ;
for ( int i = 0 ; i < 4 ; + + i )
{
if ( m2stats [ i ] )
{
components . push_back ( Component ( Component : : PRIM_SKILL , i , m2stats [ i ] , 0 ) ) ;
loot < < " %d %s " ;
loot . addReplacement ( m2stats [ i ] ) ;
loot . addReplacement ( VLC - > generaltexth - > primarySkillNames [ i ] ) ;
2012-07-19 21:52:44 +03:00
}
2012-07-08 09:33:41 +03:00
}
if ( ! isCustom )
iwText . addReplacement ( loot . buildList ( ) ) ;
}
break ;
case MISSION_KILL_HERO :
components . push_back ( Component ( Component : : HERO , heroPortrait , 0 , 0 ) ) ;
if ( ! isCustom )
addReplacements ( iwText , text ) ;
break ;
case MISSION_HERO :
components . push_back ( Component ( Component : : HERO , m13489val , 0 , 0 ) ) ;
if ( ! isCustom )
iwText . addReplacement ( VLC - > heroh - > heroes [ m13489val ] - > name ) ;
break ;
case MISSION_KILL_CREATURE :
{
components . push_back ( Component ( stackToKill ) ) ;
if ( ! isCustom )
addReplacements ( iwText , text ) ;
}
break ;
case MISSION_ART :
{
MetaString loot ;
for ( std : : vector < ui16 > : : const_iterator it = m5arts . begin ( ) ; it ! = m5arts . end ( ) ; + + it )
{
components . push_back ( Component ( Component : : ARTIFACT , * it , 0 , 0 ) ) ;
loot < < " %s " ;
loot . addReplacement ( MetaString : : ART_NAMES , * it ) ;
}
if ( ! isCustom )
iwText . addReplacement ( loot . buildList ( ) ) ;
}
break ;
case MISSION_ARMY :
{
MetaString loot ;
for ( std : : vector < CStackBasicDescriptor > : : const_iterator it = m6creatures . begin ( ) ; it ! = m6creatures . end ( ) ; + + it )
{
components . push_back ( Component ( * it ) ) ;
loot < < " %s " ;
loot . addReplacement ( * it ) ;
}
if ( ! isCustom )
iwText . addReplacement ( loot . buildList ( ) ) ;
}
break ;
case MISSION_RESOURCES :
{
MetaString loot ;
for ( int i = 0 ; i < 7 ; + + i )
{
if ( m7resources [ i ] )
{
components . push_back ( Component ( Component : : RESOURCE , i , m7resources [ i ] , 0 ) ) ;
loot < < " %d %s " ;
loot . addReplacement ( m7resources [ i ] ) ;
loot . addReplacement ( MetaString : : RES_NAMES , i ) ;
}
}
if ( ! isCustom )
iwText . addReplacement ( loot . buildList ( ) ) ;
}
break ;
case MISSION_PLAYER :
components . push_back ( Component ( Component : : FLAG , m13489val , 0 , 0 ) ) ;
if ( ! isCustom )
iwText . addReplacement ( VLC - > generaltexth - > colors [ m13489val ] ) ;
break ;
}
}
void CQuest : : getRolloverText ( MetaString & ms , bool onHover ) const
{
if ( onHover )
ms < < " \n \n " ;
ms < < VLC - > generaltexth - > quests [ missionType - 1 ] [ onHover ? 3 : 4 ] [ textOption ] ;
switch ( missionType )
{
case MISSION_LEVEL :
ms . addReplacement ( m13489val ) ;
break ;
case MISSION_PRIMARY_STAT :
{
MetaString loot ;
for ( int i = 0 ; i < 4 ; + + i )
{
if ( m2stats [ i ] )
{
loot < < " %d %s " ;
loot . addReplacement ( m2stats [ i ] ) ;
loot . addReplacement ( VLC - > generaltexth - > primarySkillNames [ i ] ) ;
2012-07-19 21:52:44 +03:00
}
2012-07-08 09:33:41 +03:00
}
ms . addReplacement ( loot . buildList ( ) ) ;
}
break ;
case MISSION_KILL_HERO :
ms . addReplacement ( heroName ) ;
break ;
case MISSION_KILL_CREATURE :
ms . addReplacement ( stackToKill ) ;
break ;
case MISSION_ART :
{
MetaString loot ;
for ( std : : vector < ui16 > : : const_iterator it = m5arts . begin ( ) ; it ! = m5arts . end ( ) ; + + it )
{
loot < < " %s " ;
loot . addReplacement ( MetaString : : ART_NAMES , * it ) ;
}
ms . addReplacement ( loot . buildList ( ) ) ;
}
break ;
case MISSION_ARMY :
{
MetaString loot ;
for ( std : : vector < CStackBasicDescriptor > : : const_iterator it = m6creatures . begin ( ) ; it ! = m6creatures . end ( ) ; + + it )
{
loot < < " %s " ;
loot . addReplacement ( * it ) ;
}
ms . addReplacement ( loot . buildList ( ) ) ;
}
break ;
case MISSION_RESOURCES :
{
MetaString loot ;
for ( int i = 0 ; i < 7 ; + + i )
{
if ( m7resources [ i ] )
{
loot < < " %d %s " ;
loot . addReplacement ( m7resources [ i ] ) ;
loot . addReplacement ( MetaString : : RES_NAMES , i ) ;
}
}
ms . addReplacement ( loot . buildList ( ) ) ;
}
break ;
case MISSION_HERO :
ms . addReplacement ( VLC - > heroh - > heroes [ m13489val ] - > name ) ;
break ;
case MISSION_PLAYER :
ms . addReplacement ( VLC - > generaltexth - > colors [ m13489val ] ) ;
break ;
default :
break ;
}
}
void CQuest : : getCompletionText ( MetaString & iwText , std : : vector < Component > & components , bool isCustom , const CGHeroInstance * h ) const
{
iwText < < completedText ;
switch ( missionType )
{
case CQuest : : MISSION_LEVEL :
if ( ! isCustomComplete )
iwText . addReplacement ( m13489val ) ;
break ;
case CQuest : : MISSION_PRIMARY_STAT :
if ( vstd : : contains ( completedText , ' % ' ) ) //there's one case when there's nothing to replace
{
MetaString loot ;
for ( int i = 0 ; i < 4 ; + + i )
{
if ( m2stats [ i ] )
{
loot < < " %d %s " ;
loot . addReplacement ( m2stats [ i ] ) ;
loot . addReplacement ( VLC - > generaltexth - > primarySkillNames [ i ] ) ;
}
}
if ( ! isCustomComplete )
iwText . addReplacement ( loot . buildList ( ) ) ;
}
break ;
case CQuest : : MISSION_ART :
{
MetaString loot ;
for ( std : : vector < ui16 > : : const_iterator it = m5arts . begin ( ) ; it ! = m5arts . end ( ) ; + + it )
{
loot < < " %s " ;
loot . addReplacement ( MetaString : : ART_NAMES , * it ) ;
}
if ( ! isCustomComplete )
iwText . addReplacement ( loot . buildList ( ) ) ;
}
break ;
case CQuest : : MISSION_ARMY :
{
MetaString loot ;
for ( std : : vector < CStackBasicDescriptor > : : const_iterator it = m6creatures . begin ( ) ; it ! = m6creatures . end ( ) ; + + it )
{
loot < < " %s " ;
loot . addReplacement ( * it ) ;
}
if ( ! isCustomComplete )
iwText . addReplacement ( loot . buildList ( ) ) ;
}
break ;
case CQuest : : MISSION_RESOURCES :
{
MetaString loot ;
for ( int i = 0 ; i < 7 ; + + i )
{
if ( m7resources [ i ] )
{
loot < < " %d %s " ;
loot . addReplacement ( m7resources [ i ] ) ;
loot . addReplacement ( MetaString : : RES_NAMES , i ) ;
}
}
if ( ! isCustomComplete )
iwText . addReplacement ( loot . buildList ( ) ) ;
}
break ;
case MISSION_KILL_HERO :
case MISSION_KILL_CREATURE :
if ( ! isCustomComplete )
addReplacements ( iwText , completedText ) ;
break ;
case MISSION_HERO :
if ( ! isCustomComplete )
iwText . addReplacement ( VLC - > heroh - > heroes [ m13489val ] - > name ) ;
break ;
case MISSION_PLAYER :
if ( ! isCustomComplete )
iwText . addReplacement ( VLC - > generaltexth - > colors [ m13489val ] ) ;
break ;
}
}
2012-07-19 12:10:55 +03:00
void CGSeerHut : : setObjToKill ( )
{
2012-10-03 17:49:29 +03:00
if ( quest - > missionType = = CQuest : : MISSION_KILL_CREATURE )
2012-07-19 12:10:55 +03:00
{
2012-10-03 17:49:29 +03:00
quest - > stackToKill = getCreatureToKill ( false ) - > getStack ( 0 ) ; //FIXME: stacks tend to dissapear (desync?) on server :?
assert ( quest - > stackToKill . type ) ;
quest - > stackToKill . count = 0 ; //no count in info window
quest - > stackDirection = checkDirection ( ) ;
2012-07-19 12:10:55 +03:00
}
2012-10-03 17:49:29 +03:00
else if ( quest - > missionType = = CQuest : : MISSION_KILL_HERO )
2012-07-19 12:10:55 +03:00
{
2012-10-03 17:49:29 +03:00
quest - > heroName = getHeroToKill ( false ) - > name ;
quest - > heroPortrait = getHeroToKill ( false ) - > portrait ;
2012-07-19 12:10:55 +03:00
}
}
2012-07-08 09:33:41 +03:00
2010-01-01 14:15:20 +02:00
void CGSeerHut : : initObj ( )
{
2010-01-27 07:11:31 +02:00
seerName = VLC - > generaltexth - > seerNames [ ran ( ) % VLC - > generaltexth - > seerNames . size ( ) ] ;
2012-10-03 17:49:29 +03:00
quest - > textOption = ran ( ) % 3 ;
quest - > progress = 0 ;
if ( quest - > missionType )
2010-02-02 19:05:03 +02:00
{
2012-10-03 17:49:29 +03:00
if ( ! quest - > isCustomFirst )
quest - > firstVisitText = VLC - > generaltexth - > quests [ quest - > missionType - 1 ] [ 0 ] [ quest - > textOption ] ;
if ( ! quest - > isCustomNext )
quest - > nextVisitText = VLC - > generaltexth - > quests [ quest - > missionType - 1 ] [ 1 ] [ quest - > textOption ] ;
if ( ! quest - > isCustomComplete )
quest - > completedText = VLC - > generaltexth - > quests [ quest - > missionType - 1 ] [ 2 ] [ quest - > textOption ] ;
2010-02-02 19:05:03 +02:00
}
else
2012-10-03 17:49:29 +03:00
quest - > firstVisitText = VLC - > generaltexth - > seerEmpty [ quest - > textOption ] ;
2010-06-06 12:50:37 +03:00
2010-01-01 14:15:20 +02:00
}
2012-07-08 19:36:20 +03:00
void CGSeerHut : : getRolloverText ( MetaString & text , bool onHover ) const
{
2012-10-03 17:49:29 +03:00
quest - > getRolloverText ( text , onHover ) ; //TODO: simplify?
2012-07-08 19:36:20 +03:00
if ( ! onHover )
text . addReplacement ( seerName ) ;
}
2009-12-30 22:49:10 +02:00
const std : : string & CGSeerHut : : getHoverText ( ) const
{
2010-02-07 18:05:27 +02:00
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : SEER_HUT :
2012-10-03 17:49:29 +03:00
if ( quest - > progress )
2012-09-23 21:01:04 +03:00
{
hoverName = VLC - > generaltexth - > allTexts [ 347 ] ;
boost : : algorithm : : replace_first ( hoverName , " %s " , seerName ) ;
}
else //just seer hut
hoverName = VLC - > generaltexth - > names [ ID ] ;
break ;
case Obj : : QUEST_GUARD :
hoverName = VLC - > generaltexth - > names [ ID ] ;
break ;
default :
tlog5 < < " unrecognized quest object \n " ;
2010-02-07 18:05:27 +02:00
}
2012-10-03 17:49:29 +03:00
if ( quest - > progress & quest - > missionType ) //rollover when the quest is active
2010-02-07 18:05:27 +02:00
{
MetaString ms ;
2012-07-19 21:52:44 +03:00
getRolloverText ( ms , true ) ;
2012-07-08 09:33:41 +03:00
hoverName + = ms . toString ( ) ;
}
return hoverName ;
}
void CQuest : : addReplacements ( MetaString & out , const std : : string & base ) const
{
switch ( missionType )
{
case MISSION_KILL_CREATURE :
out . addReplacement ( stackToKill ) ;
if ( std : : count ( base . begin ( ) , base . end ( ) , ' % ' ) = = 2 ) //say where is placed monster
2010-02-07 18:05:27 +02:00
{
2012-07-08 09:33:41 +03:00
out . addReplacement ( VLC - > generaltexth - > arraytxt [ 147 + stackDirection ] ) ;
2010-02-07 18:05:27 +02:00
}
2012-07-08 09:33:41 +03:00
break ;
case MISSION_KILL_HERO :
out . addReplacement ( heroName ) ;
break ;
}
}
2012-09-16 16:34:01 +03:00
bool IQuestObject : : checkQuest ( const CGHeroInstance * h ) const
{
2012-10-03 17:49:29 +03:00
return quest - > checkQuest ( h ) ;
2012-09-16 16:34:01 +03:00
}
void IQuestObject : : getVisitText ( MetaString & text , std : : vector < Component > & components , bool isCustom , bool FirstVisit , const CGHeroInstance * h ) const
{
2012-10-03 17:49:29 +03:00
quest - > getVisitText ( text , components , isCustom , FirstVisit , h ) ;
2012-09-16 16:34:01 +03:00
}
2012-07-08 09:33:41 +03:00
void CGSeerHut : : getCompletionText ( MetaString & text , std : : vector < Component > & components , bool isCustom , const CGHeroInstance * h ) const
{
2012-10-03 17:49:29 +03:00
quest - > getCompletionText ( text , components , isCustom , h ) ;
2012-07-08 09:33:41 +03:00
switch ( rewardType )
{
case 1 : components . push_back ( Component ( Component : : EXPERIENCE , 0 , rVal * ( 100 + h - > getSecSkillLevel ( CGHeroInstance : : LEARNING ) * 5 ) / 100.0 , 0 ) ) ;
break ;
case 2 : components . push_back ( Component ( Component : : PRIM_SKILL , 5 , rVal , 0 ) ) ;
break ;
case 3 : components . push_back ( Component ( Component : : MORALE , 0 , rVal , 0 ) ) ;
break ;
case 4 : components . push_back ( Component ( Component : : LUCK , 0 , rVal , 0 ) ) ;
break ;
case 5 : components . push_back ( Component ( Component : : RESOURCE , rID , rVal , 0 ) ) ;
break ;
case 6 : components . push_back ( Component ( Component : : PRIM_SKILL , rID , rVal , 0 ) ) ;
break ;
case 7 : components . push_back ( Component ( Component : : SEC_SKILL , rID , rVal , 0 ) ) ;
break ;
case 8 : components . push_back ( Component ( Component : : ARTIFACT , rID , 0 , 0 ) ) ;
break ;
case 9 : components . push_back ( Component ( Component : : SPELL , rID , 0 , 0 ) ) ;
break ;
case 10 : components . push_back ( Component ( Component : : CREATURE , rID , rVal , 0 ) ) ;
break ;
2010-02-07 18:05:27 +02:00
}
2009-12-30 22:49:10 +02:00
}
2010-01-30 22:53:47 +02:00
void CGSeerHut : : setPropertyDer ( ui8 what , ui32 val )
2010-01-01 14:15:20 +02:00
{
2010-01-30 22:53:47 +02:00
switch ( what )
2010-01-01 14:15:20 +02:00
{
2010-01-30 22:53:47 +02:00
case 10 :
2012-10-03 17:49:29 +03:00
quest - > progress = val ;
2010-01-30 22:53:47 +02:00
break ;
case 11 :
2012-10-03 17:49:29 +03:00
quest - > missionType = CQuest : : MISSION_NONE ;
2010-01-30 22:53:47 +02:00
break ;
2010-01-01 14:15:20 +02:00
}
2010-01-30 22:53:47 +02:00
}
2010-02-02 19:05:03 +02:00
void CGSeerHut : : newTurn ( ) const
{
2012-10-03 17:49:29 +03:00
if ( quest - > lastDay > = 0 & & quest - > lastDay < cb - > getDate ( 0 ) ) //time is up
2010-02-02 19:05:03 +02:00
{
2012-07-21 12:58:42 +03:00
cb - > setObjProperty ( id , 11 , 0 ) ;
cb - > setObjProperty ( id , 10 , 0 ) ;
2010-02-02 19:05:03 +02:00
}
2010-01-30 22:53:47 +02:00
2010-02-02 19:05:03 +02:00
}
2010-01-30 22:53:47 +02:00
void CGSeerHut : : onHeroVisit ( const CGHeroInstance * h ) const
{
InfoWindow iw ;
iw . player = h - > getOwner ( ) ;
2012-10-03 17:49:29 +03:00
if ( quest - > missionType )
2010-01-01 14:15:20 +02:00
{
2012-10-03 17:49:29 +03:00
bool firstVisit = ! quest - > progress ;
2011-10-16 22:24:05 +03:00
bool failRequirements = ! checkQuest ( h ) ;
2012-07-06 23:19:54 +03:00
bool isCustom = false ;
2012-07-08 09:33:41 +03:00
2012-07-19 21:52:44 +03:00
if ( firstVisit )
2011-10-16 22:24:05 +03:00
{
2012-10-03 17:49:29 +03:00
isCustom = quest - > isCustomFirst ;
2012-09-16 16:34:01 +03:00
cb - > setObjProperty ( id , 10 , CQuest : : IN_PROGRESS ) ;
2012-07-06 22:12:04 +03:00
AddQuest aq ;
2012-10-03 17:49:29 +03:00
aq . quest = QuestInfo ( quest , this , visitablePos ( ) ) ;
2012-07-06 22:12:04 +03:00
aq . player = h - > tempOwner ;
cb - > sendAndApply ( & aq ) ; //TODO: merge with setObjProperty?
2011-10-16 22:24:05 +03:00
}
else if ( failRequirements )
{
2012-10-03 17:49:29 +03:00
isCustom = quest - > isCustomNext ;
2011-10-16 22:24:05 +03:00
}
if ( firstVisit | | failRequirements )
2010-01-01 14:15:20 +02:00
{
2012-07-08 09:33:41 +03:00
getVisitText ( iw . text , iw . components , isCustom , firstVisit , h ) ;
2010-02-07 18:05:27 +02:00
cb - > showInfoDialog ( & iw ) ;
}
2011-10-16 22:24:05 +03:00
if ( ! failRequirements ) // propose completion, also on first visit
2010-02-07 18:05:27 +02:00
{
BlockingDialog bd ( true , false ) ;
bd . player = h - > getOwner ( ) ;
bd . soundID = soundBase : : QUEST ;
2012-07-19 21:52:44 +03:00
2012-07-08 09:33:41 +03:00
getCompletionText ( bd . text , bd . components , isCustom , h ) ;
2012-07-19 21:52:44 +03:00
2010-02-07 18:05:27 +02:00
cb - > showBlockingDialog ( & bd , boost : : bind ( & CGSeerHut : : finishQuest , this , h , _1 ) ) ;
return ;
2010-02-02 19:05:03 +02:00
}
2010-01-01 14:15:20 +02:00
}
2010-01-30 22:53:47 +02:00
else
{
2012-10-03 17:49:29 +03:00
iw . text < < VLC - > generaltexth - > seerEmpty [ quest - > textOption ] ;
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : SEER_HUT )
2010-02-04 22:34:20 +02:00
iw . text . addReplacement ( seerName ) ;
2010-02-07 18:05:27 +02:00
cb - > showInfoDialog ( & iw ) ;
2010-01-30 22:53:47 +02:00
}
}
2010-02-04 22:34:20 +02:00
int CGSeerHut : : checkDirection ( ) const
{
2011-02-11 14:27:38 +02:00
int3 cord = getCreatureToKill ( ) - > pos ;
2010-02-04 22:34:20 +02:00
if ( ( double ) cord . x / ( double ) cb - > getMapSize ( ) . x < 0.34 ) //north
{
if ( ( double ) cord . y / ( double ) cb - > getMapSize ( ) . y < 0.34 ) //northwest
return 8 ;
else if ( ( double ) cord . y / ( double ) cb - > getMapSize ( ) . y < 0.67 ) //north
return 1 ;
else //northeast
return 2 ;
}
else if ( ( double ) cord . x / ( double ) cb - > getMapSize ( ) . x < 0.67 ) //horizontal
{
if ( ( double ) cord . y / ( double ) cb - > getMapSize ( ) . y < 0.34 ) //west
return 7 ;
else if ( ( double ) cord . y / ( double ) cb - > getMapSize ( ) . y < 0.67 ) //central
return 9 ;
else //east
return 3 ;
}
else //south
{
if ( ( double ) cord . y / ( double ) cb - > getMapSize ( ) . y < 0.34 ) //southwest
return 6 ;
else if ( ( double ) cord . y / ( double ) cb - > getMapSize ( ) . y < 0.67 ) //south
return 5 ;
else //southeast
return 4 ;
}
}
2010-12-06 01:10:02 +02:00
void CGSeerHut : : finishQuest ( const CGHeroInstance * h , ui32 accept ) const
2010-02-02 19:05:03 +02:00
{
if ( accept )
{
2012-10-03 17:49:29 +03:00
switch ( quest - > missionType )
2010-02-02 19:05:03 +02:00
{
case CQuest : : MISSION_ART :
2012-10-03 17:49:29 +03:00
for ( std : : vector < ui16 > : : const_iterator it = quest - > m5arts . begin ( ) ; it ! = quest - > m5arts . end ( ) ; + + it )
2010-02-07 18:05:27 +02:00
{
2010-12-29 23:04:22 +02:00
cb - > removeArtifact ( ArtifactLocation ( h , h - > getArtPos ( * it , false ) ) ) ;
2010-02-07 18:05:27 +02:00
}
2010-02-02 19:05:03 +02:00
break ;
case CQuest : : MISSION_ARMY :
2012-10-03 17:49:29 +03:00
cb - > takeCreatures ( h - > id , quest - > m6creatures ) ;
2010-02-02 19:05:03 +02:00
break ;
case CQuest : : MISSION_RESOURCES :
for ( int i = 0 ; i < 7 ; + + i )
{
2012-10-03 17:49:29 +03:00
cb - > giveResource ( h - > getOwner ( ) , i , - quest - > m7resources [ i ] ) ;
2010-02-02 19:05:03 +02:00
}
break ;
default :
break ;
}
2012-09-16 16:34:01 +03:00
cb - > setObjProperty ( id , 10 , CQuest : : COMPLETE ) ; //mission complete - for AI
2012-07-18 13:10:14 +03:00
cb - > setObjProperty ( id , 11 , 0 ) ; //no more mission available - redundant?
2012-07-19 21:52:44 +03:00
completeQuest ( h ) ; //make sure to remove QuestQuard at the very end
2010-02-02 19:05:03 +02:00
}
}
2010-02-04 22:34:20 +02:00
void CGSeerHut : : completeQuest ( const CGHeroInstance * h ) const //reward
2010-01-30 22:53:47 +02:00
{
2010-02-04 22:34:20 +02:00
switch ( rewardType )
{
case 1 : //experience
2010-07-26 01:47:59 +03:00
{
2012-09-23 21:01:04 +03:00
TExpType expVal = h - > calculateXp ( rVal ) ;
2010-07-26 01:47:59 +03:00
cb - > changePrimSkill ( h - > id , 4 , expVal , false ) ;
2010-02-04 22:34:20 +02:00
break ;
2010-07-26 01:47:59 +03:00
}
2010-02-04 22:34:20 +02:00
case 2 : //mana points
2010-07-26 01:47:59 +03:00
{
2010-02-04 22:34:20 +02:00
cb - > setManaPoints ( h - > id , h - > mana + rVal ) ;
break ;
2010-07-26 01:47:59 +03:00
}
2010-02-04 22:34:20 +02:00
case 3 : case 4 : //morale /luck
{
2010-05-02 21:20:26 +03:00
Bonus hb ( Bonus : : ONE_WEEK , ( rewardType = = 3 ? Bonus : : MORALE : Bonus : : LUCK ) ,
Bonus : : OBJECT , rVal , h - > id , " " , - 1 ) ;
2010-02-04 22:34:20 +02:00
GiveBonus gb ;
2010-02-10 04:56:00 +02:00
gb . id = h - > id ;
2010-02-04 22:34:20 +02:00
gb . bonus = hb ;
cb - > giveHeroBonus ( & gb ) ;
}
break ;
case 5 : //resources
cb - > giveResource ( h - > getOwner ( ) , rID , rVal ) ;
break ;
case 6 : //main ability bonus (attak, defence etd.)
cb - > changePrimSkill ( h - > id , rID , rVal , false ) ;
break ;
case 7 : // secondary ability gain
cb - > changeSecSkill ( h - > id , rID , rVal , false ) ;
break ;
case 8 : // artifact
2010-12-26 16:34:11 +02:00
cb - > giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ rID ] , - 2 ) ;
2010-02-04 22:34:20 +02:00
break ;
case 9 : // spell
{
std : : set < ui32 > spell ;
spell . insert ( rID ) ;
cb - > changeSpells ( h - > id , true , spell ) ;
}
break ;
case 10 : // creature
2010-12-06 01:10:02 +02:00
{
CCreatureSet creatures ;
creatures . setCreature ( 0 , rID , rVal ) ;
2010-12-13 01:44:16 +02:00
cb - > giveCreatures ( this , h , creatures , false ) ;
2010-12-06 01:10:02 +02:00
}
2010-02-04 22:34:20 +02:00
break ;
default :
break ;
}
2010-01-01 14:15:20 +02:00
}
2011-02-11 14:27:38 +02:00
const CGHeroInstance * CGSeerHut : : getHeroToKill ( bool allowNull ) const
{
2012-10-03 17:49:29 +03:00
const CGObjectInstance * o = cb - > getObjByQuestIdentifier ( quest - > m13489val ) ;
2011-02-11 14:27:38 +02:00
if ( allowNull & & ! o )
return NULL ;
2012-09-23 21:01:04 +03:00
assert ( o & & o - > ID = = Obj : : HERO ) ;
2011-02-11 14:27:38 +02:00
return static_cast < const CGHeroInstance * > ( o ) ;
}
const CGCreature * CGSeerHut : : getCreatureToKill ( bool allowNull ) const
{
2012-10-03 17:49:29 +03:00
const CGObjectInstance * o = cb - > getObjByQuestIdentifier ( quest - > m13489val ) ;
2011-02-11 14:27:38 +02:00
if ( allowNull & & ! o )
return NULL ;
2012-09-23 21:01:04 +03:00
assert ( o & & o - > ID = = Obj : : MONSTER ) ;
2011-02-11 14:27:38 +02:00
return static_cast < const CGCreature * > ( o ) ;
}
2010-01-30 22:53:47 +02:00
void CGQuestGuard : : initObj ( )
{
2010-02-04 22:34:20 +02:00
blockVisit = true ;
2012-10-03 17:49:29 +03:00
quest - > progress = 0 ;
quest - > textOption = ran ( ) % 3 + 3 ; //3-5
if ( quest - > missionType )
2010-02-02 19:05:03 +02:00
{
2012-10-03 17:49:29 +03:00
if ( ! quest - > isCustomFirst )
quest - > firstVisitText = VLC - > generaltexth - > quests [ quest - > missionType - 1 ] [ 0 ] [ quest - > textOption ] ;
if ( ! quest - > isCustomNext )
quest - > nextVisitText = VLC - > generaltexth - > quests [ quest - > missionType - 1 ] [ 1 ] [ quest - > textOption ] ;
if ( ! quest - > isCustomComplete )
quest - > completedText = VLC - > generaltexth - > quests [ quest - > missionType - 1 ] [ 2 ] [ quest - > textOption ] ;
2010-02-02 19:05:03 +02:00
}
else
2012-10-03 17:49:29 +03:00
quest - > firstVisitText = VLC - > generaltexth - > seerEmpty [ quest - > textOption ] ;
2010-01-30 22:53:47 +02:00
}
2010-02-04 22:34:20 +02:00
void CGQuestGuard : : completeQuest ( const CGHeroInstance * h ) const
2010-01-30 22:53:47 +02:00
{
2010-02-04 22:34:20 +02:00
cb - > removeObject ( id ) ;
2010-01-30 22:53:47 +02:00
}
2009-01-11 00:08:18 +02:00
void CGWitchHut : : initObj ( )
{
ability = allowedAbilities [ ran ( ) % allowedAbilities . size ( ) ] ;
}
void CGWitchHut : : onHeroVisit ( const CGHeroInstance * h ) const
{
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = soundBase : : gazebo ;
2009-01-11 00:08:18 +02:00
iw . player = h - > getOwner ( ) ;
2012-03-14 16:02:38 +03:00
if ( ! wasVisited ( h - > tempOwner ) )
2009-02-20 14:39:27 +02:00
cb - > setObjProperty ( id , 10 , h - > tempOwner ) ;
2009-01-11 00:08:18 +02:00
2010-12-23 22:18:10 +02:00
if ( h - > getSecSkillLevel ( static_cast < CGHeroInstance : : SecondarySkill > ( ability ) ) ) //you alredy know this skill
2009-01-11 00:08:18 +02:00
{
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 172 ) ;
2009-07-09 22:15:22 +03:00
iw . text . addReplacement ( MetaString : : SEC_SKILL_NAME , ability ) ;
2009-01-11 00:08:18 +02:00
}
2012-12-18 14:24:13 +03:00
else if ( ! h - > canLearnSkill ( ) ) //already all skills slots used
2009-01-11 00:08:18 +02:00
{
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 173 ) ;
2009-07-09 22:15:22 +03:00
iw . text . addReplacement ( MetaString : : SEC_SKILL_NAME , ability ) ;
2009-01-11 00:08:18 +02:00
}
else //give sec skill
{
2012-08-31 20:39:11 +03:00
iw . components . push_back ( Component ( Component : : SEC_SKILL , ability , 1 , 0 ) ) ;
2009-01-11 00:08:18 +02:00
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 171 ) ;
2009-07-09 22:15:22 +03:00
iw . text . addReplacement ( MetaString : : SEC_SKILL_NAME , ability ) ;
2009-01-11 00:08:18 +02:00
cb - > changeSecSkill ( h - > id , ability , 1 , true ) ;
}
cb - > showInfoDialog ( & iw ) ;
2009-02-03 07:28:05 +02:00
}
2009-02-20 14:39:27 +02:00
const std : : string & CGWitchHut : : getHoverText ( ) const
{
hoverName = VLC - > generaltexth - > names [ ID ] ;
2012-03-14 16:02:38 +03:00
if ( wasVisited ( cb - > getCurrentPlayer ( ) ) ) //TODO: use local player, not current
2009-02-20 14:39:27 +02:00
{
hoverName + = " \n " + VLC - > generaltexth - > allTexts [ 356 ] ; // + (learn %s)
boost : : algorithm : : replace_first ( hoverName , " %s " , VLC - > generaltexth - > skillName [ ability ] ) ;
2009-07-27 20:55:56 +03:00
const CGHeroInstance * h = cb - > getSelectedHero ( cb - > getCurrentPlayer ( ) ) ;
2010-12-23 22:18:10 +02:00
if ( h & & h - > getSecSkillLevel ( static_cast < CGHeroInstance : : SecondarySkill > ( ability ) ) ) //hero knows that ability
2009-02-20 14:39:27 +02:00
hoverName + = " \n \n " + VLC - > generaltexth - > allTexts [ 357 ] ; // (Already learned)
}
return hoverName ;
}
2012-03-14 16:02:38 +03:00
bool CGBonusingObject : : wasVisited ( const CGHeroInstance * h ) const
{
return h - > hasBonusFrom ( Bonus : : OBJECT , ID ) ;
}
2009-02-04 15:40:54 +02:00
void CGBonusingObject : : onHeroVisit ( const CGHeroInstance * h ) const
{
2010-05-02 21:20:26 +03:00
bool visited = h - > hasBonusFrom ( Bonus : : OBJECT , ID ) ;
2012-07-06 23:19:54 +03:00
int messageID = 0 ;
2009-04-24 00:09:10 +03:00
int bonusMove = 0 , sound = - 1 ;
2009-02-04 15:40:54 +02:00
InfoWindow iw ;
iw . player = h - > tempOwner ;
2009-02-05 11:49:45 +02:00
GiveBonus gbonus ;
2010-02-10 04:56:00 +02:00
gbonus . id = h - > id ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . duration = Bonus : : ONE_BATTLE ;
gbonus . bonus . source = Bonus : : OBJECT ;
2011-02-20 20:32:39 +02:00
gbonus . bonus . sid = ID ;
2009-02-04 15:40:54 +02:00
2010-05-02 21:20:26 +03:00
bool second = false ;
Bonus secondBonus ;
2009-02-04 15:40:54 +02:00
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : BUOY :
2009-07-19 10:16:33 +03:00
messageID = 21 ;
2009-07-26 06:33:13 +03:00
sound = soundBase : : MORALE ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . type = Bonus : : MORALE ;
2009-07-19 10:16:33 +03:00
gbonus . bonus . val = + 1 ;
gbonus . bdescr < < std : : pair < ui8 , ui32 > ( 6 , 94 ) ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : SWAN_POND :
2009-02-04 15:40:54 +02:00
messageID = 29 ;
2009-04-24 00:09:10 +03:00
sound = soundBase : : LUCK ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . type = Bonus : : LUCK ;
2009-02-05 11:49:45 +02:00
gbonus . bonus . val = 2 ;
gbonus . bdescr < < std : : pair < ui8 , ui32 > ( 6 , 67 ) ;
bonusMove = - h - > movement ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : FAERIE_RING :
2009-02-04 15:40:54 +02:00
messageID = 49 ;
2009-04-24 00:09:10 +03:00
sound = soundBase : : LUCK ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . type = Bonus : : LUCK ;
2009-02-05 11:49:45 +02:00
gbonus . bonus . val = 1 ;
gbonus . bdescr < < std : : pair < ui8 , ui32 > ( 6 , 71 ) ;
2009-02-04 15:40:54 +02:00
break ;
2012-09-23 21:01:04 +03:00
case Obj : : FOUNTAIN_OF_FORTUNE :
2009-02-04 15:40:54 +02:00
messageID = 55 ;
2009-04-24 00:09:10 +03:00
sound = soundBase : : LUCK ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . type = Bonus : : LUCK ;
2009-02-05 11:49:45 +02:00
gbonus . bonus . val = rand ( ) % 5 - 1 ;
gbonus . bdescr < < std : : pair < ui8 , ui32 > ( 6 , 69 ) ;
2009-07-09 22:15:22 +03:00
gbonus . bdescr . addReplacement ( ( gbonus . bonus . val < 0 ? " - " : " + " ) + boost : : lexical_cast < std : : string > ( gbonus . bonus . val ) ) ;
2009-02-04 15:40:54 +02:00
break ;
2012-09-23 21:01:04 +03:00
case Obj : : IDOL_OF_FORTUNE :
2009-02-04 15:40:54 +02:00
messageID = 62 ;
2009-11-30 18:14:41 +02:00
sound = soundBase : : experience ;
2010-05-02 21:20:26 +03:00
2009-02-05 11:49:45 +02:00
gbonus . bonus . val = 1 ;
gbonus . bdescr < < std : : pair < ui8 , ui32 > ( 6 , 68 ) ;
2010-05-02 21:20:26 +03:00
if ( cb - > getDate ( 1 ) = = 7 ) //7th day of week
{
gbonus . bonus . type = Bonus : : MORALE ;
second = true ;
secondBonus = gbonus . bonus ;
secondBonus . type = Bonus : : LUCK ;
}
else
{
gbonus . bonus . type = ( cb - > getDate ( 1 ) % 2 ) ? Bonus : : LUCK : Bonus : : MORALE ;
}
2009-02-05 11:49:45 +02:00
break ;
2012-09-23 21:01:04 +03:00
case Obj : : MERMAID :
2009-07-19 10:16:33 +03:00
messageID = 83 ;
sound = soundBase : : LUCK ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . type = Bonus : : LUCK ;
2009-07-19 10:16:33 +03:00
gbonus . bonus . val = 1 ;
gbonus . bdescr < < std : : pair < ui8 , ui32 > ( 6 , 72 ) ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : RALLY_FLAG :
2009-04-24 00:09:10 +03:00
sound = soundBase : : MORALE ;
2009-02-05 11:49:45 +02:00
messageID = 111 ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . type = Bonus : : MORALE ;
2009-02-05 11:49:45 +02:00
gbonus . bonus . val = 1 ;
gbonus . bdescr < < std : : pair < ui8 , ui32 > ( 6 , 102 ) ;
2010-05-02 21:20:26 +03:00
second = true ;
secondBonus = gbonus . bonus ;
secondBonus . type = Bonus : : LUCK ;
2009-02-05 11:49:45 +02:00
bonusMove = 400 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : OASIS :
2009-02-05 11:49:45 +02:00
messageID = 95 ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . type = Bonus : : MORALE ;
2009-02-05 11:49:45 +02:00
gbonus . bonus . val = 1 ;
gbonus . bdescr < < std : : pair < ui8 , ui32 > ( 6 , 95 ) ;
bonusMove = 800 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : TEMPLE :
2009-02-05 11:49:45 +02:00
messageID = 140 ;
2009-04-22 21:48:56 +03:00
iw . soundID = soundBase : : temple ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . type = Bonus : : MORALE ;
2009-02-05 11:49:45 +02:00
if ( cb - > getDate ( 1 ) = = 7 ) //sunday
{
gbonus . bonus . val = 2 ;
gbonus . bdescr < < std : : pair < ui8 , ui32 > ( 6 , 97 ) ;
}
else
{
gbonus . bonus . val = 1 ;
gbonus . bdescr < < std : : pair < ui8 , ui32 > ( 6 , 96 ) ;
}
break ;
2012-09-23 21:01:04 +03:00
case Obj : : WATERING_HOLE :
2009-04-24 00:09:10 +03:00
sound = soundBase : : MORALE ;
2009-02-05 11:49:45 +02:00
messageID = 166 ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . type = Bonus : : MORALE ;
2009-02-05 11:49:45 +02:00
gbonus . bonus . val = 1 ;
gbonus . bdescr < < std : : pair < ui8 , ui32 > ( 6 , 100 ) ;
bonusMove = 400 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : FOUNTAIN_OF_YOUTH :
2009-04-24 00:09:10 +03:00
sound = soundBase : : MORALE ;
2009-02-05 11:49:45 +02:00
messageID = 57 ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . type = Bonus : : MORALE ;
2009-02-05 11:49:45 +02:00
gbonus . bonus . val = 1 ;
gbonus . bdescr < < std : : pair < ui8 , ui32 > ( 6 , 103 ) ;
bonusMove = 400 ;
2009-02-04 15:40:54 +02:00
break ;
2012-09-23 21:01:04 +03:00
case Obj : : STABLES :
2012-05-18 20:35:46 +03:00
sound = soundBase : : STORE ;
2011-09-02 08:44:53 +03:00
bool someUpgradeDone = false ;
2010-05-02 21:20:26 +03:00
for ( TSlots : : const_iterator i = h - > Slots ( ) . begin ( ) ; i ! = h - > Slots ( ) . end ( ) ; + + i )
2009-08-15 19:12:30 +03:00
{
2010-11-22 02:34:46 +02:00
if ( i - > second - > type - > idNumber = = 10 )
2011-09-02 08:44:53 +03:00
{
cb - > changeStackType ( StackLocation ( h , i - > first ) , VLC - > creh - > creatures [ 11 ] ) ;
someUpgradeDone = true ;
}
2009-08-15 19:12:30 +03:00
}
2011-09-02 08:44:53 +03:00
if ( someUpgradeDone )
2009-08-15 19:12:30 +03:00
{
2010-07-07 15:20:15 +03:00
messageID = 138 ;
2010-08-17 18:50:17 +03:00
iw . components . push_back ( Component ( Component : : CREATURE , 11 , 0 , 1 ) ) ;
2009-08-15 19:12:30 +03:00
}
2010-07-07 15:20:15 +03:00
else
messageID = 137 ;
2011-09-02 08:44:53 +03:00
2010-05-02 21:20:26 +03:00
gbonus . bonus . type = Bonus : : LAND_MOVEMENT ;
2009-08-11 10:50:29 +03:00
gbonus . bonus . val = 600 ;
2010-02-14 08:53:37 +02:00
bonusMove = 600 ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . duration = Bonus : : ONE_WEEK ;
2009-08-11 10:50:29 +03:00
gbonus . bdescr < < std : : pair < ui8 , ui32 > ( 6 , 100 ) ;
break ;
2009-02-04 15:40:54 +02:00
}
2012-07-06 23:19:54 +03:00
assert ( messageID ) ;
2009-02-04 15:40:54 +02:00
if ( visited )
{
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : RALLY_FLAG | | ID = = Obj : : OASIS | | ID = = Obj : : MERMAID | | ID = = Obj : : STABLES )
2009-02-05 11:49:45 +02:00
messageID - - ;
else
messageID + + ;
2009-02-04 15:40:54 +02:00
}
else
{
2010-05-02 21:20:26 +03:00
//TODO: fix if second bonus val != main bonus val
if ( gbonus . bonus . type = = Bonus : : MORALE | | secondBonus . type = = Bonus : : MORALE )
2012-08-31 20:39:11 +03:00
iw . components . push_back ( Component ( Component : : MORALE , 0 , gbonus . bonus . val , 0 ) ) ;
2010-05-02 21:20:26 +03:00
if ( gbonus . bonus . type = = Bonus : : LUCK | | secondBonus . type = = Bonus : : LUCK )
2012-08-31 20:39:11 +03:00
iw . components . push_back ( Component ( Component : : LUCK , 0 , gbonus . bonus . val , 0 ) ) ;
2009-02-04 15:40:54 +02:00
cb - > giveHeroBonus ( & gbonus ) ;
2010-05-02 21:20:26 +03:00
if ( second )
{
gbonus . bonus = secondBonus ;
cb - > giveHeroBonus ( & gbonus ) ;
}
2010-02-14 08:53:37 +02:00
if ( bonusMove ) //swan pond - take all move points, stables - give move point this day
2009-02-04 15:40:54 +02:00
{
SetMovePoints smp ;
smp . hid = h - > id ;
2009-02-05 11:49:45 +02:00
smp . val = h - > movement + bonusMove ;
2009-02-04 15:40:54 +02:00
cb - > setMovePoints ( & smp ) ;
}
}
2009-04-24 00:09:10 +03:00
iw . soundID = sound ;
2009-02-04 15:40:54 +02:00
iw . text < < std : : pair < ui8 , ui32 > ( 11 , messageID ) ;
cb - > showInfoDialog ( & iw ) ;
}
const std : : string & CGBonusingObject : : getHoverText ( ) const
2009-02-06 13:15:39 +02:00
{
const CGHeroInstance * h = cb - > getSelectedHero ( cb - > getCurrentPlayer ( ) ) ;
hoverName = VLC - > generaltexth - > names [ ID ] ;
2012-07-19 21:52:44 +03:00
if ( h )
2009-02-06 13:15:39 +02:00
{
2010-05-02 21:20:26 +03:00
if ( ! h - > hasBonusFrom ( Bonus : : OBJECT , ID ) )
2009-02-06 13:15:39 +02:00
hoverName + = " " + VLC - > generaltexth - > allTexts [ 353 ] ; //not visited
else
hoverName + = " " + VLC - > generaltexth - > allTexts [ 352 ] ; //visited
}
return hoverName ;
}
2009-07-19 10:16:33 +03:00
void CGBonusingObject : : initObj ( )
{
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : BUOY | | ID = = Obj : : MERMAID )
2009-07-19 10:16:33 +03:00
{
blockVisit = true ;
}
}
2012-07-19 21:52:44 +03:00
void CGMagicSpring : : onHeroVisit ( const CGHeroInstance * h ) const
{
int messageID ;
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . soundID = soundBase : : GENIE ;
if ( ! visited )
{
if ( h - > mana > h - > manaLimit ( ) )
messageID = 76 ;
else
{
messageID = 74 ;
cb - > setManaPoints ( h - > id , 2 * h - > manaLimit ( ) ) ;
cb - > setObjProperty ( id , ObjProperty : : VISITED , true ) ;
}
}
else
messageID = 75 ;
iw . text < < std : : pair < ui8 , ui32 > ( 11 , messageID ) ;
cb - > showInfoDialog ( & iw ) ;
}
const std : : string & CGMagicSpring : : getHoverText ( ) const
{
2009-09-23 15:42:14 +03:00
hoverName = VLC - > generaltexth - > names [ ID ] ;
if ( ! visited )
hoverName + = " " + VLC - > generaltexth - > allTexts [ 353 ] ; //not visited
else
hoverName + = " " + VLC - > generaltexth - > allTexts [ 352 ] ; //visited
return hoverName ;
2012-07-19 21:52:44 +03:00
}
2009-09-23 15:42:14 +03:00
2009-02-06 13:15:39 +02:00
void CGMagicWell : : onHeroVisit ( const CGHeroInstance * h ) const
{
int message ;
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = soundBase : : faerie ;
2009-02-06 13:15:39 +02:00
iw . player = h - > tempOwner ;
2010-05-02 21:20:26 +03:00
if ( h - > hasBonusFrom ( Bonus : : OBJECT , ID ) ) //has already visited Well today
2009-02-06 13:15:39 +02:00
{
message = 78 ;
}
else if ( h - > mana < h - > manaLimit ( ) )
{
2009-07-19 10:16:33 +03:00
giveDummyBonus ( h - > id ) ;
2009-02-06 13:15:39 +02:00
cb - > setManaPoints ( h - > id , h - > manaLimit ( ) ) ;
message = 77 ;
}
else
{
message = 79 ;
}
iw . text < < std : : pair < ui8 , ui32 > ( 11 , message ) ; //"A second drink at the well in one day will not help you."
cb - > showInfoDialog ( & iw ) ;
}
const std : : string & CGMagicWell : : getHoverText ( ) const
2009-02-04 15:40:54 +02:00
{
2009-07-19 10:16:33 +03:00
getNameVis ( hoverName ) ;
2009-02-04 15:40:54 +02:00
return hoverName ;
2009-02-08 08:42:15 +02:00
}
2009-08-16 18:39:18 +03:00
void CGPandoraBox : : initObj ( )
{
2012-09-23 21:01:04 +03:00
blockVisit = ( ID = = Obj : : PANDORAS_BOX ) ; //block only if it's really pandora's box (events also derive from that class)
2009-08-16 18:39:18 +03:00
}
2009-08-17 13:47:08 +03:00
2009-08-16 18:39:18 +03:00
void CGPandoraBox : : onHeroVisit ( const CGHeroInstance * h ) const
{
BlockingDialog bd ( true , false ) ;
bd . player = h - > getOwner ( ) ;
bd . soundID = soundBase : : QUEST ;
bd . text . addTxt ( MetaString : : ADVOB_TXT , 14 ) ;
2012-07-19 21:52:44 +03:00
cb - > showBlockingDialog ( & bd , boost : : bind ( & CGPandoraBox : : open , this , h , _1 ) ) ;
2009-08-16 18:39:18 +03:00
}
2009-08-17 13:47:08 +03:00
2009-08-16 18:39:18 +03:00
void CGPandoraBox : : open ( const CGHeroInstance * h , ui32 accept ) const
{
if ( accept )
{
2010-05-02 21:20:26 +03:00
if ( stacksCount ( ) > 0 ) //if pandora's box is protected by army
2009-08-16 18:39:18 +03:00
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , 16 ) ;
cb - > showInfoDialog ( & iw ) ;
2009-09-13 01:17:23 +03:00
cb - > startBattleI ( h , this , boost : : bind ( & CGPandoraBox : : endBattle , this , h , _1 ) ) ; //grants things after battle
2009-08-16 18:39:18 +03:00
}
2009-08-17 13:47:08 +03:00
else if ( message . size ( ) = = 0 & & resources . size ( ) = = 0
& & primskills . size ( ) = = 0 & & abilities . size ( ) = = 0
& & abilityLevels . size ( ) = = 0 & & artifacts . size ( ) = = 0
2010-05-02 21:20:26 +03:00
& & spells . size ( ) = = 0 & & creatures . Slots ( ) . size ( ) > 0
2009-08-17 13:47:08 +03:00
& & gainedExp = = 0 & & manaDiff = = 0 & & moraleDiff = = 0 & & luckDiff = = 0 ) //if it gives nothing without battle
2009-08-16 18:39:18 +03:00
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , 15 ) ;
cb - > showInfoDialog ( & iw ) ;
}
2009-08-17 13:47:08 +03:00
else //if it gives something without battle
2009-08-16 18:39:18 +03:00
{
giveContents ( h , false ) ;
}
}
}
2009-08-17 13:47:08 +03:00
2009-08-19 09:56:53 +03:00
void CGPandoraBox : : endBattle ( const CGHeroInstance * h , BattleResult * result ) const
{
if ( result - > winner )
return ;
giveContents ( h , true ) ;
}
2009-08-16 18:39:18 +03:00
void CGPandoraBox : : giveContents ( const CGHeroInstance * h , bool afterBattle ) const
2009-02-08 08:42:15 +02:00
{
InfoWindow iw ;
2009-04-21 01:57:07 +03:00
iw . player = h - > getOwner ( ) ;
2011-03-08 15:27:32 +02:00
std : : string msg = message ; //in case box is removed in the meantime
2009-04-21 01:57:07 +03:00
bool changesPrimSkill = false ;
for ( int i = 0 ; i < primskills . size ( ) ; i + + )
{
if ( primskills [ i ] )
{
changesPrimSkill = true ;
break ;
}
}
if ( gainedExp | | changesPrimSkill | | abilities . size ( ) )
{
2012-09-23 21:01:04 +03:00
TExpType expVal = h - > calculateXp ( gainedExp ) ;
2011-06-25 12:28:28 +03:00
//getText(iw,afterBattle,175,h); //wtf?
iw . text . addTxt ( MetaString : : ADVOB_TXT , 175 ) ; //%s learns something
iw . text . addReplacement ( h - > name ) ;
2009-04-21 01:57:07 +03:00
2010-07-26 01:47:59 +03:00
if ( expVal )
iw . components . push_back ( Component ( Component : : EXPERIENCE , 0 , expVal , 0 ) ) ;
2009-04-21 01:57:07 +03:00
for ( int i = 0 ; i < primskills . size ( ) ; i + + )
if ( primskills [ i ] )
iw . components . push_back ( Component ( Component : : PRIM_SKILL , i , primskills [ i ] , 0 ) ) ;
for ( int i = 0 ; i < abilities . size ( ) ; i + + )
iw . components . push_back ( Component ( Component : : SEC_SKILL , abilities [ i ] , abilityLevels [ i ] , 0 ) ) ;
cb - > showInfoDialog ( & iw ) ;
//give exp
2010-07-26 01:47:59 +03:00
if ( expVal )
cb - > changePrimSkill ( h - > id , 4 , expVal , false ) ;
2009-04-21 01:57:07 +03:00
//give prim skills
for ( int i = 0 ; i < primskills . size ( ) ; i + + )
if ( primskills [ i ] )
cb - > changePrimSkill ( h - > id , i , primskills [ i ] , false ) ;
//give sec skills
for ( int i = 0 ; i < abilities . size ( ) ; i + + )
{
2010-12-23 22:18:10 +02:00
int curLev = h - > getSecSkillLevel ( static_cast < CGHeroInstance : : SecondarySkill > ( abilities [ i ] ) ) ;
2009-04-21 01:57:07 +03:00
2012-12-18 14:24:13 +03:00
if ( ( curLev & & curLev < abilityLevels [ i ] ) | | ( h - > canLearnSkill ( ) ) )
2009-04-21 01:57:07 +03:00
{
cb - > changeSecSkill ( h - > id , abilities [ i ] , abilityLevels [ i ] , true ) ;
}
}
}
2009-04-23 18:48:53 +03:00
if ( spells . size ( ) )
{
std : : set < ui32 > spellsToGive ;
iw . components . clear ( ) ;
2011-06-25 12:28:28 +03:00
if ( spells . size ( ) > 1 )
{
iw . text . addTxt ( MetaString : : ADVOB_TXT , 188 ) ; //%s learns spells
}
else
{
iw . text . addTxt ( MetaString : : ADVOB_TXT , 184 ) ; //%s learns a spell
}
iw . text . addReplacement ( h - > name ) ;
2010-12-19 16:39:56 +02:00
std : : vector < ConstTransitivePtr < CSpell > > * sp = & VLC - > spellh - > spells ;
2010-09-26 19:11:50 +03:00
for ( std : : vector < si32 > : : const_iterator i = spells . begin ( ) ; i ! = spells . end ( ) ; i + + )
2009-04-23 18:48:53 +03:00
{
2010-12-23 22:18:10 +02:00
if ( ( * sp ) [ * i ] - > level < = h - > getSecSkillLevel ( CGHeroInstance : : WISDOM ) + 2 ) //enough wisdom
2010-09-26 19:11:50 +03:00
{
iw . components . push_back ( Component ( Component : : SPELL , * i , 0 , 0 ) ) ;
spellsToGive . insert ( * i ) ;
}
2009-04-23 18:48:53 +03:00
}
if ( spellsToGive . size ( ) )
{
cb - > changeSpells ( h - > id , true , spellsToGive ) ;
cb - > showInfoDialog ( & iw ) ;
}
}
2009-04-21 01:57:07 +03:00
if ( manaDiff )
{
2009-04-23 18:48:53 +03:00
getText ( iw , afterBattle , manaDiff , 176 , 177 , h ) ;
2009-04-21 01:57:07 +03:00
iw . components . push_back ( Component ( Component : : PRIM_SKILL , 5 , manaDiff , 0 ) ) ;
cb - > showInfoDialog ( & iw ) ;
cb - > setManaPoints ( h - > id , h - > mana + manaDiff ) ;
}
if ( moraleDiff )
{
2009-04-23 18:48:53 +03:00
getText ( iw , afterBattle , moraleDiff , 178 , 179 , h ) ;
2009-04-21 01:57:07 +03:00
iw . components . push_back ( Component ( Component : : MORALE , 0 , moraleDiff , 0 ) ) ;
cb - > showInfoDialog ( & iw ) ;
GiveBonus gb ;
2010-05-02 21:20:26 +03:00
gb . bonus = Bonus ( Bonus : : ONE_BATTLE , Bonus : : MORALE , Bonus : : OBJECT , moraleDiff , id , " " ) ;
2010-02-10 04:56:00 +02:00
gb . id = h - > id ;
2009-04-21 01:57:07 +03:00
cb - > giveHeroBonus ( & gb ) ;
}
if ( luckDiff )
{
getText ( iw , afterBattle , luckDiff , 180 , 181 , h ) ;
iw . components . push_back ( Component ( Component : : LUCK , 0 , luckDiff , 0 ) ) ;
cb - > showInfoDialog ( & iw ) ;
GiveBonus gb ;
2010-05-02 21:20:26 +03:00
gb . bonus = Bonus ( Bonus : : ONE_BATTLE , Bonus : : LUCK , Bonus : : OBJECT , luckDiff , id , " " ) ;
2010-02-10 04:56:00 +02:00
gb . id = h - > id ;
2009-04-21 01:57:07 +03:00
cb - > giveHeroBonus ( & gb ) ;
}
iw . components . clear ( ) ;
iw . text . clear ( ) ;
for ( int i = 0 ; i < resources . size ( ) ; i + + )
{
if ( resources [ i ] < 0 )
iw . components . push_back ( Component ( Component : : RESOURCE , i , resources [ i ] , 0 ) ) ;
}
if ( iw . components . size ( ) )
{
getText ( iw , afterBattle , 182 , h ) ;
cb - > showInfoDialog ( & iw ) ;
}
iw . components . clear ( ) ;
iw . text . clear ( ) ;
for ( int i = 0 ; i < resources . size ( ) ; i + + )
{
if ( resources [ i ] > 0 )
iw . components . push_back ( Component ( Component : : RESOURCE , i , resources [ i ] , 0 ) ) ;
}
if ( iw . components . size ( ) )
{
getText ( iw , afterBattle , 183 , h ) ;
cb - > showInfoDialog ( & iw ) ;
}
2009-11-24 22:29:50 +02:00
iw . components . clear ( ) ;
// getText(iw,afterBattle,183,h);
2011-06-25 12:28:28 +03:00
iw . text . addTxt ( MetaString : : ADVOB_TXT , 183 ) ; //% has found treasure
iw . text . addReplacement ( h - > name ) ;
2009-04-21 01:57:07 +03:00
for ( int i = 0 ; i < artifacts . size ( ) ; i + + )
{
iw . components . push_back ( Component ( Component : : ARTIFACT , artifacts [ i ] , 0 , 0 ) ) ;
2009-07-16 01:46:00 +03:00
if ( iw . components . size ( ) > = 14 )
{
cb - > showInfoDialog ( & iw ) ;
iw . components . clear ( ) ;
2011-06-25 12:28:28 +03:00
iw . text . addTxt ( MetaString : : ADVOB_TXT , 183 ) ; //% has found treasure - once more?
iw . text . addReplacement ( h - > name ) ;
2009-07-16 01:46:00 +03:00
}
2009-04-21 01:57:07 +03:00
}
if ( iw . components . size ( ) )
{
cb - > showInfoDialog ( & iw ) ;
}
for ( int i = 0 ; i < resources . size ( ) ; i + + )
if ( resources [ i ] )
cb - > giveResource ( h - > getOwner ( ) , i , resources [ i ] ) ;
for ( int i = 0 ; i < artifacts . size ( ) ; i + + )
2010-12-26 16:34:11 +02:00
cb - > giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ artifacts [ i ] ] , - 2 ) ;
2009-04-21 01:57:07 +03:00
iw . components . clear ( ) ;
iw . text . clear ( ) ;
2009-09-23 15:05:33 +03:00
2010-05-02 21:20:26 +03:00
if ( creatures . Slots ( ) . size ( ) )
2009-09-26 12:16:59 +03:00
{ //this part is taken straight from creature bank
2012-07-19 21:52:44 +03:00
MetaString loot ;
2010-12-12 01:11:26 +02:00
for ( TSlots : : const_iterator i = creatures . Slots ( ) . begin ( ) ; i ! = creatures . Slots ( ) . end ( ) ; i + + )
2009-09-26 12:16:59 +03:00
{ //build list of joined creatures
2010-11-22 02:34:46 +02:00
iw . components . push_back ( Component ( * i - > second ) ) ;
2009-09-26 12:16:59 +03:00
loot < < " %s " ;
2010-11-22 02:34:46 +02:00
loot . addReplacement ( * i - > second ) ;
2009-09-25 18:40:28 +03:00
}
2009-09-23 15:05:33 +03:00
2010-12-12 01:11:26 +02:00
if ( creatures . Slots ( ) . size ( ) = = 1 & & creatures . Slots ( ) . begin ( ) - > second - > count = = 1 )
iw . text . addTxt ( MetaString : : ADVOB_TXT , 185 ) ;
2009-09-26 12:16:59 +03:00
else
2010-12-12 01:11:26 +02:00
iw . text . addTxt ( MetaString : : ADVOB_TXT , 186 ) ;
2009-09-26 12:16:59 +03:00
2010-11-22 02:34:46 +02:00
iw . text . addReplacement ( loot . buildList ( ) ) ;
iw . text . addReplacement ( h - > name ) ;
2009-09-26 12:16:59 +03:00
cb - > showInfoDialog ( & iw ) ;
2010-12-13 01:44:16 +02:00
cb - > giveCreatures ( this , h , creatures , true ) ;
2009-09-23 15:05:33 +03:00
}
2011-03-08 15:27:32 +02:00
if ( ! afterBattle & & msg . size ( ) )
2009-04-21 01:57:07 +03:00
{
2011-03-08 15:27:32 +02:00
iw . text < < msg ;
2009-04-21 01:57:07 +03:00
cb - > showInfoDialog ( & iw ) ;
}
2010-11-15 14:51:53 +02:00
if ( ! creatures . Slots ( ) . size ( ) )
cb - > removeObject ( id ) ; //only when we don't need to display garrison window
2009-04-21 01:57:07 +03:00
}
2009-08-16 18:39:18 +03:00
void CGPandoraBox : : getText ( InfoWindow & iw , bool & afterBattle , int text , const CGHeroInstance * h ) const
2009-04-21 01:57:07 +03:00
{
2009-07-16 01:46:00 +03:00
if ( afterBattle | | ! message . size ( ) )
2009-04-21 01:57:07 +03:00
{
iw . text . addTxt ( MetaString : : ADVOB_TXT , text ) ; //%s has lost treasure.
2009-07-09 22:15:22 +03:00
iw . text . addReplacement ( h - > name ) ;
2009-04-21 01:57:07 +03:00
}
else
{
iw . text < < message ;
afterBattle = true ;
}
}
2009-08-16 18:39:18 +03:00
void CGPandoraBox : : getText ( InfoWindow & iw , bool & afterBattle , int val , int negative , int positive , const CGHeroInstance * h ) const
2009-04-21 01:57:07 +03:00
{
iw . components . clear ( ) ;
iw . text . clear ( ) ;
2009-07-16 01:46:00 +03:00
if ( afterBattle | | ! message . size ( ) )
2009-04-21 01:57:07 +03:00
{
iw . text . addTxt ( MetaString : : ADVOB_TXT , val < 0 ? negative : positive ) ; //%s's luck takes a turn for the worse / %s's luck increases
2009-07-09 22:15:22 +03:00
iw . text . addReplacement ( h - > name ) ;
2009-04-21 01:57:07 +03:00
}
else
{
iw . text < < message ;
afterBattle = true ;
}
2009-03-14 13:25:25 +02:00
}
2009-08-19 09:56:53 +03:00
void CGEvent : : onHeroVisit ( const CGHeroInstance * h ) const
{
2012-07-19 21:52:44 +03:00
if ( ! ( availableFor & ( 1 < < h - > tempOwner ) ) )
2009-08-19 09:56:53 +03:00
return ;
2012-12-02 15:21:44 +03:00
if ( cb - > getPlayerSettings ( h - > tempOwner ) - > playerID )
2009-08-19 09:56:53 +03:00
{
if ( humanActivate )
activated ( h ) ;
}
else if ( computerActivate )
activated ( h ) ;
}
void CGEvent : : activated ( const CGHeroInstance * h ) const
{
2010-05-02 21:20:26 +03:00
if ( stacksCount ( ) > 0 )
2009-08-19 09:56:53 +03:00
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
if ( message . size ( ) )
iw . text < < message ;
else
iw . text . addTxt ( MetaString : : ADVOB_TXT , 16 ) ;
cb - > showInfoDialog ( & iw ) ;
2009-09-13 01:17:23 +03:00
cb - > startBattleI ( h , this , boost : : bind ( & CGEvent : : endBattle , this , h , _1 ) ) ;
2009-08-19 09:56:53 +03:00
}
else
{
giveContents ( h , false ) ;
}
}
2009-03-14 13:25:25 +02:00
void CGObservatory : : onHeroVisit ( const CGHeroInstance * h ) const
{
2009-04-15 02:29:26 +03:00
InfoWindow iw ;
iw . player = h - > tempOwner ;
2010-06-06 09:05:39 +03:00
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : REDWOOD_OBSERVATORY :
case Obj : : PILLAR_OF_FIRE :
{
iw . soundID = soundBase : : LIGHTHOUSE ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , 98 + ( ID = = Obj : : PILLAR_OF_FIRE ) ) ;
2010-06-06 09:05:39 +03:00
2012-09-23 21:01:04 +03:00
FoWChange fw ;
fw . player = h - > tempOwner ;
fw . mode = 1 ;
cb - > getTilesInRange ( fw . tiles , pos , 20 , h - > tempOwner , 1 ) ;
cb - > sendAndApply ( & fw ) ;
break ;
}
case Obj : : COVER_OF_DARKNESS :
{
iw . text . addTxt ( MetaString : : ADVOB_TXT , 31 ) ;
hideTiles ( h - > tempOwner , 20 ) ;
break ;
}
2010-06-06 09:05:39 +03:00
}
2009-04-15 02:29:26 +03:00
cb - > showInfoDialog ( & iw ) ;
2009-03-14 13:25:25 +02:00
}
void CGShrine : : onHeroVisit ( const CGHeroInstance * h ) const
{
if ( spell = = 255 )
{
tlog1 < < " Not initialized shrine visited! \n " ;
return ;
}
2012-03-14 16:02:38 +03:00
if ( ! wasVisited ( h - > tempOwner ) )
2009-03-14 19:19:53 +02:00
cb - > setObjProperty ( id , 10 , h - > tempOwner ) ;
2009-03-14 13:25:25 +02:00
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = soundBase : : temple ;
2009-03-14 13:25:25 +02:00
iw . player = h - > getOwner ( ) ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , 127 + ID - 88 ) ;
iw . text . addTxt ( MetaString : : SPELL_NAME , spell ) ;
iw . text < < " . " ;
2012-09-25 17:40:39 +03:00
if ( ! h - > getArt ( ArtifactPosition : : SPELLBOOK ) )
2009-03-14 13:25:25 +02:00
{
iw . text . addTxt ( MetaString : : ADVOB_TXT , 131 ) ;
}
2012-09-23 21:01:04 +03:00
else if ( ID = = Obj : : SHRINE_OF_MAGIC_THOUGHT & & ! h - > getSecSkillLevel ( CGHeroInstance : : WISDOM ) ) //it's third level spell and hero doesn't have wisdom
2009-03-14 13:25:25 +02:00
{
iw . text . addTxt ( MetaString : : ADVOB_TXT , 130 ) ;
}
else if ( vstd : : contains ( h - > spells , spell ) ) //hero already knows the spell
{
iw . text . addTxt ( MetaString : : ADVOB_TXT , 174 ) ;
}
else //give spell
{
std : : set < ui32 > spells ;
spells . insert ( spell ) ;
2010-06-21 07:43:10 +03:00
cb - > changeSpells ( h - > id , true , spells ) ;
2009-03-14 13:25:25 +02:00
iw . components . push_back ( Component ( Component : : SPELL , spell , 0 , 0 ) ) ;
}
cb - > showInfoDialog ( & iw ) ;
}
void CGShrine : : initObj ( )
{
if ( spell = = 255 ) //spell not set
{
int level = ID - 87 ;
2009-09-24 22:28:26 +03:00
std : : vector < ui16 > possibilities ;
cb - > getAllowedSpells ( possibilities , level ) ;
2009-03-14 13:25:25 +02:00
if ( ! possibilities . size ( ) )
{
tlog1 < < " Error: cannot init shrine, no allowed spells! \n " ;
return ;
}
spell = possibilities [ ran ( ) % possibilities . size ( ) ] ;
}
}
const std : : string & CGShrine : : getHoverText ( ) const
{
2009-03-14 19:19:53 +02:00
hoverName = VLC - > generaltexth - > names [ ID ] ;
2012-03-14 16:02:38 +03:00
if ( wasVisited ( cb - > getCurrentPlayer ( ) ) ) //TODO: use local player, not current
2009-03-14 19:19:53 +02:00
{
hoverName + = " \n " + VLC - > generaltexth - > allTexts [ 355 ] ; // + (learn %s)
2010-12-19 16:39:56 +02:00
boost : : algorithm : : replace_first ( hoverName , " %s " , VLC - > spellh - > spells [ spell ] - > name ) ;
2009-03-14 19:19:53 +02:00
const CGHeroInstance * h = cb - > getSelectedHero ( cb - > getCurrentPlayer ( ) ) ;
if ( h & & vstd : : contains ( h - > spells , spell ) ) //hero knows that ability
hoverName + = " \n \n " + VLC - > generaltexth - > allTexts [ 354 ] ; // (Already learned)
}
2009-03-14 13:25:25 +02:00
return hoverName ;
2009-03-14 19:19:53 +02:00
}
2009-03-19 16:17:19 +02:00
void CGSignBottle : : initObj ( )
{
//if no text is set than we pick random from the predefined ones
if ( ! message . size ( ) )
message = VLC - > generaltexth - > randsign [ ran ( ) % VLC - > generaltexth - > randsign . size ( ) ] ;
2009-07-19 10:16:33 +03:00
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : OCEAN_BOTTLE )
2009-07-19 10:16:33 +03:00
{
blockVisit = true ;
}
2009-03-19 16:17:19 +02:00
}
void CGSignBottle : : onHeroVisit ( const CGHeroInstance * h ) const
{
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = soundBase : : STORE ;
2009-03-19 16:17:19 +02:00
iw . player = h - > getOwner ( ) ;
iw . text < < message ;
cb - > showInfoDialog ( & iw ) ;
2009-07-20 01:16:07 +03:00
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : OCEAN_BOTTLE )
2009-07-20 01:16:07 +03:00
cb - > removeObject ( id ) ;
2009-04-08 22:55:50 +03:00
}
void CGScholar : : giveAnyBonus ( const CGHeroInstance * h ) const
{
}
void CGScholar : : onHeroVisit ( const CGHeroInstance * h ) const
{
int type = bonusType ;
int bid = bonusID ;
//check if the bonus if applicable, if not - give primary skill (always possible)
2010-12-23 22:18:10 +02:00
int ssl = h - > getSecSkillLevel ( static_cast < CGHeroInstance : : SecondarySkill > ( bid ) ) ; //current sec skill level, used if bonusType == 1
2009-04-08 22:55:50 +03:00
if ( ( type = = 1
2012-12-18 14:24:13 +03:00
& & ( ( ssl = = 3 ) | | ( ! ssl & & ! h - > canLearnSkill ( ) ) ) ) ////hero already has expert level in the skill or (don't know skill and doesn't have free slot)
2010-12-23 22:18:10 +02:00
| | ( type = = 2 & & ( ! h - > getArt ( 17 ) | | vstd : : contains ( h - > spells , ( ui32 ) bid )
| | ( VLC - > spellh - > spells [ bid ] - > level > h - > getSecSkillLevel ( CGHeroInstance : : WISDOM ) + 2 )
2010-09-26 19:11:50 +03:00
) ) ) //hero doesn't have a spellbook or already knows the spell or doesn't have Wisdom
2009-04-08 22:55:50 +03:00
{
type = 0 ;
2011-12-14 00:23:17 +03:00
bid = ran ( ) % GameConstants : : PRIMARY_SKILLS ;
2009-04-08 22:55:50 +03:00
}
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = soundBase : : gazebo ;
2009-04-08 22:55:50 +03:00
iw . player = h - > getOwner ( ) ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , 115 ) ;
switch ( type )
{
case 0 :
cb - > changePrimSkill ( h - > id , bid , + 1 ) ;
2009-07-16 01:46:00 +03:00
iw . components . push_back ( Component ( Component : : PRIM_SKILL , bid , + 1 , 0 ) ) ;
2009-04-08 22:55:50 +03:00
break ;
case 1 :
2009-06-10 03:06:35 +03:00
cb - > changeSecSkill ( h - > id , bid , + 1 ) ;
iw . components . push_back ( Component ( Component : : SEC_SKILL , bid , ssl + 1 , 0 ) ) ;
2009-04-08 22:55:50 +03:00
break ;
case 2 :
{
std : : set < ui32 > hlp ;
hlp . insert ( bid ) ;
cb - > changeSpells ( h - > id , true , hlp ) ;
iw . components . push_back ( Component ( Component : : SPELL , bid , 0 , 0 ) ) ;
}
break ;
default :
tlog1 < < " Error: wrong bonustype ( " < < ( int ) type < < " ) for Scholar! \n " ;
return ;
}
cb - > showInfoDialog ( & iw ) ;
cb - > removeObject ( id ) ;
}
void CGScholar : : initObj ( )
{
blockVisit = true ;
if ( bonusType = = 255 )
{
bonusType = ran ( ) % 3 ;
switch ( bonusType )
{
case 0 :
2011-12-14 00:23:17 +03:00
bonusID = ran ( ) % GameConstants : : PRIMARY_SKILLS ;
2009-04-08 22:55:50 +03:00
break ;
case 1 :
2011-12-14 00:23:17 +03:00
bonusID = ran ( ) % GameConstants : : SKILL_QUANTITY ;
2009-04-08 22:55:50 +03:00
break ;
case 2 :
2009-10-23 22:58:21 +03:00
std : : vector < ui16 > possibilities ;
for ( int i = 1 ; i < 6 ; + + i )
cb - > getAllowedSpells ( possibilities , i ) ;
bonusID = possibilities [ ran ( ) % possibilities . size ( ) ] ;
2009-04-08 22:55:50 +03:00
break ;
}
}
}
2009-04-16 03:28:54 +03:00
2009-08-25 18:08:18 +03:00
void CGGarrison : : onHeroVisit ( const CGHeroInstance * h ) const
{
2010-08-13 13:46:08 +03:00
int ally = cb - > gameState ( ) - > getPlayerRelations ( h - > tempOwner , tempOwner ) ;
2010-08-06 16:14:10 +03:00
if ( ! ally & & stacksCount ( ) > 0 ) {
2009-08-25 18:08:18 +03:00
//TODO: Find a way to apply magic garrison effects in battle.
2009-09-13 01:17:23 +03:00
cb - > startBattleI ( h , this , boost : : bind ( & CGGarrison : : fightOver , this , h , _1 ) ) ;
2009-08-25 18:08:18 +03:00
return ;
}
//New owner.
2010-08-06 16:14:10 +03:00
if ( ! ally )
2009-08-25 18:08:18 +03:00
cb - > setOwner ( id , h - > tempOwner ) ;
2009-09-09 20:49:03 +03:00
cb - > showGarrisonDialog ( id , h - > id , removableUnits , 0 ) ;
2009-08-25 18:08:18 +03:00
}
void CGGarrison : : fightOver ( const CGHeroInstance * h , BattleResult * result ) const
{
if ( result - > winner = = 0 )
onHeroVisit ( h ) ;
}
2009-12-20 19:14:14 +02:00
ui8 CGGarrison : : getPassableness ( ) const
{
2010-08-06 16:14:10 +03:00
if ( ! stacksCount ( ) ) //empty - anyone can visit
2011-12-14 00:23:17 +03:00
return GameConstants : : ALL_PLAYERS ;
2012-09-29 20:36:48 +03:00
if ( tempOwner = = GameConstants : : NEUTRAL_PLAYER ) //neutral guarded - no one can visit
2010-08-06 16:14:10 +03:00
return 0 ;
2012-07-19 21:52:44 +03:00
2010-08-08 17:39:41 +03:00
ui8 mask = 0 ;
2010-08-06 16:14:10 +03:00
TeamState * ts = cb - > gameState ( ) - > getPlayerTeam ( tempOwner ) ;
BOOST_FOREACH ( ui8 it , ts - > players )
mask | = 1 < < it ; //allies - add to possible visitors
2012-07-19 21:52:44 +03:00
2010-08-06 16:14:10 +03:00
return mask ;
2009-12-20 19:14:14 +02:00
}
2009-04-16 03:28:54 +03:00
void CGOnceVisitable : : onHeroVisit ( const CGHeroInstance * h ) const
{
2012-06-07 06:02:30 +03:00
int sound ;
int txtid ;
2009-04-16 03:28:54 +03:00
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : CORPSE :
2012-07-19 21:52:44 +03:00
txtid = 37 ;
2009-04-24 00:09:10 +03:00
sound = soundBase : : MYSTERY ;
2009-04-16 03:28:54 +03:00
break ;
2012-09-23 21:01:04 +03:00
case Obj : : LEAN_TO :
2009-04-24 00:09:10 +03:00
sound = soundBase : : GENIE ;
2009-04-16 03:28:54 +03:00
txtid = 64 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : WAGON :
2009-04-24 00:09:10 +03:00
sound = soundBase : : GENIE ;
2009-04-16 03:28:54 +03:00
txtid = 154 ;
break ;
2012-09-23 21:01:04 +03:00
case Obj : : WARRIORS_TOMB :
2012-06-07 06:02:30 +03:00
{
//ask if player wants to search the Tomb
BlockingDialog bd ( true , false ) ;
bd . soundID = soundBase : : GRAVEYARD ;
bd . player = h - > getOwner ( ) ;
bd . text . addTxt ( MetaString : : ADVOB_TXT , 161 ) ;
cb - > showBlockingDialog ( & bd , boost : : bind ( & CGOnceVisitable : : searchTomb , this , h , _1 ) ) ;
return ;
}
2009-04-16 03:28:54 +03:00
break ;
default :
tlog1 < < " Error: Unknown object ( " < < ID < < " ) treated as CGOnceVisitable! \n " ;
return ;
}
InfoWindow iw ;
2009-04-24 00:09:10 +03:00
iw . soundID = sound ;
2009-04-16 03:28:54 +03:00
iw . player = h - > getOwner ( ) ;
if ( players . size ( ) ) //we have been already visited...
{
txtid + + ;
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : WAGON ) //wagon has extra text (for finding art) we need to omit
2009-04-16 03:28:54 +03:00
txtid + + ;
2010-01-01 14:15:20 +02:00
iw . text . addTxt ( MetaString : : ADVOB_TXT , txtid ) ;
2009-04-16 03:28:54 +03:00
}
else //first visit - give bonus!
{
switch ( artOrRes )
{
2009-10-22 08:39:04 +03:00
case 0 : // first visit but empty
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : CORPSE )
2009-12-01 12:10:42 +02:00
+ + txtid ;
else
txtid + = 2 ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , txtid ) ;
2009-04-16 03:28:54 +03:00
break ;
case 1 : //art
iw . components . push_back ( Component ( Component : : ARTIFACT , bonusType , 0 , 0 ) ) ;
2010-12-26 16:34:11 +02:00
cb - > giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ bonusType ] , - 2 ) ;
2009-12-01 12:10:42 +02:00
iw . text . addTxt ( MetaString : : ADVOB_TXT , txtid ) ;
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : CORPSE )
2009-12-01 12:10:42 +02:00
{
iw . text < < " %s " ;
2010-11-22 02:34:46 +02:00
iw . text . addReplacement ( MetaString : : ART_NAMES , bonusType ) ;
2009-12-01 12:10:42 +02:00
}
2009-04-16 03:28:54 +03:00
break ;
case 2 : //res
2009-12-01 12:10:42 +02:00
iw . text . addTxt ( MetaString : : ADVOB_TXT , txtid ) ;
2009-12-01 13:47:52 +02:00
iw . components . push_back ( Component ( Component : : RESOURCE , bonusType , bonusVal , 0 ) ) ;
cb - > giveResource ( h - > getOwner ( ) , bonusType , bonusVal ) ;
2009-04-16 03:28:54 +03:00
break ;
}
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : WAGON & & artOrRes = = 1 )
2009-07-12 17:07:36 +03:00
{
iw . text . localStrings . back ( ) . second + + ;
iw . text . addReplacement ( MetaString : : ART_NAMES , bonusType ) ;
}
2009-04-16 03:28:54 +03:00
}
cb - > showInfoDialog ( & iw ) ;
cb - > setObjProperty ( id , 10 , h - > getOwner ( ) ) ;
}
const std : : string & CGOnceVisitable : : getHoverText ( ) const
{
hoverName = VLC - > generaltexth - > names [ ID ] + " " ;
2012-03-14 16:02:38 +03:00
hoverName + = ( wasVisited ( cb - > getCurrentPlayer ( ) )
2009-04-16 03:28:54 +03:00
? ( VLC - > generaltexth - > allTexts [ 352 ] ) //visited
: ( VLC - > generaltexth - > allTexts [ 353 ] ) ) ; //not visited
return hoverName ;
}
void CGOnceVisitable : : initObj ( )
{
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : CORPSE :
2009-04-16 03:28:54 +03:00
{
blockVisit = true ;
int hlp = ran ( ) % 100 ;
if ( hlp < 20 )
{
artOrRes = 1 ;
2009-10-24 22:21:32 +03:00
bonusType = cb - > getRandomArt ( CArtifact : : ART_TREASURE | CArtifact : : ART_MINOR | CArtifact : : ART_MAJOR ) ;
2009-04-16 03:28:54 +03:00
}
else
{
artOrRes = 0 ;
}
}
break ;
2012-09-23 21:01:04 +03:00
case Obj : : LEAN_TO :
2009-04-16 03:28:54 +03:00
{
artOrRes = 2 ;
bonusType = ran ( ) % 6 ; //any basic resource without gold
bonusVal = ran ( ) % 4 + 1 ;
}
2009-11-28 13:26:32 +02:00
break ;
2009-04-16 03:28:54 +03:00
2012-09-23 21:01:04 +03:00
case Obj : : WARRIORS_TOMB :
2009-04-16 03:28:54 +03:00
{
artOrRes = 1 ;
int hlp = ran ( ) % 100 ;
if ( hlp < 30 )
2009-10-24 22:21:32 +03:00
bonusType = cb - > getRandomArt ( CArtifact : : ART_TREASURE ) ;
2009-04-16 03:28:54 +03:00
else if ( hlp < 80 )
2009-10-24 22:21:32 +03:00
bonusType = cb - > getRandomArt ( CArtifact : : ART_MINOR ) ;
2009-04-16 03:28:54 +03:00
else if ( hlp < 95 )
2009-10-24 22:21:32 +03:00
bonusType = cb - > getRandomArt ( CArtifact : : ART_MAJOR ) ;
2009-04-16 03:28:54 +03:00
else
2009-10-24 22:21:32 +03:00
bonusType = cb - > getRandomArt ( CArtifact : : ART_RELIC ) ;
2009-04-16 03:28:54 +03:00
}
break ;
2012-09-23 21:01:04 +03:00
case Obj : : WAGON :
2009-04-16 03:28:54 +03:00
{
int hlp = ran ( ) % 100 ;
if ( hlp < 10 )
{
artOrRes = 0 ; // nothing... :(
}
else if ( hlp < 50 ) //minor or treasure art
{
artOrRes = 1 ;
2009-10-24 22:21:32 +03:00
bonusType = cb - > getRandomArt ( CArtifact : : ART_TREASURE | CArtifact : : ART_MINOR ) ;
2009-04-16 03:28:54 +03:00
}
else //2 - 5 of non-gold resource
{
artOrRes = 2 ;
bonusType = ran ( ) % 6 ;
bonusVal = ran ( ) % 4 + 2 ;
}
}
2009-11-28 13:26:32 +02:00
break ;
2009-04-16 03:28:54 +03:00
}
}
void CGOnceVisitable : : searchTomb ( const CGHeroInstance * h , ui32 accept ) const
{
if ( accept )
{
InfoWindow iw ;
iw . player = h - > getOwner ( ) ;
iw . components . push_back ( Component ( Component : : MORALE , 0 , - 3 , 0 ) ) ;
if ( players . size ( ) ) //we've been already visited, player found nothing
{
iw . text . addTxt ( MetaString : : ADVOB_TXT , 163 ) ;
}
else //first visit - give artifact
{
iw . text . addTxt ( MetaString : : ADVOB_TXT , 162 ) ;
iw . components . push_back ( Component ( Component : : ARTIFACT , bonusType , 0 , 0 ) ) ;
2009-07-09 22:15:22 +03:00
iw . text . addReplacement ( MetaString : : ART_NAMES , bonusType ) ;
2009-04-16 03:28:54 +03:00
2010-12-26 16:34:11 +02:00
cb - > giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ bonusType ] , - 2 ) ;
2012-07-19 21:52:44 +03:00
}
2010-05-02 21:20:26 +03:00
if ( ! h - > hasBonusFrom ( Bonus : : OBJECT , ID ) ) //we don't have modifier from this object yet
2009-08-13 04:03:11 +03:00
{
2012-07-19 21:52:44 +03:00
//ruin morale
2009-08-13 04:03:11 +03:00
GiveBonus gb ;
2010-02-10 04:56:00 +02:00
gb . id = h - > id ;
2010-05-02 21:20:26 +03:00
gb . bonus = Bonus ( Bonus : : ONE_BATTLE , Bonus : : MORALE , Bonus : : OBJECT , - 3 , id , " " ) ;
2009-08-13 04:03:11 +03:00
gb . bdescr . addTxt ( MetaString : : ARRAY_TXT , 104 ) ; //Warrior Tomb Visited -3
cb - > giveHeroBonus ( & gb ) ;
2009-04-16 03:28:54 +03:00
}
2009-10-03 14:16:42 +03:00
cb - > showInfoDialog ( & iw ) ;
cb - > setObjProperty ( id , 10 , h - > getOwner ( ) ) ;
2009-08-11 10:50:29 +03:00
}
}
void CBank : : initObj ( )
{
2012-03-03 13:08:01 +03:00
index = VLC - > objh - > bankObjToIndex ( this ) ;
2009-08-11 10:50:29 +03:00
bc = NULL ;
daycounter = 0 ;
multiplier = 1 ;
}
2009-09-17 17:27:28 +03:00
const std : : string & CBank : : getHoverText ( ) const
{
hoverName = VLC - > objh - > creBanksNames [ index ] ;
if ( bc = = NULL )
hoverName + = " " + VLC - > generaltexth - > allTexts [ 352 ] ;
else
hoverName + = " " + VLC - > generaltexth - > allTexts [ 353 ] ;
return hoverName ;
}
2009-10-24 22:21:32 +03:00
void CBank : : reset ( ui16 var1 ) //prevents desync
2009-08-11 10:50:29 +03:00
{
2009-09-01 17:36:48 +03:00
ui8 chance = 0 ;
2009-08-23 21:01:08 +03:00
for ( ui8 i = 0 ; i < VLC - > objh - > banksInfo [ index ] . size ( ) ; i + + )
2012-07-19 21:52:44 +03:00
{
2009-12-29 03:07:17 +02:00
if ( var1 < ( chance + = VLC - > objh - > banksInfo [ index ] [ i ] - > chance ) )
2009-08-26 08:08:59 +03:00
{
2009-12-29 03:07:17 +02:00
bc = VLC - > objh - > banksInfo [ index ] [ i ] ;
2009-08-26 08:08:59 +03:00
break ;
}
2009-08-11 10:50:29 +03:00
}
artifacts . clear ( ) ;
2009-10-24 22:21:32 +03:00
}
void CBank : : initialize ( ) const
{
cb - > setObjProperty ( id , 14 , ran ( ) ) ; //synchronous reset
2009-08-26 08:08:59 +03:00
for ( ui8 i = 0 ; i < = 3 ; i + + )
2009-10-24 22:21:32 +03:00
{
for ( ui8 n = 0 ; n < bc - > artifacts [ i ] ; n + + ) //new function using proper randomization algorithm
2012-07-19 21:52:44 +03:00
{
2010-08-17 18:16:08 +03:00
cb - > setObjProperty ( id , 18 + i , ran ( ) ) ; //synchronic artifacts
2009-08-24 17:59:24 +03:00
}
2009-08-11 10:50:29 +03:00
}
2010-08-17 18:16:08 +03:00
cb - > setObjProperty ( id , 17 , ran ( ) ) ; //get army
2009-08-11 10:50:29 +03:00
}
void CBank : : setPropertyDer ( ui8 what , ui32 val )
2009-08-26 08:08:59 +03:00
/// random values are passed as arguments and processed identically on all clients
2009-08-11 10:50:29 +03:00
{
switch ( what )
{
case 11 : //daycounter
2009-08-23 18:02:21 +03:00
if ( val = = 0 )
2010-06-28 08:07:21 +03:00
daycounter = 1 ; //yes, 1
2009-08-23 18:02:21 +03:00
else
daycounter + + ;
2009-08-11 10:50:29 +03:00
break ;
2009-08-26 08:08:59 +03:00
case 12 : //multiplier, in percent
2011-12-14 00:23:17 +03:00
multiplier = val / 100.0 ;
2009-08-11 10:50:29 +03:00
break ;
case 13 : //bank preset
2009-12-29 03:07:17 +02:00
bc = VLC - > objh - > banksInfo [ index ] [ val ] ;
2009-08-11 10:50:29 +03:00
break ;
2009-08-19 09:56:53 +03:00
case 14 :
2009-10-24 22:21:32 +03:00
reset ( val % 100 ) ;
2009-08-19 09:56:53 +03:00
break ;
case 15 :
bc = NULL ;
break ;
2010-03-24 16:58:17 +02:00
case 16 : //remove rewards from Derelict Ship
2009-08-23 18:02:21 +03:00
artifacts . clear ( ) ;
break ;
2009-08-26 08:08:59 +03:00
case 17 : //set ArmedInstance army
{
2009-10-24 22:21:32 +03:00
int upgraded = 0 ;
if ( val % 100 < bc - > upgradeChance ) //once again anti-desync
upgraded = 1 ;
switch ( bc - > guards . size ( ) )
2009-08-26 08:08:59 +03:00
{
2009-10-24 22:21:32 +03:00
case 1 :
2010-03-02 20:13:22 +02:00
for ( int i = 0 ; i < 4 ; + + i )
2010-05-02 21:20:26 +03:00
setCreature ( i , bc - > guards [ 0 ] . first , bc - > guards [ 0 ] . second / 5 ) ;
setCreature ( 4 , bc - > guards [ 0 ] . first + upgraded , bc - > guards [ 0 ] . second / 5 ) ;
2009-10-24 22:21:32 +03:00
break ;
case 4 :
2009-08-26 08:08:59 +03:00
{
2009-10-24 22:21:32 +03:00
std : : vector < std : : pair < ui16 , ui32 > > : : const_iterator it ;
2010-03-24 16:58:17 +02:00
if ( bc - > guards . back ( ) . second ) //all stacks are present
2009-10-24 22:21:32 +03:00
{
2010-03-24 16:58:17 +02:00
for ( it = bc - > guards . begin ( ) ; it ! = bc - > guards . end ( ) ; it + + )
{
2010-05-02 21:20:26 +03:00
setCreature ( stacksCount ( ) , it - > first , it - > second ) ;
2010-03-24 16:58:17 +02:00
}
}
2010-06-26 11:31:31 +03:00
else if ( bc - > guards [ 2 ] . second ) //Wraiths are present, split two stacks in Crypt
2010-03-24 16:58:17 +02:00
{
2010-05-02 21:20:26 +03:00
setCreature ( 0 , bc - > guards [ 0 ] . first , bc - > guards [ 0 ] . second / 2 ) ;
2010-06-26 11:31:31 +03:00
setCreature ( 1 , bc - > guards [ 1 ] . first , bc - > guards [ 1 ] . second / 2 ) ;
setCreature ( 2 , bc - > guards [ 2 ] . first + upgraded , bc - > guards [ 2 ] . second ) ;
setCreature ( 3 , bc - > guards [ 1 ] . first , bc - > guards [ 1 ] . second / 2 ) ;
setCreature ( 4 , bc - > guards [ 0 ] . first , bc - > guards [ 0 ] . second - ( bc - > guards [ 0 ] . second / 2 ) ) ;
}
else //split both stacks
{
for ( int i = 0 ; i < 3 ; + + i ) //skellies
setCreature ( 2 * i , bc - > guards [ 0 ] . first , bc - > guards [ 0 ] . second / 3 ) ;
for ( int i = 0 ; i < 2 ; + + i ) //zombies
setCreature ( 2 * i + 1 , bc - > guards [ 1 ] . first , bc - > guards [ 1 ] . second / 2 ) ;
2009-10-24 22:21:32 +03:00
}
2009-08-26 08:08:59 +03:00
}
2009-10-24 22:21:32 +03:00
break ;
default :
2009-10-26 11:11:10 +02:00
tlog2 < < " Error: Unexpected army data: " < < bc - > guards . size ( ) < < " items found " ;
2009-10-24 22:21:32 +03:00
return ;
2009-08-26 08:08:59 +03:00
}
}
break ;
2009-10-24 22:21:32 +03:00
case 18 : //add Artifact
2010-06-28 08:07:21 +03:00
{
int id = cb - > getArtSync ( val , CArtifact : : ART_TREASURE ) ;
artifacts . push_back ( id ) ;
cb - > erasePickedArt ( id ) ;
break ;
}
case 19 : //add Artifact
{
int id = cb - > getArtSync ( val , CArtifact : : ART_MINOR ) ;
artifacts . push_back ( id ) ;
cb - > erasePickedArt ( id ) ;
2009-10-24 22:21:32 +03:00
break ;
2010-06-28 08:07:21 +03:00
}
case 20 : //add Artifact
{
int id = cb - > getArtSync ( val , CArtifact : : ART_MAJOR ) ;
artifacts . push_back ( id ) ;
cb - > erasePickedArt ( id ) ;
break ;
}
case 21 : //add Artifact
{
int id = cb - > getArtSync ( val , CArtifact : : ART_RELIC ) ;
artifacts . push_back ( id ) ;
cb - > erasePickedArt ( id ) ;
break ;
}
2009-08-11 10:50:29 +03:00
}
}
2012-07-19 21:52:44 +03:00
void CBank : : newTurn ( ) const
2009-08-11 10:50:29 +03:00
{
if ( bc = = NULL )
{
2009-09-25 18:40:28 +03:00
if ( cb - > getDate ( 0 ) = = 1 )
2009-10-24 22:21:32 +03:00
initialize ( ) ; //initialize on first day
2009-09-25 18:40:28 +03:00
else if ( daycounter > = 28 & & ( subID < 13 | | subID > 16 ) ) //no reset for Emissaries
2009-08-11 10:50:29 +03:00
{
2009-10-24 22:21:32 +03:00
initialize ( ) ;
2009-09-25 18:40:28 +03:00
cb - > setObjProperty ( id , 11 , 0 ) ; //daycounter 0
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : DERELICT_SHIP & & cb - > getDate ( 0 ) > 1 )
2010-03-24 16:58:17 +02:00
{
cb - > setObjProperty ( id , 12 , 0 ) ; //ugly hack to make derelict ships usable only once
cb - > setObjProperty ( id , 16 , 0 ) ;
}
2009-08-11 10:50:29 +03:00
}
else
2009-08-26 08:08:59 +03:00
cb - > setObjProperty ( id , 11 , 1 ) ; //daycounter++
2009-08-11 10:50:29 +03:00
}
}
2012-03-14 16:02:38 +03:00
bool CBank : : wasVisited ( ui8 player ) const
{
return ! bc ;
}
2009-08-11 10:50:29 +03:00
void CBank : : onHeroVisit ( const CGHeroInstance * h ) const
{
2010-03-24 16:58:17 +02:00
if ( bc )
2009-08-11 10:50:29 +03:00
{
int banktext = 0 ;
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : CREATURE_BANK :
banktext = 32 ;
break ;
case Obj : : DERELICT_SHIP :
banktext = 41 ;
break ;
case Obj : : DRAGON_UTOPIA :
banktext = 47 ;
break ;
case Obj : : CRYPT :
banktext = 119 ;
break ;
case Obj : : SHIPWRECK :
banktext = 122 ;
break ;
2009-08-11 10:50:29 +03:00
}
BlockingDialog bd ( true , false ) ;
bd . player = h - > getOwner ( ) ;
2011-06-01 08:10:28 +03:00
bd . soundID = soundBase : : ROGUE ;
2009-09-17 17:27:28 +03:00
bd . text < < VLC - > generaltexth - > advobtxt [ banktext ] ;
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : CREATURE_BANK )
2010-11-22 02:34:46 +02:00
bd . text . addReplacement ( VLC - > objh - > creBanksNames [ index ] ) ;
2009-08-11 10:50:29 +03:00
cb - > showBlockingDialog ( & bd , boost : : bind ( & CBank : : fightGuards , this , h , _1 ) ) ;
}
else
{
InfoWindow iw ;
iw . soundID = soundBase : : GRAVEYARD ;
iw . player = h - > getOwner ( ) ;
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : CRYPT ) //morale penalty for empty Crypt
2010-03-24 16:58:17 +02:00
{
GiveBonus gbonus ;
gbonus . id = h - > id ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . duration = Bonus : : ONE_BATTLE ;
gbonus . bonus . source = Bonus : : OBJECT ;
2011-02-20 20:32:39 +02:00
gbonus . bonus . sid = ID ;
2010-03-24 16:58:17 +02:00
gbonus . bdescr < < " \n " < < VLC - > generaltexth - > arraytxt [ 98 ] ;
2010-05-02 21:20:26 +03:00
gbonus . bonus . type = Bonus : : MORALE ;
2010-03-24 16:58:17 +02:00
gbonus . bonus . val = - 1 ;
cb - > giveHeroBonus ( & gbonus ) ;
iw . text < < VLC - > generaltexth - > advobtxt [ 120 ] ;
iw . components . push_back ( Component ( Component : : MORALE , 0 , - 1 , 0 ) ) ;
}
else
{
iw . text < < VLC - > generaltexth - > advobtxt [ 33 ] ;
2010-11-22 02:34:46 +02:00
iw . text . addReplacement ( VLC - > objh - > creBanksNames [ index ] ) ;
2010-03-24 16:58:17 +02:00
}
2009-04-16 03:28:54 +03:00
cb - > showInfoDialog ( & iw ) ;
2009-08-11 10:50:29 +03:00
}
}
2012-07-19 21:52:44 +03:00
void CBank : : fightGuards ( const CGHeroInstance * h , ui32 accept ) const
2009-08-11 10:50:29 +03:00
{
if ( accept )
{
2009-09-13 01:17:23 +03:00
cb - > startBattleI ( h , this , boost : : bind ( & CBank : : endBattle , this , h , _1 ) , true ) ;
2009-08-11 10:50:29 +03:00
}
}
2009-08-19 09:56:53 +03:00
void CBank : : endBattle ( const CGHeroInstance * h , const BattleResult * result ) const
2009-08-11 10:50:29 +03:00
{
if ( result - > winner = = 0 )
{
int textID = - 1 ;
InfoWindow iw ;
2009-09-16 19:16:57 +03:00
iw . player = h - > getOwner ( ) ;
MetaString loot ;
2009-08-11 10:50:29 +03:00
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : CREATURE_BANK : case Obj : : DRAGON_UTOPIA :
textID = 34 ;
break ;
case Obj : : DERELICT_SHIP :
if ( multiplier )
textID = 43 ;
else
{
GiveBonus gbonus ;
gbonus . id = h - > id ;
gbonus . bonus . duration = Bonus : : ONE_BATTLE ;
gbonus . bonus . source = Bonus : : OBJECT ;
gbonus . bonus . sid = ID ;
gbonus . bdescr < < " \n " < < VLC - > generaltexth - > arraytxt [ 101 ] ;
gbonus . bonus . type = Bonus : : MORALE ;
gbonus . bonus . val = - 1 ;
cb - > giveHeroBonus ( & gbonus ) ;
textID = 42 ;
iw . components . push_back ( Component ( Component : : MORALE , 0 , - 1 , 0 ) ) ;
}
break ;
case Obj : : CRYPT :
if ( bc - > resources . size ( ) ! = 0 )
textID = 121 ;
else
{
iw . components . push_back ( Component ( Component : : MORALE , 0 , - 1 , 0 ) ) ;
GiveBonus gbonus ;
gbonus . id = h - > id ;
gbonus . bonus . duration = Bonus : : ONE_BATTLE ;
gbonus . bonus . source = Bonus : : OBJECT ;
gbonus . bonus . sid = ID ;
gbonus . bdescr < < " \n " < < VLC - > generaltexth - > arraytxt [ ID ] ;
gbonus . bonus . type = Bonus : : MORALE ;
gbonus . bonus . val = - 1 ;
cb - > giveHeroBonus ( & gbonus ) ;
textID = 120 ;
iw . components . push_back ( Component ( Component : : MORALE , 0 , - 1 , 0 ) ) ;
}
break ;
case Obj : : SHIPWRECK :
if ( bc - > resources . size ( ) )
textID = 124 ;
else
textID = 123 ;
break ;
2009-08-11 10:50:29 +03:00
}
2009-09-16 19:16:57 +03:00
2009-08-20 13:07:23 +03:00
//grant resources
2010-03-24 16:58:17 +02:00
if ( textID ! = 42 ) //empty derelict ship gives no cash
{
for ( int it = 0 ; it < bc - > resources . size ( ) ; it + + )
2012-07-19 21:52:44 +03:00
{
2010-03-24 16:58:17 +02:00
if ( bc - > resources [ it ] ! = 0 )
{
iw . components . push_back ( Component ( Component : : RESOURCE , it , bc - > resources [ it ] , 0 ) ) ;
loot < < " %d %s " ;
2010-11-22 02:34:46 +02:00
loot . addReplacement ( iw . components . back ( ) . val ) ;
loot . addReplacement ( MetaString : : RES_NAMES , iw . components . back ( ) . subtype ) ;
2010-03-24 16:58:17 +02:00
cb - > giveResource ( h - > getOwner ( ) , it , bc - > resources [ it ] ) ;
2012-07-19 21:52:44 +03:00
}
2010-03-24 16:58:17 +02:00
}
2009-08-11 10:50:29 +03:00
}
2009-08-20 13:07:23 +03:00
//grant artifacts
2009-08-26 08:08:59 +03:00
for ( std : : vector < ui32 > : : const_iterator it = artifacts . begin ( ) ; it ! = artifacts . end ( ) ; it + + )
2009-08-11 10:50:29 +03:00
{
iw . components . push_back ( Component ( Component : : ARTIFACT , * it , 0 , 0 ) ) ;
2009-09-16 19:16:57 +03:00
loot < < " %s " ;
2010-11-22 02:34:46 +02:00
loot . addReplacement ( MetaString : : ART_NAMES , * it ) ;
2010-12-26 16:34:11 +02:00
cb - > giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ * it ] , - 2 ) ;
2009-08-11 10:50:29 +03:00
}
2009-09-16 19:16:57 +03:00
//display loot
2010-03-24 16:58:17 +02:00
if ( ! iw . components . empty ( ) )
2009-09-16 19:16:57 +03:00
{
2009-09-17 09:19:27 +03:00
if ( textID = = 34 )
{
iw . text . addTxt ( MetaString : : ADVOB_TXT , 34 ) ; //Heaving defeated %s, you discover %s
2010-11-22 02:34:46 +02:00
iw . text . addReplacement ( MetaString : : CRE_PL_NAMES , result - > casualties [ 1 ] . begin ( ) - > first ) ;
iw . text . addReplacement ( loot . buildList ( ) ) ;
2009-09-17 09:19:27 +03:00
}
else
iw . text . addTxt ( MetaString : : ADVOB_TXT , textID ) ;
cb - > showInfoDialog ( & iw ) ;
2009-09-16 19:16:57 +03:00
}
2009-09-17 09:19:27 +03:00
loot . clear ( ) ;
iw . components . clear ( ) ;
iw . text . clear ( ) ;
2009-09-16 19:16:57 +03:00
2009-08-20 13:07:23 +03:00
//grant creatures
2009-08-11 10:50:29 +03:00
CCreatureSet ourArmy ;
2009-08-19 09:56:53 +03:00
for ( std : : vector < std : : pair < ui16 , ui32 > > : : const_iterator it = bc - > creatures . begin ( ) ; it ! = bc - > creatures . end ( ) ; it + + )
2009-08-11 10:50:29 +03:00
{
2010-04-02 05:07:40 +03:00
int slot = ourArmy . getSlotFor ( it - > first ) ;
ourArmy . addToSlot ( slot , it - > first , it - > second ) ;
2009-08-11 10:50:29 +03:00
}
2010-05-02 21:20:26 +03:00
for ( TSlots : : const_iterator i = ourArmy . Slots ( ) . begin ( ) ; i ! = ourArmy . Slots ( ) . end ( ) ; i + + )
2009-09-17 09:19:27 +03:00
{
2010-11-22 02:34:46 +02:00
iw . components . push_back ( Component ( * i - > second ) ) ;
2009-09-17 09:19:27 +03:00
loot < < " %s " ;
2010-11-22 02:34:46 +02:00
loot . addReplacement ( * i - > second ) ;
2009-09-17 09:19:27 +03:00
}
2010-05-02 21:20:26 +03:00
if ( ourArmy . Slots ( ) . size ( ) )
2009-09-17 09:19:27 +03:00
{
2010-11-22 02:34:46 +02:00
if ( ourArmy . Slots ( ) . size ( ) = = 1 & & ourArmy . Slots ( ) . begin ( ) - > second - > count = = 1 )
2009-09-17 09:19:27 +03:00
iw . text . addTxt ( MetaString : : ADVOB_TXT , 185 ) ;
else
iw . text . addTxt ( MetaString : : ADVOB_TXT , 186 ) ;
2010-11-22 02:34:46 +02:00
iw . text . addReplacement ( loot . buildList ( ) ) ;
iw . text . addReplacement ( h - > name ) ;
2009-09-17 09:19:27 +03:00
cb - > showInfoDialog ( & iw ) ;
2010-12-13 01:44:16 +02:00
cb - > giveCreatures ( this , h , ourArmy , false ) ;
2009-09-17 09:19:27 +03:00
}
2010-11-09 14:07:57 +02:00
cb - > setObjProperty ( id , 15 , 0 ) ; //bc = NULL
2009-08-11 10:50:29 +03:00
}
2010-11-09 14:33:44 +02:00
2009-08-11 10:50:29 +03:00
}
2009-09-24 20:54:02 +03:00
void CGPyramid : : initObj ( )
{
2009-09-25 07:05:01 +03:00
std : : vector < ui16 > available ;
cb - > getAllowedSpells ( available , 5 ) ;
2010-08-18 16:42:46 +03:00
if ( available . size ( ) )
{
bc = VLC - > objh - > banksInfo [ 21 ] . front ( ) ; //TODO: remove hardcoded value?
spell = ( available [ rand ( ) % available . size ( ) ] ) ;
}
else
{
tlog1 < < " No spells available for Pyramid! Object set to empty. \n " ;
}
setPropertyDer ( 17 , ran ( ) ) ; //set guards at game start
2009-09-24 20:54:02 +03:00
}
2009-09-25 18:40:28 +03:00
const std : : string & CGPyramid : : getHoverText ( ) const
{
2010-08-18 16:42:46 +03:00
hoverName = VLC - > objh - > creBanksNames [ 21 ] ;
2009-09-25 18:40:28 +03:00
if ( bc = = NULL )
hoverName + = " " + VLC - > generaltexth - > allTexts [ 352 ] ;
else
hoverName + = " " + VLC - > generaltexth - > allTexts [ 353 ] ;
return hoverName ;
}
2009-09-24 20:54:02 +03:00
void CGPyramid : : onHeroVisit ( const CGHeroInstance * h ) const
{
if ( bc )
{
BlockingDialog bd ( true , false ) ;
bd . player = h - > getOwner ( ) ;
2010-08-17 18:16:08 +03:00
bd . soundID = soundBase : : MYSTERY ;
2009-09-24 20:54:02 +03:00
bd . text < < VLC - > generaltexth - > advobtxt [ 105 ] ;
2012-07-19 21:52:44 +03:00
cb - > showBlockingDialog ( & bd , boost : : bind ( & CBank : : fightGuards , this , h , _1 ) ) ;
2009-09-24 20:54:02 +03:00
}
else
{
InfoWindow iw ;
2009-09-25 18:40:28 +03:00
iw . player = h - > getOwner ( ) ;
2009-09-24 20:54:02 +03:00
iw . text < < VLC - > generaltexth - > advobtxt [ 107 ] ;
iw . components . push_back ( Component ( Component : : LUCK , 0 , - 2 , 0 ) ) ;
GiveBonus gb ;
2010-05-02 21:20:26 +03:00
gb . bonus = Bonus ( Bonus : : ONE_BATTLE , Bonus : : LUCK , Bonus : : OBJECT , - 2 , id , VLC - > generaltexth - > arraytxt [ 70 ] ) ;
2010-02-10 04:56:00 +02:00
gb . id = h - > id ;
2009-09-24 20:54:02 +03:00
cb - > giveHeroBonus ( & gb ) ;
cb - > showInfoDialog ( & iw ) ;
}
}
void CGPyramid : : endBattle ( const CGHeroInstance * h , const BattleResult * result ) const
{
if ( result - > winner = = 0 )
{
InfoWindow iw ;
2009-09-25 18:40:28 +03:00
iw . player = h - > getOwner ( ) ;
2009-09-24 20:54:02 +03:00
iw . text . addTxt ( MetaString : : ADVOB_TXT , 106 ) ;
iw . text . addTxt ( MetaString : : SPELL_NAME , spell ) ;
2012-07-19 21:52:44 +03:00
if ( ! h - > getArt ( 17 ) )
2009-09-25 18:40:28 +03:00
iw . text . addTxt ( MetaString : : ADVOB_TXT , 109 ) ; //no spellbook
2012-07-19 21:52:44 +03:00
else if ( h - > getSecSkillLevel ( CGHeroInstance : : WISDOM ) < 3 )
2009-09-25 18:40:28 +03:00
iw . text . addTxt ( MetaString : : ADVOB_TXT , 108 ) ; //no expert Wisdom
2009-09-24 20:54:02 +03:00
else
{
std : : set < ui32 > spells ;
spells . insert ( spell ) ;
cb - > changeSpells ( h - > id , true , spells ) ;
iw . components . push_back ( Component ( Component : : SPELL , spell , 0 , 0 ) ) ;
}
cb - > showInfoDialog ( & iw ) ;
2009-09-25 18:40:28 +03:00
cb - > setObjProperty ( id , 15 , 0 ) ;
2009-09-24 20:54:02 +03:00
}
}
2009-08-11 10:50:29 +03:00
void CGKeys : : setPropertyDer ( ui8 what , ui32 val ) //101-108 - enable key for player 1-8
{
2011-12-14 00:23:17 +03:00
if ( what > = 101 & & what < = ( 100 + GameConstants : : PLAYER_LIMIT ) )
2010-07-06 05:10:26 +03:00
playerKeyMap . find ( what - 101 ) - > second . insert ( ( ui8 ) val ) ;
2009-08-11 10:50:29 +03:00
}
2009-04-16 03:28:54 +03:00
2009-08-11 10:50:29 +03:00
bool CGKeys : : wasMyColorVisited ( int player ) const
{
if ( vstd : : contains ( playerKeyMap [ player ] , subID ) ) //creates set if it's not there
return true ;
else
return false ;
}
2012-07-08 19:36:20 +03:00
const std : : string CGKeys : : getName ( ) const
{
std : : string name ;
name = VLC - > generaltexth - > tentColors [ subID ] + " " + VLC - > generaltexth - > names [ ID ] ;
return name ;
}
2009-08-11 10:50:29 +03:00
const std : : string & CGKeymasterTent : : getHoverText ( ) const
{
2012-07-08 19:36:20 +03:00
hoverName = getName ( ) ;
2009-08-11 10:50:29 +03:00
if ( wasMyColorVisited ( cb - > getCurrentPlayer ( ) ) ) //TODO: use local player, not current
hoverName + = " \n " + VLC - > generaltexth - > allTexts [ 352 ] ;
else
hoverName + = " \n " + VLC - > generaltexth - > allTexts [ 353 ] ;
return hoverName ;
}
2012-03-14 16:02:38 +03:00
bool CGKeymasterTent : : wasVisited ( ui8 player ) const
{
return wasMyColorVisited ( player ) ;
}
2009-08-11 10:50:29 +03:00
void CGKeymasterTent : : onHeroVisit ( const CGHeroInstance * h ) const
{
InfoWindow iw ;
iw . soundID = soundBase : : CAVEHEAD ;
iw . player = h - > getOwner ( ) ;
if ( ! wasMyColorVisited ( h - > getOwner ( ) ) )
{
cb - > setObjProperty ( id , h - > tempOwner + 101 , subID ) ;
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 19 ) ;
2009-04-16 03:28:54 +03:00
}
2009-08-11 10:50:29 +03:00
else
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 20 ) ;
cb - > showInfoDialog ( & iw ) ;
}
void CGBorderGuard : : initObj ( )
{
2012-07-19 21:52:44 +03:00
//ui32 m13489val = subID; //store color as quest info
2009-08-11 10:50:29 +03:00
blockVisit = true ;
2009-05-24 01:57:39 +03:00
}
2009-07-19 04:00:19 +03:00
2009-08-11 10:50:29 +03:00
const std : : string & CGBorderGuard : : getHoverText ( ) const
{
2012-07-08 19:36:20 +03:00
hoverName = getName ( ) ;
2009-08-11 10:50:29 +03:00
if ( wasMyColorVisited ( cb - > getCurrentPlayer ( ) ) ) //TODO: use local player, not current
hoverName + = " \n " + VLC - > generaltexth - > allTexts [ 352 ] ;
else
hoverName + = " \n " + VLC - > generaltexth - > allTexts [ 353 ] ;
return hoverName ;
}
2012-07-08 19:36:20 +03:00
void CGBorderGuard : : getVisitText ( MetaString & text , std : : vector < Component > & components , bool isCustom , bool FirstVisit , const CGHeroInstance * h ) const
{
text < < std : : pair < ui8 , ui32 > ( 11 , 18 ) ;
}
void CGBorderGuard : : getRolloverText ( MetaString & text , bool onHover ) const
{
if ( ! onHover )
text < < VLC - > generaltexth - > tentColors [ subID ] < < " " < < VLC - > generaltexth - > names [ Obj : : KEYMASTER ] ;
}
2012-07-19 12:10:55 +03:00
bool CGBorderGuard : : checkQuest ( const CGHeroInstance * h ) const
{
return wasMyColorVisited ( h - > tempOwner ) ;
}
2012-07-19 21:52:44 +03:00
void CGBorderGuard : : onHeroVisit ( const CGHeroInstance * h ) const
2009-08-11 10:50:29 +03:00
{
if ( wasMyColorVisited ( h - > getOwner ( ) ) )
{
BlockingDialog bd ( true , false ) ;
bd . player = h - > getOwner ( ) ;
bd . soundID = soundBase : : QUEST ;
bd . text . addTxt ( MetaString : : ADVOB_TXT , 17 ) ;
2012-07-19 21:52:44 +03:00
cb - > showBlockingDialog ( & bd , boost : : bind ( & CGBorderGuard : : openGate , this , h , _1 ) ) ;
}
2009-08-11 10:50:29 +03:00
else
{
InfoWindow iw ;
iw . player = h - > getOwner ( ) ;
iw . soundID = soundBase : : CAVEHEAD ;
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 18 ) ;
cb - > showInfoDialog ( & iw ) ;
2012-07-06 22:12:04 +03:00
2012-07-08 19:36:20 +03:00
AddQuest aq ;
2012-10-03 17:49:29 +03:00
aq . quest = QuestInfo ( quest , this , visitablePos ( ) ) ;
2012-07-08 19:36:20 +03:00
aq . player = h - > tempOwner ;
cb - > sendAndApply ( & aq ) ;
//TODO: add this quest only once OR check for multiple instances later
2009-08-11 10:50:29 +03:00
}
}
void CGBorderGuard : : openGate ( const CGHeroInstance * h , ui32 accept ) const
{
if ( accept )
cb - > removeObject ( id ) ;
}
2012-07-19 21:52:44 +03:00
void CGBorderGate : : onHeroVisit ( const CGHeroInstance * h ) const //TODO: passability
2009-08-11 10:50:29 +03:00
{
if ( ! wasMyColorVisited ( h - > getOwner ( ) ) )
{
2009-12-20 19:14:14 +02:00
InfoWindow iw ;
iw . player = h - > getOwner ( ) ;
2009-08-11 10:50:29 +03:00
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 18 ) ;
cb - > showInfoDialog ( & iw ) ;
2012-07-08 19:36:20 +03:00
AddQuest aq ;
2012-10-03 17:49:29 +03:00
aq . quest = QuestInfo ( quest , this , visitablePos ( ) ) ;
2012-07-08 19:36:20 +03:00
aq . player = h - > tempOwner ;
cb - > sendAndApply ( & aq ) ;
2009-08-11 10:50:29 +03:00
}
}
2009-12-20 19:14:14 +02:00
ui8 CGBorderGate : : getPassableness ( ) const
{
ui8 ret = 0 ;
2011-12-14 00:23:17 +03:00
for ( int i = 0 ; i < GameConstants : : PLAYER_LIMIT ; i + + )
2009-12-20 19:14:14 +02:00
ret | = wasMyColorVisited ( i ) < < i ;
return ret ;
}
2009-08-11 10:50:29 +03:00
void CGMagi : : initObj ( )
{
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : EYE_OF_MAGI )
2009-08-11 10:50:29 +03:00
{
blockVisit = true ;
2009-08-13 04:03:11 +03:00
eyelist [ subID ] . push_back ( id ) ;
2009-08-11 10:50:29 +03:00
}
}
void CGMagi : : onHeroVisit ( const CGHeroInstance * h ) const
{
2012-09-23 21:01:04 +03:00
if ( ID = = Obj : : HUT_OF_MAGI )
2009-08-11 10:50:29 +03:00
{
InfoWindow iw ;
2010-08-17 19:26:48 +03:00
CenterView cv ;
FoWChange fw ;
2010-08-18 13:52:30 +03:00
cv . player = iw . player = fw . player = h - > tempOwner ;
iw . soundID = soundBase : : LIGHTHOUSE ;
iw . player = h - > tempOwner ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , 61 ) ;
cb - > showInfoDialog ( & iw ) ;
2009-08-11 10:50:29 +03:00
fw . mode = 1 ;
2009-08-13 04:03:11 +03:00
std : : vector < si32 > : : iterator it ;
for ( it = eyelist [ subID ] . begin ( ) ; it < eyelist [ subID ] . end ( ) ; it + + )
{
const CGObjectInstance * eye = cb - > getObj ( * it ) ;
2010-06-21 07:43:10 +03:00
cb - > getTilesInRange ( fw . tiles , eye - > pos , 10 , h - > tempOwner , 1 ) ;
2009-08-11 10:50:29 +03:00
cb - > sendAndApply ( & fw ) ;
2009-08-13 04:03:11 +03:00
cv . pos = eye - > pos ;
cv . focusTime = 2000 ;
2009-08-11 10:50:29 +03:00
cb - > sendAndApply ( & cv ) ;
2012-07-19 21:52:44 +03:00
}
2009-08-13 04:03:11 +03:00
cv . pos = h - > getPosition ( false ) ;
2012-07-19 21:52:44 +03:00
cb - > sendAndApply ( & cv ) ;
2010-08-18 13:52:30 +03:00
}
2012-09-23 21:01:04 +03:00
else if ( ID = = Obj : : EYE_OF_MAGI )
2010-08-18 13:52:30 +03:00
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , 48 ) ;
cb - > showInfoDialog ( & iw ) ;
}
2010-08-17 19:26:48 +03:00
2010-08-18 13:52:30 +03:00
}
2009-07-19 04:00:19 +03:00
void CGBoat : : initObj ( )
{
hero = NULL ;
2009-07-19 10:16:33 +03:00
}
void CGSirens : : initObj ( )
{
blockVisit = true ;
}
const std : : string & CGSirens : : getHoverText ( ) const
{
getNameVis ( hoverName ) ;
return hoverName ;
}
void CGSirens : : onHeroVisit ( const CGHeroInstance * h ) const
{
InfoWindow iw ;
iw . soundID = soundBase : : DANGER ;
iw . player = h - > tempOwner ;
2010-05-02 21:20:26 +03:00
if ( h - > hasBonusFrom ( Bonus : : OBJECT , ID ) ) //has already visited Sirens
2009-07-19 10:16:33 +03:00
{
iw . text . addTxt ( 11 , 133 ) ;
}
else
{
2010-05-02 21:20:26 +03:00
giveDummyBonus ( h - > id , Bonus : : ONE_BATTLE ) ;
2009-07-19 10:16:33 +03:00
int xp = 0 ;
2010-11-27 22:17:28 +02:00
2010-05-02 21:20:26 +03:00
for ( TSlots : : const_iterator i = h - > Slots ( ) . begin ( ) ; i ! = h - > Slots ( ) . end ( ) ; i + + )
2009-07-19 10:16:33 +03:00
{
2010-11-27 22:17:28 +02:00
TQuantity drown = i - > second - > count * 0.3 ;
2009-07-19 10:16:33 +03:00
if ( drown )
{
2010-11-27 22:17:28 +02:00
cb - > changeStackCount ( StackLocation ( h , i - > first ) , - drown ) ;
2010-11-22 02:34:46 +02:00
xp + = drown * i - > second - > type - > valOfBonuses ( Bonus : : STACK_HEALTH ) ;
2009-07-19 10:16:33 +03:00
}
}
if ( xp )
{
2011-02-12 20:48:11 +02:00
xp = h - > calculateXp ( xp ) ;
2009-07-19 10:16:33 +03:00
iw . text . addTxt ( 11 , 132 ) ;
iw . text . addReplacement ( xp ) ;
2011-09-06 12:59:06 +03:00
cb - > changePrimSkill ( h - > id , 4 , xp , false ) ;
2009-07-19 10:16:33 +03:00
}
else
{
iw . text . addTxt ( 11 , 134 ) ;
}
}
cb - > showInfoDialog ( & iw ) ;
2011-02-12 20:48:11 +02:00
2009-07-26 06:33:13 +03:00
}
//bool IShipyard::validLocation() const
//{
// std::vector<int3> offsets;
// getOutOffsets(offsets);
//
// TerrainTile *tile;
// for(int i = 0; i < offsets.size(); i++)
2012-11-06 19:39:29 +03:00
// if((tile = IObjectInterface::cb->getTile(o->pos + offsets[i])) && tile->terType == TerrainTile::water) //tile is in the map and is water
2009-07-26 06:33:13 +03:00
// return true;
// return false;
//}
2010-03-11 01:16:30 +02:00
int3 IBoatGenerator : : bestLocation ( ) const
2009-07-26 06:33:13 +03:00
{
std : : vector < int3 > offsets ;
getOutOffsets ( offsets ) ;
2010-06-26 15:57:16 +03:00
for ( int i = 0 ; i < offsets . size ( ) ; + + i )
{
2011-05-25 17:48:49 +03:00
if ( const TerrainTile * tile = IObjectInterface : : cb - > getTile ( o - > pos + offsets [ i ] , false ) ) //tile is in the map
2010-06-26 15:57:16 +03:00
{
2012-11-06 19:39:29 +03:00
if ( tile - > terType = = ETerrainType : : WATER & & ( ! tile - > blocked | | tile - > blockingObjects . front ( ) - > ID = = 8 ) ) //and is water and is not blocked or is blocked by boat
2010-06-26 15:57:16 +03:00
return o - > pos + offsets [ i ] ;
}
}
return int3 ( - 1 , - 1 , - 1 ) ;
2009-07-26 06:33:13 +03:00
}
2012-09-23 21:01:04 +03:00
IBoatGenerator : : EGeneratorState IBoatGenerator : : state ( ) const
2009-07-26 06:33:13 +03:00
{
int3 tile = bestLocation ( ) ;
2011-05-10 01:20:47 +03:00
const TerrainTile * t = IObjectInterface : : cb - > getTile ( tile ) ;
2009-07-26 06:33:13 +03:00
if ( ! t )
2012-09-23 21:01:04 +03:00
return TILE_BLOCKED ; //no available water
2009-07-26 06:33:13 +03:00
else if ( ! t - > blockingObjects . size ( ) )
2012-09-23 21:01:04 +03:00
return GOOD ; //OK
else if ( t - > blockingObjects . front ( ) - > ID = = Obj : : BOAT )
return BOAT_ALREADY_BUILT ; //blocked with boat
2009-07-26 06:33:13 +03:00
else
2012-09-23 21:01:04 +03:00
return TILE_BLOCKED ; //blocked
2009-07-26 06:33:13 +03:00
}
2010-03-11 01:16:30 +02:00
int IBoatGenerator : : getBoatType ( ) const
{
//We make good ships by default
return 1 ;
}
2012-07-19 21:52:44 +03:00
IBoatGenerator : : IBoatGenerator ( const CGObjectInstance * O )
2010-03-11 01:16:30 +02:00
: o ( O )
{
}
2010-07-13 08:25:40 +03:00
void IBoatGenerator : : getProblemText ( MetaString & out , const CGHeroInstance * visitor ) const
{
switch ( state ( ) )
{
2012-09-23 21:01:04 +03:00
case BOAT_ALREADY_BUILT :
2010-07-13 08:25:40 +03:00
out . addTxt ( MetaString : : GENERAL_TXT , 51 ) ;
break ;
2012-09-23 21:01:04 +03:00
case TILE_BLOCKED :
2010-07-13 08:25:40 +03:00
if ( visitor )
{
out . addTxt ( MetaString : : GENERAL_TXT , 134 ) ;
out . addReplacement ( visitor - > name ) ;
}
else
out . addTxt ( MetaString : : ADVOB_TXT , 189 ) ;
break ;
2012-09-23 21:01:04 +03:00
case NO_WATER :
2010-07-13 08:25:40 +03:00
tlog1 < < " Shipyard without water!!! " < < o - > pos < < " \t " < < o - > id < < std : : endl ;
return ;
}
}
2010-03-11 01:16:30 +02:00
void IShipyard : : getBoatCost ( std : : vector < si32 > & cost ) const
{
2011-12-14 00:23:17 +03:00
cost . resize ( GameConstants : : RESOURCE_QUANTITY ) ;
2010-03-11 01:16:30 +02:00
cost [ 0 ] = 10 ;
cost [ 6 ] = 1000 ;
}
2012-07-19 21:52:44 +03:00
IShipyard : : IShipyard ( const CGObjectInstance * O )
2010-03-11 01:16:30 +02:00
: IBoatGenerator ( O )
{
}
2009-07-26 06:33:13 +03:00
IShipyard * IShipyard : : castFrom ( CGObjectInstance * obj )
{
2010-07-13 08:25:40 +03:00
if ( ! obj )
return NULL ;
2012-09-23 21:01:04 +03:00
if ( obj - > ID = = Obj : : TOWN )
2009-07-26 06:33:13 +03:00
{
return static_cast < CGTownInstance * > ( obj ) ;
}
2012-09-23 21:01:04 +03:00
else if ( obj - > ID = = Obj : : SHIPYARD )
2009-07-26 06:33:13 +03:00
{
return static_cast < CGShipyard * > ( obj ) ;
}
else
{
2010-07-13 08:25:40 +03:00
//tlog1 << "Cannot cast to IShipyard object with ID " << obj->ID << std::endl;
2009-07-26 06:33:13 +03:00
return NULL ;
}
}
const IShipyard * IShipyard : : castFrom ( const CGObjectInstance * obj )
{
return castFrom ( const_cast < CGObjectInstance * > ( obj ) ) ;
}
CGShipyard : : CGShipyard ( )
: IShipyard ( this )
{
}
void CGShipyard : : getOutOffsets ( std : : vector < int3 > & offsets ) const
{
2010-07-13 08:25:40 +03:00
// H J L K I
// A x S x B
// C E G F D
offsets + = int3 ( - 3 , 0 , 0 ) , int3 ( 1 , 0 , 0 ) , //AB
int3 ( - 3 , 1 , 0 ) , int3 ( 1 , 1 , 0 ) , int3 ( - 2 , 1 , 0 ) , int3 ( 0 , 1 , 0 ) , int3 ( - 1 , 1 , 0 ) , //CDEFG
int3 ( - 3 , - 1 , 0 ) , int3 ( 1 , - 1 , 0 ) , int3 ( - 2 , - 1 , 0 ) , int3 ( 0 , - 1 , 0 ) , int3 ( - 1 , - 1 , 0 ) ; //HIJKL
2009-07-26 06:33:13 +03:00
}
void CGShipyard : : onHeroVisit ( const CGHeroInstance * h ) const
{
2010-08-13 13:46:08 +03:00
if ( ! cb - > gameState ( ) - > getPlayerRelations ( tempOwner , h - > tempOwner ) )
2009-07-26 06:33:13 +03:00
cb - > setOwner ( id , h - > tempOwner ) ;
2012-09-23 21:01:04 +03:00
auto s = state ( ) ;
2012-09-30 13:35:17 +03:00
if ( s ! = IBoatGenerator : : GOOD )
2009-07-26 06:33:13 +03:00
{
InfoWindow iw ;
iw . player = tempOwner ;
2010-07-13 08:25:40 +03:00
getProblemText ( iw . text , h ) ;
2009-07-26 06:33:13 +03:00
cb - > showInfoDialog ( & iw ) ;
}
else
{
OpenWindow ow ;
ow . id1 = id ;
ow . id2 = h - > id ;
ow . window = OpenWindow : : SHIPYARD_WINDOW ;
cb - > sendAndApply ( & ow ) ;
}
2009-08-12 05:02:58 +03:00
}
2009-08-19 09:56:53 +03:00
2012-07-19 21:52:44 +03:00
void CCartographer : : onHeroVisit ( const CGHeroInstance * h ) const
2009-08-19 09:56:53 +03:00
{
2012-03-14 16:02:38 +03:00
if ( ! wasVisited ( h - > getOwner ( ) ) ) //if hero has not visited yet this cartographer
2009-08-19 09:56:53 +03:00
{
2009-08-19 13:59:42 +03:00
if ( cb - > getResource ( h - > tempOwner , 6 ) > = 1000 ) //if he can afford a map
2009-08-19 09:56:53 +03:00
{
2009-08-19 13:59:42 +03:00
//ask if he wants to buy one
2012-07-06 23:19:54 +03:00
int text = 0 ;
2009-12-30 12:02:31 +02:00
switch ( subID )
2009-08-19 09:56:53 +03:00
{
2009-12-30 12:02:31 +02:00
case 0 :
text = 25 ;
break ;
case 1 :
2009-08-19 09:56:53 +03:00
text = 26 ;
2009-12-30 12:02:31 +02:00
break ;
case 2 :
2009-08-19 09:56:53 +03:00
text = 27 ;
2009-12-30 12:02:31 +02:00
break ;
2012-07-19 21:52:44 +03:00
default :
2009-12-30 12:02:31 +02:00
tlog2 < < " Unrecognized subtype of cartographer " < < std : : endl ;
2009-08-19 09:56:53 +03:00
}
2012-07-06 23:19:54 +03:00
assert ( text ) ;
2009-08-19 09:56:53 +03:00
BlockingDialog bd ( true , false ) ;
bd . player = h - > getOwner ( ) ;
bd . soundID = soundBase : : LIGHTHOUSE ;
bd . text . addTxt ( MetaString : : ADVOB_TXT , text ) ;
cb - > showBlockingDialog ( & bd , boost : : bind ( & CCartographer : : buyMap , this , h , _1 ) ) ;
}
2009-08-19 13:59:42 +03:00
else //if he cannot afford
2009-08-19 09:56:53 +03:00
{
InfoWindow iw ;
iw . player = h - > getOwner ( ) ;
iw . soundID = soundBase : : CAVEHEAD ;
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 28 ) ;
cb - > showInfoDialog ( & iw ) ;
}
2012-07-19 21:52:44 +03:00
}
2009-08-19 13:59:42 +03:00
else //if he already visited carographer
2009-08-19 09:56:53 +03:00
{
InfoWindow iw ;
iw . player = h - > getOwner ( ) ;
iw . soundID = soundBase : : CAVEHEAD ;
iw . text < < std : : pair < ui8 , ui32 > ( 11 , 24 ) ;
cb - > showInfoDialog ( & iw ) ;
}
}
2009-08-19 13:59:42 +03:00
2009-08-19 09:56:53 +03:00
void CCartographer : : buyMap ( const CGHeroInstance * h , ui32 accept ) const
{
2009-08-19 13:59:42 +03:00
if ( accept ) //if hero wants to buy map
2009-08-19 09:56:53 +03:00
{
2009-08-19 10:03:12 +03:00
cb - > giveResource ( h - > tempOwner , 6 , - 1000 ) ;
2009-08-19 09:56:53 +03:00
FoWChange fw ;
2010-07-20 12:16:48 +03:00
fw . mode = 1 ;
2009-08-19 09:56:53 +03:00
fw . player = h - > tempOwner ;
2009-11-14 22:14:15 +02:00
//subIDs of different types of cartographers:
//water = 0; land = 1; underground = 2;
cb - > getAllTiles ( fw . tiles , h - > tempOwner , subID - 1 , ! subID + 1 ) ; //reveal appropriate tiles
2009-08-19 09:56:53 +03:00
cb - > sendAndApply ( & fw ) ;
cb - > setObjProperty ( id , 10 , h - > tempOwner ) ;
}
2009-08-19 13:59:42 +03:00
}
2009-09-21 12:00:33 +03:00
2010-02-06 15:49:14 +02:00
void CGDenOfthieves : : onHeroVisit ( const CGHeroInstance * h ) const
{
2012-03-07 17:05:54 +03:00
cb - > showThievesGuildWindow ( h - > tempOwner , id ) ;
2010-02-10 04:56:00 +02:00
}
void CGObelisk : : onHeroVisit ( const CGHeroInstance * h ) const
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
2010-08-06 16:14:10 +03:00
TeamState * ts = cb - > gameState ( ) - > getPlayerTeam ( h - > tempOwner ) ;
assert ( ts ) ;
int team = ts - > id ;
2012-07-19 21:52:44 +03:00
2012-03-14 16:02:38 +03:00
if ( ! wasVisited ( team ) )
2010-02-10 04:56:00 +02:00
{
iw . text . addTxt ( MetaString : : ADVOB_TXT , 96 ) ;
cb - > sendAndApply ( & iw ) ;
2010-08-06 16:14:10 +03:00
cb - > setObjProperty ( id , 20 , team ) ; //increment general visited obelisks counter
2010-02-10 04:56:00 +02:00
OpenWindow ow ;
ow . id1 = h - > tempOwner ;
ow . window = OpenWindow : : PUZZLE_MAP ;
cb - > sendAndApply ( & ow ) ;
2010-08-06 16:14:10 +03:00
cb - > setObjProperty ( id , 10 , team ) ; //mark that particular obelisk as visited
2010-02-10 04:56:00 +02:00
}
else
{
iw . text . addTxt ( MetaString : : ADVOB_TXT , 97 ) ;
cb - > sendAndApply ( & iw ) ;
}
}
void CGObelisk : : initObj ( )
{
obeliskCount + + ;
}
const std : : string & CGObelisk : : getHoverText ( ) const
{
hoverName = VLC - > generaltexth - > names [ ID ] ;
2012-03-14 16:02:38 +03:00
if ( wasVisited ( cb - > getCurrentPlayer ( ) ) )
2010-02-10 04:56:00 +02:00
hoverName + = " " + VLC - > generaltexth - > allTexts [ 352 ] ; //not visited
else
hoverName + = " " + VLC - > generaltexth - > allTexts [ 353 ] ; //visited
return hoverName ;
}
void CGObelisk : : setPropertyDer ( ui8 what , ui32 val )
{
CPlayersVisited : : setPropertyDer ( what , val ) ;
switch ( what )
{
case 20 :
2011-12-14 00:23:17 +03:00
assert ( val < GameConstants : : PLAYER_LIMIT ) ;
2010-02-10 04:56:00 +02:00
visited [ val ] + + ;
2010-07-14 04:08:27 +03:00
if ( visited [ val ] > obeliskCount )
{
tlog0 < < " Error: Visited " < < visited [ val ] < < " \t \t " < < obeliskCount < < std : : endl ;
assert ( 0 ) ;
}
2010-02-10 04:56:00 +02:00
break ;
}
}
void CGLighthouse : : onHeroVisit ( const CGHeroInstance * h ) const
{
if ( h - > tempOwner ! = tempOwner )
{
ui8 oldOwner = tempOwner ;
cb - > setOwner ( id , h - > tempOwner ) ; //not ours? flag it!
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , 69 ) ;
iw . soundID = soundBase : : LIGHTHOUSE ;
cb - > sendAndApply ( & iw ) ;
giveBonusTo ( h - > tempOwner ) ;
2011-12-14 00:23:17 +03:00
if ( oldOwner < GameConstants : : PLAYER_LIMIT ) //remove bonus from old owner
2010-02-10 04:56:00 +02:00
{
RemoveBonus rb ( RemoveBonus : : PLAYER ) ;
rb . whoID = oldOwner ;
2010-05-02 21:20:26 +03:00
rb . source = Bonus : : OBJECT ;
2010-02-10 04:56:00 +02:00
rb . id = id ;
cb - > sendAndApply ( & rb ) ;
}
}
}
void CGLighthouse : : initObj ( )
{
2011-12-14 00:23:17 +03:00
if ( tempOwner < GameConstants : : PLAYER_LIMIT )
2010-02-10 04:56:00 +02:00
{
giveBonusTo ( tempOwner ) ;
}
}
const std : : string & CGLighthouse : : getHoverText ( ) const
{
hoverName = VLC - > generaltexth - > names [ ID ] ;
//TODO: owned by %s player
return hoverName ;
}
void CGLighthouse : : giveBonusTo ( ui8 player ) const
{
GiveBonus gb ( GiveBonus : : PLAYER ) ;
2010-05-02 21:20:26 +03:00
gb . bonus . type = Bonus : : SEA_MOVEMENT ;
2010-02-10 04:56:00 +02:00
gb . bonus . val = 500 ;
gb . id = player ;
2010-05-02 21:20:26 +03:00
gb . bonus . duration = Bonus : : PERMANENT ;
gb . bonus . source = Bonus : : OBJECT ;
2011-02-20 20:32:39 +02:00
gb . bonus . sid = id ;
2010-02-10 04:56:00 +02:00
cb - > sendAndApply ( & gb ) ;
2010-05-02 21:20:26 +03:00
}
void CArmedInstance : : randomizeArmy ( int type )
{
int max = VLC - > creh - > creatures . size ( ) ;
2011-01-21 04:36:30 +02:00
for ( TSlots : : iterator j = stacks . begin ( ) ; j ! = stacks . end ( ) ; j + + )
2010-05-02 21:20:26 +03:00
{
2010-11-22 02:34:46 +02:00
int randID = j - > second - > idRand ;
if ( randID > max )
2010-05-02 21:20:26 +03:00
{
2012-09-28 13:17:07 +03:00
int level = ( randID - VLC - > creh - > creatures . size ( ) ) / 2 - 1 ;
bool upgrade = ! ( randID % 2 ) ;
j - > second - > setType ( VLC - > townh - > towns [ type ] . creatures [ level ] [ upgrade ] ) ;
2010-11-22 02:34:46 +02:00
randID = - 1 ;
2010-05-02 21:20:26 +03:00
}
2010-11-22 02:34:46 +02:00
assert ( j - > second - > armyObj = = this ) ;
2010-05-02 21:20:26 +03:00
}
return ;
}
CArmedInstance : : CArmedInstance ( )
{
battle = NULL ;
}
2010-10-02 12:43:22 +03:00
int CArmedInstance : : valOfGlobalBonuses ( CSelector selector ) const
2010-09-25 23:23:55 +03:00
{
//if (tempOwner != NEUTRAL_PLAYER)
2010-11-13 22:26:15 +02:00
return cb - > gameState ( ) - > players [ tempOwner ] . valOfBonuses ( selector ) ;
2010-09-25 23:23:55 +03:00
}
2011-02-04 16:58:14 +02:00
void CArmedInstance : : updateMoraleBonusFromArmy ( )
{
if ( ! validTypes ( false ) ) //object not randomized, don't bother
return ;
2011-07-13 21:39:02 +03:00
Bonus * b = getBonusList ( ) . getFirst ( Selector : : sourceType ( Bonus : : ARMY ) & & Selector : : type ( Bonus : : MORALE ) ) ;
2011-02-04 16:58:14 +02:00
if ( ! b )
{
b = new Bonus ( Bonus : : PERMANENT , Bonus : : MORALE , Bonus : : ARMY , 0 , - 1 ) ;
addNewBonus ( b ) ;
}
//number of alignments and presence of undead
bool canMix = hasBonusOfType ( Bonus : : NONEVIL_ALIGNMENT_MIX ) ;
std : : set < si8 > factions ;
for ( TSlots : : const_iterator i = Slots ( ) . begin ( ) ; i ! = Slots ( ) . end ( ) ; i + + )
{
// Take Angelic Alliance troop-mixing freedom of non-evil, non-Conflux units into account.
const si8 faction = i - > second - > type - > faction ;
if ( canMix
& & ( ( faction > = 0 & & faction < = 2 ) | | faction = = 6 | | faction = = 7 ) )
{
factions . insert ( 0 ) ; // Insert a single faction of the affected group, Castle will do.
}
else
{
factions . insert ( faction ) ;
}
}
2012-07-19 21:52:44 +03:00
2011-02-04 16:58:14 +02:00
if ( factions . size ( ) = = 1 )
{
b - > val = + 1 ;
b - > description = VLC - > generaltexth - > arraytxt [ 115 ] ; //All troops of one alignment +1
}
else
{
b - > val = 2 - factions . size ( ) ;
b - > description = boost : : str ( boost : : format ( VLC - > generaltexth - > arraytxt [ 114 ] ) % factions . size ( ) % b - > val ) ; //Troops of %d alignments %d
}
2011-02-22 11:47:25 +02:00
boost : : algorithm : : trim ( b - > description ) ;
2012-07-19 21:52:44 +03:00
2011-02-21 06:13:00 +02:00
//-1 modifier for any Necropolis unit in army
const ui8 UNDEAD_MODIFIER_ID = - 2 ;
2011-07-13 21:39:02 +03:00
Bonus * undeadModifier = getBonusList ( ) . getFirst ( Selector : : source ( Bonus : : ARMY , UNDEAD_MODIFIER_ID ) ) ;
2012-09-05 15:49:23 +03:00
if ( vstd : : contains ( factions , ETownType : : NECROPOLIS ) )
2011-02-21 06:13:00 +02:00
{
if ( ! undeadModifier )
addNewBonus ( new Bonus ( Bonus : : PERMANENT , Bonus : : MORALE , Bonus : : ARMY , - 1 , UNDEAD_MODIFIER_ID , VLC - > generaltexth - > arraytxt [ 116 ] ) ) ;
}
else if ( undeadModifier )
removeBonus ( undeadModifier ) ;
2012-07-19 21:52:44 +03:00
2011-02-04 16:58:14 +02:00
}
void CArmedInstance : : armyChanged ( )
{
updateMoraleBonusFromArmy ( ) ;
}
2011-02-22 11:47:25 +02:00
CBonusSystemNode * CArmedInstance : : whereShouldBeAttached ( CGameState * gs )
{
2011-12-14 00:23:17 +03:00
if ( tempOwner < GameConstants : : PLAYER_LIMIT )
2011-02-22 11:47:25 +02:00
return gs - > getPlayer ( tempOwner ) ;
else
return & gs - > globalEffects ;
}
CBonusSystemNode * CArmedInstance : : whatShouldBeAttached ( )
{
return this ;
}
2011-12-14 00:23:17 +03:00
bool IMarket : : getOffer ( int id1 , int id2 , int & val1 , int & val2 , EMarketMode : : EMarketMode mode ) const
2010-05-18 10:01:54 +03:00
{
switch ( mode )
{
2011-12-14 00:23:17 +03:00
case EMarketMode : : RESOURCE_RESOURCE :
2010-05-18 10:01:54 +03:00
{
2011-12-14 00:23:17 +03:00
double effectiveness = std : : min ( ( getMarketEfficiency ( ) + 1.0 ) / 20.0 , 0.5 ) ;
2010-05-18 10:01:54 +03:00
2011-12-14 00:23:17 +03:00
double r = VLC - > objh - > resVals [ id1 ] , //value of given resource
2010-05-18 10:01:54 +03:00
g = VLC - > objh - > resVals [ id2 ] / effectiveness ; //value of wanted resource
if ( r > g ) //if given resource is more expensive than wanted
{
val2 = ceil ( r / g ) ;
val1 = 1 ;
}
else //if wanted resource is more expensive
{
2011-12-14 00:23:17 +03:00
val1 = ( g / r ) + 0.5 ;
2010-05-18 10:01:54 +03:00
val2 = 1 ;
}
}
2010-05-26 12:47:53 +03:00
break ;
2011-12-14 00:23:17 +03:00
case EMarketMode : : CREATURE_RESOURCE :
2010-05-26 12:47:53 +03:00
{
2011-12-14 00:23:17 +03:00
const double effectivenessArray [ ] = { 0.0 , 0.3 , 0.45 , 0.50 , 0.65 , 0.7 , 0.85 , 0.9 , 1.0 } ;
double effectiveness = effectivenessArray [ std : : min ( getMarketEfficiency ( ) , 8 ) ] ;
2010-05-26 12:47:53 +03:00
2011-12-14 00:23:17 +03:00
double r = VLC - > creh - > creatures [ id1 ] - > cost [ 6 ] , //value of given creature in gold
2010-05-26 12:47:53 +03:00
g = VLC - > objh - > resVals [ id2 ] / effectiveness ; //value of wanted resource
2010-05-18 10:01:54 +03:00
2010-05-26 12:47:53 +03:00
if ( r > g ) //if given resource is more expensive than wanted
{
val2 = ceil ( r / g ) ;
val1 = 1 ;
}
else //if wanted resource is more expensive
{
2011-12-14 00:23:17 +03:00
val1 = ( g / r ) + 0.5 ;
2010-05-26 12:47:53 +03:00
val2 = 1 ;
}
}
break ;
2011-12-14 00:23:17 +03:00
case EMarketMode : : RESOURCE_PLAYER :
2010-05-26 12:47:53 +03:00
val1 = 1 ;
val2 = 1 ;
2010-05-18 10:01:54 +03:00
break ;
2011-12-14 00:23:17 +03:00
case EMarketMode : : RESOURCE_ARTIFACT :
2010-06-27 19:03:01 +03:00
{
2011-12-14 00:23:17 +03:00
double effectiveness = std : : min ( ( getMarketEfficiency ( ) + 3.0 ) / 20.0 , 0.6 ) ;
double r = VLC - > objh - > resVals [ id1 ] , //value of offered resource
2010-06-27 19:03:01 +03:00
g = VLC - > arth - > artifacts [ id2 ] - > price / effectiveness ; //value of bought artifact in gold
2012-07-19 21:52:44 +03:00
2010-06-27 19:03:01 +03:00
if ( id1 ! = 6 ) //non-gold prices are doubled
2012-07-19 21:52:44 +03:00
r / = 2 ;
2010-06-27 19:03:01 +03:00
2012-01-28 14:42:15 +03:00
val1 = std : : max ( 1 , ( int ) ( ( g / r ) + 0.5 ) ) ; //don't sell arts for less than 1 resource
2010-06-27 19:03:01 +03:00
val2 = 1 ;
}
break ;
2011-12-14 00:23:17 +03:00
case EMarketMode : : ARTIFACT_RESOURCE :
2011-04-23 00:51:10 +03:00
{
2011-12-14 00:23:17 +03:00
double effectiveness = std : : min ( ( getMarketEfficiency ( ) + 3.0 ) / 20.0 , 0.6 ) ;
double r = VLC - > arth - > artifacts [ id1 ] - > price * effectiveness ,
2012-07-19 21:52:44 +03:00
g = VLC - > objh - > resVals [ id2 ] ;
2011-04-23 00:51:10 +03:00
// if(id2 != 6) //non-gold prices are doubled
2012-07-19 21:52:44 +03:00
// r /= 2;
2010-06-27 19:03:01 +03:00
2011-04-23 00:51:10 +03:00
val1 = 1 ;
2012-01-28 14:42:15 +03:00
val2 = std : : max ( 1 , ( int ) ( ( r / g ) + 0.5 ) ) ; //at least one resource is given in return
2011-04-23 00:51:10 +03:00
}
break ;
2011-12-14 00:23:17 +03:00
case EMarketMode : : CREATURE_EXP :
2010-07-20 09:05:45 +03:00
{
val1 = 1 ;
val2 = ( VLC - > creh - > creatures [ id1 ] - > AIValue / 40 ) * 5 ;
}
break ;
2011-12-14 00:23:17 +03:00
case EMarketMode : : ARTIFACT_EXP :
2010-07-23 15:02:15 +03:00
{
val1 = 1 ;
2010-07-20 09:05:45 +03:00
2010-07-23 15:02:15 +03:00
int givenClass = VLC - > arth - > artifacts [ id1 ] - > getArtClassSerial ( ) ;
if ( givenClass < 0 | | givenClass > 3 )
{
val2 = 0 ;
return false ;
}
static const int expPerClass [ ] = { 1000 , 1500 , 3000 , 6000 } ;
val2 = expPerClass [ givenClass ] ;
}
break ;
2010-05-18 10:01:54 +03:00
default :
assert ( 0 ) ;
return false ;
}
return true ;
}
2011-12-14 00:23:17 +03:00
bool IMarket : : allowsTrade ( EMarketMode : : EMarketMode mode ) const
2010-05-18 10:01:54 +03:00
{
return false ;
}
2011-12-14 00:23:17 +03:00
int IMarket : : availableUnits ( EMarketMode : : EMarketMode mode , int marketItemSerial ) const
2010-05-18 10:01:54 +03:00
{
2010-05-26 12:47:53 +03:00
switch ( mode )
{
2011-12-14 00:23:17 +03:00
case EMarketMode : : RESOURCE_RESOURCE :
case EMarketMode : : ARTIFACT_RESOURCE :
case EMarketMode : : CREATURE_RESOURCE :
2010-05-26 12:47:53 +03:00
return - 1 ;
default :
return 1 ;
}
2010-05-18 10:01:54 +03:00
}
2011-12-14 00:23:17 +03:00
std : : vector < int > IMarket : : availableItemsIds ( EMarketMode : : EMarketMode mode ) const
2010-05-18 10:01:54 +03:00
{
std : : vector < int > ret ;
2010-05-26 12:47:53 +03:00
switch ( mode )
{
2011-12-14 00:23:17 +03:00
case EMarketMode : : RESOURCE_RESOURCE :
case EMarketMode : : ARTIFACT_RESOURCE :
case EMarketMode : : CREATURE_RESOURCE :
2010-05-18 10:01:54 +03:00
for ( int i = 0 ; i < 7 ; i + + )
ret . push_back ( i ) ;
2010-08-30 02:12:34 +03:00
default :
2010-05-26 12:47:53 +03:00
break ;
}
2010-05-18 10:01:54 +03:00
return ret ;
}
2012-02-14 21:04:45 +03:00
const IMarket * IMarket : : castFrom ( const CGObjectInstance * obj , bool verbose /*= true*/ )
2010-05-18 10:01:54 +03:00
{
switch ( obj - > ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : TOWN :
2010-05-18 10:01:54 +03:00
return static_cast < const CGTownInstance * > ( obj ) ;
2012-09-23 21:01:04 +03:00
case Obj : : ALTAR_OF_SACRIFICE :
case Obj : : BLACK_MARKET :
case Obj : : TRADING_POST :
case Obj : : TRADING_POST_SNOW :
case Obj : : FREELANCERS_GUILD :
2010-05-18 10:01:54 +03:00
return static_cast < const CGMarket * > ( obj ) ;
2012-09-23 21:01:04 +03:00
case Obj : : UNIVERSITY :
2010-07-20 17:08:13 +03:00
return static_cast < const CGUniversity * > ( obj ) ;
2010-05-18 10:01:54 +03:00
default :
2012-02-14 21:04:45 +03:00
if ( verbose )
tlog1 < < " Cannot cast to IMarket object with ID " < < obj - > ID < < std : : endl ;
2010-05-18 10:01:54 +03:00
return NULL ;
}
}
IMarket : : IMarket ( const CGObjectInstance * O )
: o ( O )
{
}
2011-12-14 00:23:17 +03:00
std : : vector < EMarketMode : : EMarketMode > IMarket : : availableModes ( ) const
2010-05-18 10:01:54 +03:00
{
2011-12-14 00:23:17 +03:00
std : : vector < EMarketMode : : EMarketMode > ret ;
for ( int i = 0 ; i < EMarketMode : : MARTKET_AFTER_LAST_PLACEHOLDER ; i + + )
if ( allowsTrade ( ( EMarketMode : : EMarketMode ) i ) )
ret . push_back ( ( EMarketMode : : EMarketMode ) i ) ;
2010-05-18 10:01:54 +03:00
return ret ;
}
void CGMarket : : onHeroVisit ( const CGHeroInstance * h ) const
{
OpenWindow ow ;
ow . id1 = id ;
ow . id2 = h - > id ;
ow . window = OpenWindow : : MARKET_WINDOW ;
cb - > sendAndApply ( & ow ) ;
}
int CGMarket : : getMarketEfficiency ( ) const
{
return 5 ;
}
2011-12-14 00:23:17 +03:00
bool CGMarket : : allowsTrade ( EMarketMode : : EMarketMode mode ) const
2010-05-18 10:01:54 +03:00
{
switch ( mode )
{
2011-12-14 00:23:17 +03:00
case EMarketMode : : RESOURCE_RESOURCE :
case EMarketMode : : RESOURCE_PLAYER :
2010-05-18 10:01:54 +03:00
switch ( ID )
{
2012-09-23 21:01:04 +03:00
case Obj : : TRADING_POST :
case Obj : : TRADING_POST_SNOW :
2010-05-18 10:01:54 +03:00
return true ;
default :
return false ;
}
2011-12-14 00:23:17 +03:00
case EMarketMode : : CREATURE_RESOURCE :
2012-09-23 21:01:04 +03:00
return ID = = Obj : : FREELANCERS_GUILD ;
2010-07-26 18:37:58 +03:00
//case ARTIFACT_RESOURCE:
2011-12-14 00:23:17 +03:00
case EMarketMode : : RESOURCE_ARTIFACT :
2012-09-23 21:01:04 +03:00
return ID = = Obj : : BLACK_MARKET ;
2011-12-14 00:23:17 +03:00
case EMarketMode : : ARTIFACT_EXP :
case EMarketMode : : CREATURE_EXP :
2012-09-23 21:01:04 +03:00
return ID = = Obj : : ALTAR_OF_SACRIFICE ; //TODO? check here for alignment of visiting hero? - would not be coherent with other checks here
2011-12-14 00:23:17 +03:00
case EMarketMode : : RESOURCE_SKILL :
2012-09-23 21:01:04 +03:00
return ID = = Obj : : UNIVERSITY ;
2010-08-30 02:12:34 +03:00
default :
return false ;
2010-05-18 10:01:54 +03:00
}
}
2011-12-14 00:23:17 +03:00
int CGMarket : : availableUnits ( EMarketMode : : EMarketMode mode , int marketItemSerial ) const
2010-05-18 10:01:54 +03:00
{
return - 1 ;
}
2011-12-14 00:23:17 +03:00
std : : vector < int > CGMarket : : availableItemsIds ( EMarketMode : : EMarketMode mode ) const
2010-05-18 10:01:54 +03:00
{
switch ( mode )
{
2011-12-14 00:23:17 +03:00
case EMarketMode : : RESOURCE_RESOURCE :
case EMarketMode : : RESOURCE_PLAYER :
2010-05-18 10:01:54 +03:00
return IMarket : : availableItemsIds ( mode ) ;
default :
return std : : vector < int > ( ) ;
}
}
CGMarket : : CGMarket ( )
: IMarket ( this )
{
2010-06-26 19:02:10 +03:00
}
2011-12-14 00:23:17 +03:00
std : : vector < int > CGBlackMarket : : availableItemsIds ( EMarketMode : : EMarketMode mode ) const
2010-06-26 19:02:10 +03:00
{
switch ( mode )
{
2011-12-14 00:23:17 +03:00
case EMarketMode : : ARTIFACT_RESOURCE :
2010-06-26 19:02:10 +03:00
return IMarket : : availableItemsIds ( mode ) ;
2011-12-14 00:23:17 +03:00
case EMarketMode : : RESOURCE_ARTIFACT :
2010-06-26 19:02:10 +03:00
{
std : : vector < int > ret ;
BOOST_FOREACH ( const CArtifact * a , artifacts )
if ( a )
ret . push_back ( a - > id ) ;
else
ret . push_back ( - 1 ) ;
return ret ;
}
default :
return std : : vector < int > ( ) ;
}
}
void CGBlackMarket : : newTurn ( ) const
{
2010-06-27 19:03:01 +03:00
if ( cb - > getDate ( 2 ) ! = 1 )
return ;
2010-06-26 19:02:10 +03:00
2010-06-27 19:03:01 +03:00
SetAvailableArtifacts saa ;
saa . id = id ;
cb - > pickAllowedArtsSet ( saa . arts ) ;
cb - > sendAndApply ( & saa ) ;
2010-07-20 17:08:13 +03:00
}
void CGUniversity : : initObj ( )
{
std : : vector < int > toChoose ;
2011-12-14 00:23:17 +03:00
for ( int i = 0 ; i < GameConstants : : SKILL_QUANTITY ; i + + )
2010-07-20 17:08:13 +03:00
if ( cb - > isAllowed ( 2 , i ) )
toChoose . push_back ( i ) ;
if ( toChoose . size ( ) < 4 )
{
tlog0 < < " Warning: less then 4 available skills was found by University initializer! \n " ;
return ;
}
2012-07-19 21:52:44 +03:00
2010-07-20 17:08:13 +03:00
for ( int i = 0 ; i < 4 ; i + + ) //get 4 skills
{
int skillPos = ran ( ) % toChoose . size ( ) ;
skills . push_back ( toChoose [ skillPos ] ) ; //move it to selected
toChoose . erase ( toChoose . begin ( ) + skillPos ) ; //remove from list
}
}
2011-12-14 00:23:17 +03:00
std : : vector < int > CGUniversity : : availableItemsIds ( EMarketMode : : EMarketMode mode ) const
2010-07-20 17:08:13 +03:00
{
switch ( mode )
{
2011-12-14 00:23:17 +03:00
case EMarketMode : : RESOURCE_SKILL :
2010-07-20 17:08:13 +03:00
return skills ;
2012-07-19 21:52:44 +03:00
2010-07-20 17:08:13 +03:00
default :
return std : : vector < int > ( ) ;
}
}
void CGUniversity : : onHeroVisit ( const CGHeroInstance * h ) const
{
OpenWindow ow ;
ow . id1 = id ;
ow . id2 = h - > id ;
ow . window = OpenWindow : : UNIVERSITY_WINDOW ;
cb - > sendAndApply ( & ow ) ;
}
2011-08-26 23:32:05 +03:00
GrowthInfo : : Entry : : Entry ( const std : : string & format , int _count )
: count ( _count )
{
description = boost : : str ( boost : : format ( format ) % count ) ;
}
2011-12-14 00:23:17 +03:00
GrowthInfo : : Entry : : Entry ( int subID , EBuilding : : EBuilding building , int _count )
2011-08-26 23:32:05 +03:00
: count ( _count )
{
2012-09-02 13:33:41 +03:00
description = boost : : str ( boost : : format ( " %s %+d " ) % VLC - > townh - > towns [ subID ] . buildings [ building ] - > Name ( ) % count ) ;
2011-08-26 23:32:05 +03:00
}
CTownAndVisitingHero : : CTownAndVisitingHero ( )
{
setNodeType ( TOWN_AND_VISITOR ) ;
}
int GrowthInfo : : totalGrowth ( ) const
{
int ret = 0 ;
BOOST_FOREACH ( const Entry & entry , entries )
ret + = entry . count ;
return ret ;
2011-09-03 17:14:43 +03:00
}