2017-07-13 10:26:03 +02:00
/*
* CArtifactHolder . 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
*
*/
2014-07-05 12:58:42 +03:00
# include "StdInc.h"
# include "CArtifactHolder.h"
2014-07-13 20:53:37 +03:00
# include "../gui/CGuiHandler.h"
# include "../gui/CCursorHandler.h"
2014-07-05 12:58:42 +03:00
2014-07-15 10:14:49 +03:00
# include "Buttons.h"
# include "CComponent.h"
2014-07-13 20:53:37 +03:00
# include "../windows/CHeroWindow.h"
# include "../windows/CSpellWindow.h"
# include "../windows/GUIClasses.h"
2014-07-05 12:58:42 +03:00
# include "../CPlayerInterface.h"
# include "../CGameInfo.h"
# include "../../CCallback.h"
# include "../../lib/CArtHandler.h"
2015-02-02 10:25:26 +02:00
# include "../../lib/spells/CSpellHandler.h"
2014-07-05 12:58:42 +03:00
# include "../../lib/CGeneralTextHandler.h"
# include "../../lib/mapObjects/CGHeroInstance.h"
2018-04-07 13:34:11 +02:00
CHeroArtPlace : : CHeroArtPlace ( Point position , const CArtifactInstance * Art )
: CArtPlace ( position , Art ) ,
locked ( false ) ,
picked ( false ) ,
marked ( false ) ,
ourOwner ( nullptr )
2014-07-05 12:58:42 +03:00
{
createImage ( ) ;
}
2016-11-04 18:54:09 +02:00
void CHeroArtPlace : : createImage ( )
2014-07-05 12:58:42 +03:00
{
2018-04-07 13:34:11 +02:00
OBJECT_CONSTRUCTION_CAPTURING ( 255 - DISPOSE ) ;
2014-07-05 12:58:42 +03:00
2018-04-07 13:34:11 +02:00
si32 imageIndex = 0 ;
if ( ourArt )
2016-11-08 18:26:24 +02:00
imageIndex = ourArt - > artType - > iconIndex ;
2018-04-07 13:34:11 +02:00
if ( locked )
2016-11-08 18:26:24 +02:00
imageIndex = ArtifactID : : ART_LOCK ;
2014-07-05 12:58:42 +03:00
2018-04-07 13:34:11 +02:00
image = std : : make_shared < CAnimImage > ( " artifact " , imageIndex ) ;
if ( ! ourArt )
2014-07-05 12:58:42 +03:00
image - > disable ( ) ;
2018-04-07 13:34:11 +02:00
selection = std : : make_shared < CAnimImage > ( " artifact " , ArtifactID : : ART_SELECTION ) ;
2014-07-05 12:58:42 +03:00
selection - > disable ( ) ;
}
2016-11-04 18:54:09 +02:00
void CHeroArtPlace : : lockSlot ( bool on )
2014-07-05 12:58:42 +03:00
{
if ( locked = = on )
return ;
locked = on ;
if ( on )
image - > setFrame ( ArtifactID : : ART_LOCK ) ;
2017-05-27 16:06:36 +02:00
else if ( ourArt )
2014-07-05 12:58:42 +03:00
image - > setFrame ( ourArt - > artType - > iconIndex ) ;
2017-05-27 16:06:36 +02:00
else
image - > setFrame ( 0 ) ;
2014-07-05 12:58:42 +03:00
}
2016-11-04 18:54:09 +02:00
void CHeroArtPlace : : pickSlot ( bool on )
2014-07-05 12:58:42 +03:00
{
if ( picked = = on )
return ;
picked = on ;
if ( on )
image - > disable ( ) ;
else
image - > enable ( ) ;
}
2016-11-04 18:54:09 +02:00
void CHeroArtPlace : : selectSlot ( bool on )
2014-07-05 12:58:42 +03:00
{
if ( marked = = on )
return ;
marked = on ;
if ( on )
selection - > enable ( ) ;
else
selection - > disable ( ) ;
}
2016-11-04 18:54:09 +02:00
void CHeroArtPlace : : clickLeft ( tribool down , bool previousState )
2014-07-05 12:58:42 +03:00
{
//LRClickableAreaWTextComp::clickLeft(down);
bool inBackpack = slotID > = GameConstants : : BACKPACK_START ,
srcInBackpack = ourOwner - > commonInfo - > src . slotID > = GameConstants : : BACKPACK_START ,
srcInSameHero = ourOwner - > commonInfo - > src . AOH = = ourOwner ;
if ( ourOwner - > highlightModeCallback & & ourArt )
{
if ( down )
{
2014-12-24 19:48:37 +02:00
if ( ! ourArt - > artType - > isTradable ( ) ) //War Machine or Spellbook
2014-07-05 12:58:42 +03:00
{
LOCPLINT - > showInfoDialog ( CGI - > generaltexth - > allTexts [ 21 ] ) ; //This item can't be traded.
}
else
{
ourOwner - > unmarkSlots ( false ) ;
selectSlot ( true ) ;
ourOwner - > highlightModeCallback ( this ) ;
}
}
return ;
}
// If clicked on spellbook, open it only if no artifact is held at the moment.
if ( ourArt & & ! down & & previousState & & ! ourOwner - > commonInfo - > src . AOH )
{
2014-12-24 17:49:12 +02:00
if ( ourArt - > artType - > id = = ArtifactID : : SPELLBOOK )
2018-07-25 00:36:48 +02:00
GH . pushIntT < CSpellWindow > ( ourOwner - > curHero , LOCPLINT , LOCPLINT - > battleInt ) ;
2014-07-05 12:58:42 +03:00
}
if ( ! down & & previousState )
{
2014-12-24 17:49:12 +02:00
if ( ourArt & & ourArt - > artType - > id = = ArtifactID : : SPELLBOOK )
2014-07-05 12:58:42 +03:00
return ; //this is handled separately
if ( ! ourOwner - > commonInfo - > src . AOH ) //nothing has been clicked
{
if ( ourArt //to prevent selecting empty slots (bugfix to what GrayFace reported)
& & ourOwner - > curHero - > tempOwner = = LOCPLINT - > playerID ) //can't take art from another player
{
2014-12-24 17:49:12 +02:00
if ( ourArt - > artType - > id = = ArtifactID : : CATAPULT ) //catapult cannot be highlighted
2014-07-05 12:58:42 +03:00
{
2018-04-07 13:34:11 +02:00
std : : vector < std : : shared_ptr < CComponent > > catapult ( 1 , std : : make_shared < CComponent > ( CComponent : : artifact , 3 , 0 ) ) ;
2014-07-05 12:58:42 +03:00
LOCPLINT - > showInfoDialog ( CGI - > generaltexth - > allTexts [ 312 ] , catapult ) ; //The Catapult must be equipped.
return ;
}
select ( ) ;
}
}
else if ( ourArt = = ourOwner - > commonInfo - > src . art ) //restore previously picked artifact
{
deselect ( ) ;
}
else //perform artifact transition
{
if ( inBackpack ) // Backpack destination.
{
if ( srcInBackpack & & slotID = = ourOwner - > commonInfo - > src . slotID + 1 ) //next slot (our is not visible, so visually same as "old" place) to the art -> make nothing, return artifact to slot
{
deselect ( ) ;
}
else
{
const CArtifact * const cur = ourOwner - > commonInfo - > src . art - > artType ;
2017-05-26 18:51:45 +02:00
if ( cur - > id = = ArtifactID : : CATAPULT )
2014-07-05 12:58:42 +03:00
{
//should not happen, catapult cannot be selected
2017-05-26 18:51:45 +02:00
logGlobal - > error ( " Attempt to move Catapult " ) ;
}
2017-06-06 18:45:34 +02:00
else if ( cur - > isBig ( ) )
2017-05-26 18:51:45 +02:00
{
//war machines cannot go to backpack
2014-07-05 12:58:42 +03:00
LOCPLINT - > showInfoDialog ( boost : : str ( boost : : format ( CGI - > generaltexth - > allTexts [ 153 ] ) % cur - > Name ( ) ) ) ;
2017-05-26 18:51:45 +02:00
}
else
{
2014-07-05 12:58:42 +03:00
setMeAsDest ( ) ;
vstd : : amin ( ourOwner - > commonInfo - > dst . slotID , ArtifactPosition (
2020-10-01 10:38:06 +02:00
( si32 ) ourOwner - > curHero - > artifactsInBackpack . size ( ) + GameConstants : : BACKPACK_START ) ) ;
2014-07-05 12:58:42 +03:00
if ( srcInBackpack & & srcInSameHero )
{
if ( ! ourArt //cannot move from backpack to AFTER backpack -> combined with vstd::amin above it will guarantee that dest is at most the last artifact
2016-01-21 19:23:45 +02:00
| | ourOwner - > commonInfo - > src . slotID < ourOwner - > commonInfo - > dst . slotID ) //rearranging arts in backpack after taking src artifact, the dest id will be shifted
2014-07-05 12:58:42 +03:00
vstd : : advance ( ourOwner - > commonInfo - > dst . slotID , - 1 ) ;
}
if ( srcInSameHero & & ourOwner - > commonInfo - > dst . slotID = = ourOwner - > commonInfo - > src . slotID ) //we came to src == dst
deselect ( ) ;
else
ourOwner - > realizeCurrentTransaction ( ) ;
}
}
}
//check if swap is possible
else if ( fitsHere ( ourOwner - > commonInfo - > src . art ) & &
( ! ourArt | | ourOwner - > curHero - > tempOwner = = LOCPLINT - > playerID ) )
{
setMeAsDest ( ) ;
//
// // Special case when the dest artifact can't be fit into the src slot.
// //CGI->arth->unequipArtifact(ourOwner->curHero->artifWorn, slotID);
// const CArtifactsOfHero* srcAOH = ourOwner->commonInfo->src.AOH;
// ui16 srcSlotID = ourOwner->commonInfo->src.slotID;
// if (ourArt && srcSlotID < 19 && !ourArt->canBePutAt(ArtifactLocation(srcAOH->curHero, srcSlotID)))
// {
// // Put dest artifact into owner's backpack.
// ourOwner->commonInfo->src.AOH = ourOwner;
// ourOwner->commonInfo->src.slotID = ourOwner->curHero->artifacts.size() + 19;
// }
ourOwner - > realizeCurrentTransaction ( ) ;
}
}
}
}
2016-11-04 18:54:09 +02:00
bool CHeroArtPlace : : askToAssemble ( const CArtifactInstance * art , ArtifactPosition slot ,
2016-01-23 14:20:51 +02:00
const CGHeroInstance * hero )
{
2016-10-30 11:00:25 +02:00
assert ( art ! = nullptr ) ;
assert ( hero ! = nullptr ) ;
2016-01-23 14:20:51 +02:00
std : : vector < const CArtifact * > assemblyPossibilities = art - > assemblyPossibilities ( hero ) ;
// If the artifact can be assembled, display dialog.
for ( const CArtifact * combination : assemblyPossibilities )
{
LOCPLINT - > showArtifactAssemblyDialog (
art - > artType - > id ,
combination - > id ,
true ,
std : : bind ( & CCallback : : assembleArtifacts , LOCPLINT - > cb . get ( ) , hero , slot , true , combination - > id ) ,
0 ) ;
if ( assemblyPossibilities . size ( ) > 2 )
2017-08-11 13:38:10 +02:00
logGlobal - > warn ( " More than one possibility of assembling on %s... taking only first " , art - > artType - > Name ( ) ) ;
2016-01-23 14:20:51 +02:00
return true ;
}
return false ;
}
2016-11-04 18:54:09 +02:00
void CHeroArtPlace : : clickRight ( tribool down , bool previousState )
2014-07-05 12:58:42 +03:00
{
2017-10-29 17:23:30 +02:00
if ( ourArt & & down & & ! locked & & text . size ( ) & & ! picked ) //if there is no description or it's a lock, do nothing ;]
2014-07-05 12:58:42 +03:00
{
if ( slotID < GameConstants : : BACKPACK_START )
{
if ( ourOwner - > allowedAssembling )
{
std : : vector < const CArtifact * > assemblyPossibilities = ourArt - > assemblyPossibilities ( ourOwner - > curHero ) ;
// If the artifact can be assembled, display dialog.
2016-01-23 14:20:51 +02:00
if ( askToAssemble ( ourArt , slotID , ourOwner - > curHero ) )
2014-07-05 12:58:42 +03:00
{
return ;
}
// Otherwise if the artifact can be diasassembled, display dialog.
if ( ourArt - > canBeDisassembled ( ) )
{
LOCPLINT - > showArtifactAssemblyDialog (
ourArt - > artType - > id ,
0 ,
false ,
2014-08-09 15:14:31 +03:00
std : : bind ( & CCallback : : assembleArtifacts , LOCPLINT - > cb . get ( ) , ourOwner - > curHero , slotID , false , ArtifactID ( ) ) ,
2014-07-05 12:58:42 +03:00
0 ) ;
return ;
}
}
}
// Lastly just show the artifact description.
LRClickableAreaWTextComp : : clickRight ( down , previousState ) ;
}
}
/**
* Selects artifact slot so that the containing artifact looks like it ' s picked up .
*/
2016-11-04 18:54:09 +02:00
void CHeroArtPlace : : select ( )
2014-07-05 12:58:42 +03:00
{
if ( locked )
return ;
selectSlot ( true ) ;
pickSlot ( true ) ;
if ( ourArt - > canBeDisassembled ( ) & & slotID < GameConstants : : BACKPACK_START ) //worn combined artifact -> locks have to disappear
{
for ( int i = 0 ; i < GameConstants : : BACKPACK_START ; i + + )
{
2018-04-07 13:34:11 +02:00
auto ap = ourOwner - > getArtPlace ( i ) ;
if ( ap ) //getArtPlace may return null
2015-02-14 14:19:50 +02:00
ap - > pickSlot ( ourArt - > isPart ( ap - > ourArt ) ) ;
2014-07-05 12:58:42 +03:00
}
}
2016-10-28 12:39:16 +02:00
CCS - > curh - > dragAndDropCursor ( make_unique < CAnimImage > ( " artifact " , ourArt - > artType - > iconIndex ) ) ;
2014-07-05 12:58:42 +03:00
ourOwner - > commonInfo - > src . setTo ( this , false ) ;
ourOwner - > markPossibleSlots ( ourArt ) ;
if ( slotID > = GameConstants : : BACKPACK_START )
ourOwner - > scrollBackpack ( 0 ) ; //will update slots
ourOwner - > updateParentWindow ( ) ;
ourOwner - > safeRedraw ( ) ;
}
/**
2015-02-14 14:43:03 +02:00
* Deselects the artifact slot .
2014-07-05 12:58:42 +03:00
*/
2016-11-04 18:54:09 +02:00
void CHeroArtPlace : : deselect ( )
2014-07-05 12:58:42 +03:00
{
pickSlot ( false ) ;
if ( ourArt & & ourArt - > canBeDisassembled ( ) ) //combined art returned to its slot -> restore locks
{
for ( int i = 0 ; i < GameConstants : : BACKPACK_START ; i + + )
2015-02-14 14:43:03 +02:00
{
auto place = ourOwner - > getArtPlace ( i ) ;
2016-01-21 09:49:46 +02:00
2015-02-14 14:58:33 +02:00
if ( nullptr ! = place ) //getArtPlace may return null
2015-02-14 14:43:03 +02:00
place - > pickSlot ( false ) ;
2016-01-21 09:49:46 +02:00
}
2014-07-05 12:58:42 +03:00
}
CCS - > curh - > dragAndDropCursor ( nullptr ) ;
ourOwner - > unmarkSlots ( ) ;
ourOwner - > commonInfo - > src . clear ( ) ;
if ( slotID > = GameConstants : : BACKPACK_START )
ourOwner - > scrollBackpack ( 0 ) ; //will update slots
ourOwner - > updateParentWindow ( ) ;
ourOwner - > safeRedraw ( ) ;
}
2016-11-04 18:54:09 +02:00
void CHeroArtPlace : : showAll ( SDL_Surface * to )
2014-07-05 12:58:42 +03:00
{
if ( ourArt & & ! picked & & ourArt = = ourOwner - > curHero - > getArt ( slotID , false ) ) //last condition is needed for disassembling -> artifact may be gone, but we don't know yet TODO: real, nice solution
{
CIntObject : : showAll ( to ) ;
}
if ( marked & & active )
{
// Draw vertical bars.
for ( int i = 0 ; i < pos . h ; + + i )
{
CSDL_Ext : : SDL_PutPixelWithoutRefresh ( to , pos . x , pos . y + i , 240 , 220 , 120 ) ;
CSDL_Ext : : SDL_PutPixelWithoutRefresh ( to , pos . x + pos . w - 1 , pos . y + i , 240 , 220 , 120 ) ;
}
// Draw horizontal bars.
for ( int i = 0 ; i < pos . w ; + + i )
{
CSDL_Ext : : SDL_PutPixelWithoutRefresh ( to , pos . x + i , pos . y , 240 , 220 , 120 ) ;
CSDL_Ext : : SDL_PutPixelWithoutRefresh ( to , pos . x + i , pos . y + pos . h - 1 , 240 , 220 , 120 ) ;
}
}
}
2016-11-04 18:54:09 +02:00
bool CHeroArtPlace : : fitsHere ( const CArtifactInstance * art ) const
2014-07-05 12:58:42 +03:00
{
// You can place 'no artifact' anywhere.
if ( ! art )
return true ;
2015-02-14 17:34:27 +02:00
// Anything but War Machines can be placed in backpack.
2014-07-05 12:58:42 +03:00
if ( slotID > = GameConstants : : BACKPACK_START )
2017-05-26 18:51:45 +02:00
return ! art - > artType - > isBig ( ) ;
2014-07-05 12:58:42 +03:00
return art - > canBePutAt ( ArtifactLocation ( ourOwner - > curHero , slotID ) , true ) ;
}
2017-07-15 13:08:20 +02:00
void CHeroArtPlace : : setMeAsDest ( bool backpackAsVoid )
2014-07-05 12:58:42 +03:00
{
ourOwner - > commonInfo - > dst . setTo ( this , backpackAsVoid ) ;
}
2016-11-04 18:54:09 +02:00
void CHeroArtPlace : : setArtifact ( const CArtifactInstance * art )
2014-07-05 12:58:42 +03:00
{
baseType = - 1 ; //by default we don't store any component
ourArt = art ;
if ( ! art )
{
image - > disable ( ) ;
text = std : : string ( ) ;
hoverText = CGI - > generaltexth - > allTexts [ 507 ] ;
2016-01-21 19:23:45 +02:00
return ;
2014-07-05 12:58:42 +03:00
}
2016-01-21 19:23:45 +02:00
image - > enable ( ) ;
image - > setFrame ( locked ? ArtifactID : : ART_LOCK : art - > artType - > iconIndex ) ;
2014-07-05 12:58:42 +03:00
2016-01-21 19:23:45 +02:00
text = art - > getEffectiveDescription ( ourOwner - > curHero ) ;
2014-07-05 12:58:42 +03:00
2016-01-21 19:23:45 +02:00
if ( art - > artType - > id = = ArtifactID : : SPELL_SCROLL )
{
int spellID = art - > getGivenSpellID ( ) ;
if ( spellID > = 0 )
2014-07-05 12:58:42 +03:00
{
2016-01-21 19:23:45 +02:00
//add spell component info (used to provide a pic in r-click popup)
baseType = CComponent : : spell ;
type = spellID ;
2014-07-05 12:58:42 +03:00
bonusValue = 0 ;
}
}
2016-01-21 19:23:45 +02:00
else
{
baseType = CComponent : : artifact ;
type = art - > artType - > id ;
bonusValue = 0 ;
}
if ( locked ) // Locks should appear as empty.
hoverText = CGI - > generaltexth - > allTexts [ 507 ] ;
else
hoverText = boost : : str ( boost : : format ( CGI - > generaltexth - > heroscrn [ 1 ] ) % ourArt - > artType - > Name ( ) ) ;
2014-07-05 12:58:42 +03:00
}
void CArtifactsOfHero : : SCommonPart : : reset ( )
{
src . clear ( ) ;
dst . clear ( ) ;
CCS - > curh - > dragAndDropCursor ( nullptr ) ;
}
void CArtifactsOfHero : : setHero ( const CGHeroInstance * hero )
{
curHero = hero ;
if ( curHero - > artifactsInBackpack . size ( ) > 0 )
backpackPos % = curHero - > artifactsInBackpack . size ( ) ;
else
backpackPos = 0 ;
// Fill the slots for worn artifacts and backpack.
2016-01-21 09:49:46 +02:00
2015-02-14 17:34:27 +02:00
for ( auto p : artWorn )
{
setSlotData ( p . second , p . first ) ;
}
2016-01-21 09:49:46 +02:00
2014-07-05 12:58:42 +03:00
scrollBackpack ( 0 ) ;
}
void CArtifactsOfHero : : dispose ( )
{
CCS - > curh - > dragAndDropCursor ( nullptr ) ;
}
void CArtifactsOfHero : : scrollBackpack ( int dir )
{
2020-10-01 10:38:06 +02:00
int artsInBackpack = static_cast < int > ( curHero - > artifactsInBackpack . size ( ) ) ;
2014-07-05 12:58:42 +03:00
backpackPos + = dir ;
if ( backpackPos < 0 ) // No guarantee of modulus behavior with negative operands -> we keep it positive
backpackPos + = artsInBackpack ;
if ( artsInBackpack )
backpackPos % = artsInBackpack ;
std : : multiset < const CArtifactInstance * > toOmit = artifactsOnAltar ;
if ( commonInfo - > src . art ) //if we picked an art from backapck, its slot has to be omitted
toOmit . insert ( commonInfo - > src . art ) ;
int omitedSoFar = 0 ;
//set new data
size_t s = 0 ;
for ( ; s < artsInBackpack ; + + s )
{
if ( s < artsInBackpack )
{
auto slotID = ArtifactPosition ( GameConstants : : BACKPACK_START + ( s + backpackPos ) % artsInBackpack ) ;
const CArtifactInstance * art = curHero - > getArt ( slotID ) ;
assert ( art ) ;
if ( ! vstd : : contains ( toOmit , art ) )
{
if ( s - omitedSoFar < backpack . size ( ) )
setSlotData ( backpack [ s - omitedSoFar ] , slotID ) ;
}
else
{
toOmit - = art ;
omitedSoFar + + ;
continue ;
}
}
}
for ( ; s - omitedSoFar < backpack . size ( ) ; s + + )
2020-10-01 10:38:06 +02:00
eraseSlotData ( backpack [ s - omitedSoFar ] , ArtifactPosition ( GameConstants : : BACKPACK_START + ( si32 ) s ) ) ;
2014-07-05 12:58:42 +03:00
//in artifact merchant selling artifacts we may have highlight on one of backpack artifacts -> market needs update, cause artifact under highlight changed
if ( highlightModeCallback )
{
for ( auto & elem : backpack )
{
if ( elem - > marked )
{
2018-04-07 13:34:11 +02:00
highlightModeCallback ( elem . get ( ) ) ;
2014-07-05 12:58:42 +03:00
break ;
}
}
}
//blocking scrolling if there is not enough artifacts to scroll
bool scrollingPossible = artsInBackpack - omitedSoFar > backpack . size ( ) ;
leftArtRoll - > block ( ! scrollingPossible ) ;
rightArtRoll - > block ( ! scrollingPossible ) ;
safeRedraw ( ) ;
}
/**
* Marks possible slots where a given artifact can be placed , except backpack .
*
* @ param art Artifact checked against .
*/
void CArtifactsOfHero : : markPossibleSlots ( const CArtifactInstance * art )
{
for ( CArtifactsOfHero * aoh : commonInfo - > participants )
2015-02-14 17:34:27 +02:00
for ( auto p : aoh - > artWorn )
p . second - > selectSlot ( art - > canBePutAt ( ArtifactLocation ( aoh - > curHero , p . second - > slotID ) , true ) ) ;
2014-07-05 12:58:42 +03:00
safeRedraw ( ) ;
}
/**
* Unamarks all slots .
*/
2017-07-15 13:08:20 +02:00
void CArtifactsOfHero : : unmarkSlots ( bool withRedraw )
2014-07-05 12:58:42 +03:00
{
if ( commonInfo )
for ( CArtifactsOfHero * aoh : commonInfo - > participants )
aoh - > unmarkLocalSlots ( false ) ;
else
unmarkLocalSlots ( false ) ; \
if ( withRedraw )
safeRedraw ( ) ;
}
2017-07-15 13:08:20 +02:00
void CArtifactsOfHero : : unmarkLocalSlots ( bool withRedraw )
2014-07-05 12:58:42 +03:00
{
2018-04-07 13:34:11 +02:00
for ( auto & p : artWorn )
2015-02-14 17:34:27 +02:00
p . second - > selectSlot ( false ) ;
2018-04-07 13:34:11 +02:00
for ( auto & place : backpack )
2014-07-05 12:58:42 +03:00
place - > selectSlot ( false ) ;
if ( withRedraw )
safeRedraw ( ) ;
}
/**
* Assigns an artifacts to an artifact place depending on it ' s new slot ID .
*/
2018-04-07 13:34:11 +02:00
void CArtifactsOfHero : : setSlotData ( ArtPlacePtr artPlace , ArtifactPosition slotID )
2014-07-05 12:58:42 +03:00
{
if ( ! artPlace & & slotID > = GameConstants : : BACKPACK_START ) //spurious call from artifactMoved in attempt to update hidden backpack slot
{
return ;
}
artPlace - > pickSlot ( false ) ;
artPlace - > slotID = slotID ;
if ( const ArtSlotInfo * asi = curHero - > getSlot ( slotID ) )
{
artPlace - > lockSlot ( asi - > locked ) ;
2016-01-21 09:49:46 +02:00
artPlace - > setArtifact ( asi - > artifact ) ;
2014-07-05 12:58:42 +03:00
}
else
artPlace - > setArtifact ( nullptr ) ;
}
/**
* Makes given artifact slot appear as empty with a certain slot ID .
*/
2018-04-07 13:34:11 +02:00
void CArtifactsOfHero : : eraseSlotData ( ArtPlacePtr artPlace , ArtifactPosition slotID )
2014-07-05 12:58:42 +03:00
{
artPlace - > pickSlot ( false ) ;
artPlace - > slotID = slotID ;
artPlace - > setArtifact ( nullptr ) ;
}
2018-04-07 13:34:11 +02:00
CArtifactsOfHero : : CArtifactsOfHero ( ArtPlaceMap ArtWorn , std : : vector < ArtPlacePtr > Backpack ,
std : : shared_ptr < CButton > leftScroll , std : : shared_ptr < CButton > rightScroll , bool createCommonPart )
: curHero ( nullptr ) ,
artWorn ( ArtWorn ) ,
backpack ( Backpack ) ,
backpackPos ( 0 ) ,
commonInfo ( nullptr ) ,
updateState ( false ) ,
leftArtRoll ( leftScroll ) ,
rightArtRoll ( rightScroll ) ,
allowedAssembling ( true ) ,
highlightModeCallback ( nullptr )
2014-07-05 12:58:42 +03:00
{
if ( createCommonPart )
{
2016-08-30 06:05:31 +02:00
commonInfo = std : : make_shared < CArtifactsOfHero : : SCommonPart > ( ) ;
2014-07-05 12:58:42 +03:00
commonInfo - > participants . insert ( this ) ;
}
// Init slots for worn artifacts.
2018-04-07 13:34:11 +02:00
for ( auto p : artWorn )
2014-07-05 12:58:42 +03:00
{
2015-02-14 17:34:27 +02:00
p . second - > ourOwner = this ;
eraseSlotData ( p . second , p . first ) ;
2014-07-05 12:58:42 +03:00
}
// Init slots for the backpack.
for ( size_t s = 0 ; s < backpack . size ( ) ; + + s )
{
backpack [ s ] - > ourOwner = this ;
2020-10-01 10:38:06 +02:00
eraseSlotData ( backpack [ s ] , ArtifactPosition ( GameConstants : : BACKPACK_START + ( si32 ) s ) ) ;
2014-07-05 12:58:42 +03:00
}
2018-04-07 13:34:11 +02:00
leftArtRoll - > addCallback ( std : : bind ( & CArtifactsOfHero : : scrollBackpack , this , - 1 ) ) ;
rightArtRoll - > addCallback ( std : : bind ( & CArtifactsOfHero : : scrollBackpack , this , + 1 ) ) ;
2014-07-05 12:58:42 +03:00
}
2018-04-07 13:34:11 +02:00
CArtifactsOfHero : : CArtifactsOfHero ( const Point & position , bool createCommonPart )
: curHero ( nullptr ) ,
backpackPos ( 0 ) ,
commonInfo ( nullptr ) ,
updateState ( false ) ,
allowedAssembling ( true ) ,
highlightModeCallback ( nullptr )
2014-07-05 12:58:42 +03:00
{
if ( createCommonPart )
{
2016-08-30 06:05:31 +02:00
commonInfo = std : : make_shared < CArtifactsOfHero : : SCommonPart > ( ) ;
2014-07-05 12:58:42 +03:00
commonInfo - > participants . insert ( this ) ;
}
2018-04-07 13:34:11 +02:00
OBJECT_CONSTRUCTION_CAPTURING ( 255 - DISPOSE ) ;
2014-07-05 12:58:42 +03:00
pos + = position ;
2014-10-02 18:43:46 +03:00
std : : vector < Point > slotPos =
{
2015-02-14 14:43:03 +02:00
Point ( 509 , 30 ) , Point ( 567 , 240 ) , Point ( 509 , 80 ) , //0-2
Point ( 383 , 68 ) , Point ( 564 , 183 ) , Point ( 509 , 130 ) , //3-5
Point ( 431 , 68 ) , Point ( 610 , 183 ) , Point ( 515 , 295 ) , //6-8
Point ( 383 , 143 ) , Point ( 399 , 194 ) , Point ( 415 , 245 ) , //9-11
Point ( 431 , 296 ) , Point ( 564 , 30 ) , Point ( 610 , 30 ) , //12-14
Point ( 610 , 76 ) , Point ( 610 , 122 ) , Point ( 610 , 310 ) , //15-17
Point ( 381 , 296 ) //18
2014-10-02 18:43:46 +03:00
} ;
2014-07-05 12:58:42 +03:00
// Create slots for worn artifacts.
2020-10-01 10:38:06 +02:00
for ( si32 g = 0 ; g < GameConstants : : BACKPACK_START ; g + + )
2014-07-05 12:58:42 +03:00
{
2018-04-07 13:34:11 +02:00
artWorn [ ArtifactPosition ( g ) ] = std : : make_shared < CHeroArtPlace > ( slotPos [ g ] ) ;
2015-02-14 18:23:19 +02:00
artWorn [ ArtifactPosition ( g ) ] - > ourOwner = this ;
eraseSlotData ( artWorn [ ArtifactPosition ( g ) ] , ArtifactPosition ( g ) ) ;
2014-07-05 12:58:42 +03:00
}
// Create slots for the backpack.
2020-10-01 10:38:06 +02:00
for ( int s = 0 ; s < 5 ; + + s )
2014-07-05 12:58:42 +03:00
{
2018-04-07 13:34:11 +02:00
auto add = std : : make_shared < CHeroArtPlace > ( Point ( 403 + 46 * s , 365 ) ) ;
2014-07-05 12:58:42 +03:00
add - > ourOwner = this ;
eraseSlotData ( add , ArtifactPosition ( GameConstants : : BACKPACK_START + s ) ) ;
backpack . push_back ( add ) ;
}
2018-04-07 13:34:11 +02:00
leftArtRoll = std : : make_shared < CButton > ( Point ( 379 , 364 ) , " hsbtns3.def " , CButton : : tooltip ( ) , [ & ] ( ) { scrollBackpack ( - 1 ) ; } , SDLK_LEFT ) ;
rightArtRoll = std : : make_shared < CButton > ( Point ( 632 , 364 ) , " hsbtns5.def " , CButton : : tooltip ( ) , [ & ] ( ) { scrollBackpack ( + 1 ) ; } , SDLK_RIGHT ) ;
2014-07-05 12:58:42 +03:00
}
CArtifactsOfHero : : ~ CArtifactsOfHero ( )
{
dispose ( ) ;
}
void CArtifactsOfHero : : updateParentWindow ( )
{
2018-07-25 00:36:48 +02:00
if ( CHeroWindow * chw = dynamic_cast < CHeroWindow * > ( GH . topInt ( ) . get ( ) ) )
2014-07-05 12:58:42 +03:00
{
if ( updateState )
chw - > curHero = curHero ;
else
chw - > update ( curHero , true ) ;
}
2018-07-25 00:36:48 +02:00
else if ( CExchangeWindow * cew = dynamic_cast < CExchangeWindow * > ( GH . topInt ( ) . get ( ) ) )
2014-07-05 12:58:42 +03:00
{
//use our copy of hero to draw window
if ( cew - > heroInst [ 0 ] - > id = = curHero - > id )
cew - > heroInst [ 0 ] = curHero ;
else
cew - > heroInst [ 1 ] = curHero ;
if ( ! updateState )
{
cew - > deactivate ( ) ;
2018-04-07 13:34:11 +02:00
cew - > updateWidgets ( ) ;
2014-07-05 12:58:42 +03:00
cew - > redraw ( ) ;
cew - > activate ( ) ;
}
}
}
void CArtifactsOfHero : : safeRedraw ( )
{
if ( active )
{
if ( parent )
parent - > redraw ( ) ;
else
redraw ( ) ;
}
}
void CArtifactsOfHero : : realizeCurrentTransaction ( )
{
assert ( commonInfo - > src . AOH ) ;
assert ( commonInfo - > dst . AOH ) ;
LOCPLINT - > cb - > swapArtifacts ( ArtifactLocation ( commonInfo - > src . AOH - > curHero , commonInfo - > src . slotID ) ,
ArtifactLocation ( commonInfo - > dst . AOH - > curHero , commonInfo - > dst . slotID ) ) ;
}
void CArtifactsOfHero : : artifactMoved ( const ArtifactLocation & src , const ArtifactLocation & dst )
{
bool isCurHeroSrc = src . isHolder ( curHero ) ,
isCurHeroDst = dst . isHolder ( curHero ) ;
if ( isCurHeroSrc & & src . slot > = GameConstants : : BACKPACK_START )
updateSlot ( src . slot ) ;
if ( isCurHeroDst & & dst . slot > = GameConstants : : BACKPACK_START )
updateSlot ( dst . slot ) ;
if ( isCurHeroSrc | | isCurHeroDst ) //we need to update all slots, artifact might be combined and affect more slots
updateWornSlots ( false ) ;
if ( ! src . isHolder ( curHero ) & & ! isCurHeroDst )
return ;
if ( commonInfo - > src = = src ) //artifact was taken from us
{
assert ( commonInfo - > dst = = dst //expected movement from slot ot slot
| | dst . slot = = dst . getHolderArtSet ( ) - > artifactsInBackpack . size ( ) + GameConstants : : BACKPACK_START //artifact moved back to backpack (eg. to make place for art we are moving)
| | dst . getHolderArtSet ( ) - > bearerType ( ) ! = ArtBearer : : HERO ) ;
commonInfo - > reset ( ) ;
unmarkSlots ( ) ;
}
else if ( commonInfo - > dst = = src ) //the dest artifact was moved -> we are picking it
{
assert ( dst . slot > = GameConstants : : BACKPACK_START ) ;
commonInfo - > reset ( ) ;
2018-04-07 13:34:11 +02:00
CArtifactsOfHero : : ArtPlacePtr ap ;
2014-07-05 12:58:42 +03:00
for ( CArtifactsOfHero * aoh : commonInfo - > participants )
{
if ( dst . isHolder ( aoh - > curHero ) )
{
commonInfo - > src . AOH = aoh ;
2015-02-14 14:58:33 +02:00
if ( ( ap = aoh - > getArtPlace ( dst . slot ) ) ) //getArtPlace may return null
2014-07-05 12:58:42 +03:00
break ;
}
}
if ( ap )
{
ap - > select ( ) ;
}
else
{
commonInfo - > src . art = dst . getArt ( ) ;
commonInfo - > src . slotID = dst . slot ;
assert ( commonInfo - > src . AOH ) ;
2016-10-28 12:39:16 +02:00
CCS - > curh - > dragAndDropCursor ( make_unique < CAnimImage > ( " artifact " , dst . getArt ( ) - > artType - > iconIndex ) ) ;
2014-07-05 12:58:42 +03:00
markPossibleSlots ( dst . getArt ( ) ) ;
}
}
else if ( src . slot > = GameConstants : : BACKPACK_START & &
src . slot < commonInfo - > src . slotID & &
2016-01-21 19:23:45 +02:00
src . isHolder ( commonInfo - > src . AOH - > curHero ) ) //artifact taken from before currently picked one
2014-07-05 12:58:42 +03:00
{
//int fixedSlot = src.hero->getArtPos(commonInfo->src.art);
vstd : : advance ( commonInfo - > src . slotID , - 1 ) ;
assert ( commonInfo - > src . valid ( ) ) ;
}
else
{
//when moving one artifact onto another it leads to two art movements: dst->backapck; src->dst
// however after first movement we pick the art from backpack and the second movement coming when
// we have a different artifact may look surprising... but it's valid.
}
updateParentWindow ( ) ;
2016-01-21 19:23:45 +02:00
int shift = 0 ;
2014-07-05 12:58:42 +03:00
// if(dst.slot >= Arts::BACKPACK_START && dst.slot - Arts::BACKPACK_START < backpackPos)
// shift++;
//
2016-01-21 19:23:45 +02:00
if ( src . slot < GameConstants : : BACKPACK_START & & dst . slot - GameConstants : : BACKPACK_START < backpackPos )
2014-07-05 12:58:42 +03:00
shift + + ;
if ( dst . slot < GameConstants : : BACKPACK_START & & src . slot - GameConstants : : BACKPACK_START < backpackPos )
2016-01-21 19:23:45 +02:00
shift - - ;
2014-07-05 12:58:42 +03:00
if ( ( isCurHeroSrc & & src . slot > = GameConstants : : BACKPACK_START )
| | ( isCurHeroDst & & dst . slot > = GameConstants : : BACKPACK_START ) )
scrollBackpack ( shift ) ; //update backpack slots
}
void CArtifactsOfHero : : artifactRemoved ( const ArtifactLocation & al )
{
if ( al . isHolder ( curHero ) )
{
if ( al . slot < GameConstants : : BACKPACK_START )
updateWornSlots ( 0 ) ;
else
scrollBackpack ( 0 ) ; //update backpack slots
}
}
2018-04-07 13:34:11 +02:00
CArtifactsOfHero : : ArtPlacePtr CArtifactsOfHero : : getArtPlace ( int slot )
2014-07-05 12:58:42 +03:00
{
if ( slot < GameConstants : : BACKPACK_START )
{
2015-02-14 18:23:19 +02:00
if ( artWorn . find ( ArtifactPosition ( slot ) ) = = artWorn . end ( ) )
2015-02-14 14:43:03 +02:00
{
2017-08-11 13:38:10 +02:00
logGlobal - > error ( " CArtifactsOfHero::getArtPlace: invalid slot %d " , slot ) ;
2015-02-14 14:43:03 +02:00
return nullptr ;
}
2015-02-14 18:23:19 +02:00
return artWorn [ ArtifactPosition ( slot ) ] ;
2014-07-05 12:58:42 +03:00
}
else
{
2018-04-07 13:34:11 +02:00
for ( ArtPlacePtr ap : backpack )
2014-07-05 12:58:42 +03:00
if ( ap - > slotID = = slot )
return ap ;
2016-01-21 09:49:46 +02:00
return nullptr ;
2014-07-05 12:58:42 +03:00
}
}
void CArtifactsOfHero : : artifactAssembled ( const ArtifactLocation & al )
{
if ( al . isHolder ( curHero ) )
updateWornSlots ( ) ;
}
void CArtifactsOfHero : : artifactDisassembled ( const ArtifactLocation & al )
{
if ( al . isHolder ( curHero ) )
updateWornSlots ( ) ;
}
2017-07-15 13:08:20 +02:00
void CArtifactsOfHero : : updateWornSlots ( bool redrawParent )
2014-07-05 12:58:42 +03:00
{
2015-02-14 17:34:27 +02:00
for ( auto p : artWorn )
updateSlot ( p . first ) ;
2014-07-05 12:58:42 +03:00
if ( redrawParent )
updateParentWindow ( ) ;
}
const CGHeroInstance * CArtifactsOfHero : : getHero ( ) const
{
return curHero ;
}
void CArtifactsOfHero : : updateSlot ( ArtifactPosition slotID )
{
setSlotData ( getArtPlace ( slotID ) , slotID ) ;
}
CArtifactHolder : : CArtifactHolder ( )
{
}
2018-04-07 13:34:11 +02:00
void CWindowWithArtifacts : : addSet ( std : : shared_ptr < CArtifactsOfHero > artSet )
{
artSets . emplace_back ( artSet ) ;
}
std : : shared_ptr < CArtifactsOfHero : : SCommonPart > CWindowWithArtifacts : : getCommonPart ( )
{
for ( auto artSetWeak : artSets )
{
std : : shared_ptr < CArtifactsOfHero > realPtr = artSetWeak . lock ( ) ;
if ( realPtr )
return realPtr - > commonInfo ;
}
return std : : shared_ptr < CArtifactsOfHero : : SCommonPart > ( ) ;
}
2014-07-05 12:58:42 +03:00
void CWindowWithArtifacts : : artifactRemoved ( const ArtifactLocation & artLoc )
{
2018-04-07 13:34:11 +02:00
for ( auto artSetWeak : artSets )
{
std : : shared_ptr < CArtifactsOfHero > realPtr = artSetWeak . lock ( ) ;
if ( realPtr )
realPtr - > artifactRemoved ( artLoc ) ;
}
2014-07-05 12:58:42 +03:00
}
void CWindowWithArtifacts : : artifactMoved ( const ArtifactLocation & artLoc , const ArtifactLocation & destLoc )
{
2018-04-07 13:34:11 +02:00
CArtifactsOfHero * destaoh = nullptr ;
for ( auto artSetWeak : artSets )
2014-07-05 12:58:42 +03:00
{
2018-04-07 13:34:11 +02:00
std : : shared_ptr < CArtifactsOfHero > realPtr = artSetWeak . lock ( ) ;
if ( realPtr )
{
realPtr - > artifactMoved ( artLoc , destLoc ) ;
realPtr - > redraw ( ) ;
if ( destLoc . isHolder ( realPtr - > getHero ( ) ) )
destaoh = realPtr . get ( ) ;
}
2014-07-05 12:58:42 +03:00
}
//Make sure the status bar is updated so it does not display old text
if ( destaoh ! = nullptr & & destaoh - > getArtPlace ( destLoc . slot ) ! = nullptr )
{
destaoh - > getArtPlace ( destLoc . slot ) - > hover ( true ) ;
}
}
void CWindowWithArtifacts : : artifactDisassembled ( const ArtifactLocation & artLoc )
{
2018-04-07 13:34:11 +02:00
for ( auto artSetWeak : artSets )
{
std : : shared_ptr < CArtifactsOfHero > realPtr = artSetWeak . lock ( ) ;
if ( realPtr )
realPtr - > artifactDisassembled ( artLoc ) ;
}
2014-07-05 12:58:42 +03:00
}
void CWindowWithArtifacts : : artifactAssembled ( const ArtifactLocation & artLoc )
{
2018-04-07 13:34:11 +02:00
for ( auto artSetWeak : artSets )
{
std : : shared_ptr < CArtifactsOfHero > realPtr = artSetWeak . lock ( ) ;
if ( realPtr )
realPtr - > artifactAssembled ( artLoc ) ;
}
2014-07-05 12:58:42 +03:00
}
void CArtifactsOfHero : : SCommonPart : : Artpos : : clear ( )
{
slotID = ArtifactPosition : : PRE_FIRST ;
AOH = nullptr ;
art = nullptr ;
}
CArtifactsOfHero : : SCommonPart : : Artpos : : Artpos ( )
{
clear ( ) ;
}
2016-11-04 18:54:09 +02:00
void CArtifactsOfHero : : SCommonPart : : Artpos : : setTo ( const CHeroArtPlace * place , bool dontTakeBackpack )
2014-07-05 12:58:42 +03:00
{
slotID = place - > slotID ;
AOH = place - > ourOwner ;
if ( slotID > = 19 & & dontTakeBackpack )
art = nullptr ;
else
art = place - > ourArt ;
}
bool CArtifactsOfHero : : SCommonPart : : Artpos : : operator = = ( const ArtifactLocation & al ) const
{
if ( ! AOH )
return false ;
bool ret = al . isHolder ( AOH - > curHero ) & & al . slot = = slotID ;
//assert(al.getArt() == art);
return ret ;
}
bool CArtifactsOfHero : : SCommonPart : : Artpos : : valid ( )
{
assert ( AOH & & art ) ;
return art = = AOH - > curHero - > getArt ( slotID ) ;
}
2016-11-04 18:54:09 +02:00
CArtPlace : : CArtPlace ( Point position , const CArtifactInstance * Art ) : ourArt ( Art )
{
2016-11-27 17:17:20 +02:00
image = nullptr ;
2016-11-04 18:54:09 +02:00
pos + = position ;
pos . w = pos . h = 44 ;
}
void CArtPlace : : clickLeft ( tribool down , bool previousState )
{
LRClickableAreaWTextComp : : clickLeft ( down , previousState ) ;
}
void CArtPlace : : clickRight ( tribool down , bool previousState )
{
LRClickableAreaWTextComp : : clickRight ( down , previousState ) ;
}
2016-11-06 14:02:00 +02:00
CCommanderArtPlace : : CCommanderArtPlace ( Point position , const CGHeroInstance * commanderOwner , ArtifactPosition artSlot , const CArtifactInstance * Art ) : CArtPlace ( position , Art ) , commanderOwner ( commanderOwner ) , commanderSlotID ( artSlot . num )
2016-11-04 18:54:09 +02:00
{
createImage ( ) ;
setArtifact ( Art ) ;
}
void CCommanderArtPlace : : clickLeft ( tribool down , bool previousState )
{
2016-11-27 22:41:17 +02:00
if ( ourArt & & text . size ( ) & & down )
2017-07-17 14:35:57 +02:00
LOCPLINT - > showYesNoDialog ( CGI - > generaltexth - > localizedTexts [ " commanderWindow " ] [ " artifactMessage " ] . String ( ) , [ this ] ( ) { returnArtToHeroCallback ( ) ; } , [ ] ( ) { } ) ;
2016-11-04 18:54:09 +02:00
}
void CCommanderArtPlace : : clickRight ( tribool down , bool previousState )
{
2016-11-27 22:41:17 +02:00
if ( ourArt & & text . size ( ) & & down )
2016-11-04 18:54:09 +02:00
CArtPlace : : clickRight ( down , previousState ) ;
}
void CCommanderArtPlace : : createImage ( )
{
2018-04-07 13:34:11 +02:00
OBJECT_CONSTRUCTION_CAPTURING ( 255 - DISPOSE ) ;
2016-11-04 18:54:09 +02:00
2016-11-08 18:26:24 +02:00
int imageIndex = 0 ;
2018-04-07 13:34:11 +02:00
if ( ourArt )
2016-11-08 18:26:24 +02:00
imageIndex = ourArt - > artType - > iconIndex ;
2016-11-04 18:54:09 +02:00
2018-04-07 13:34:11 +02:00
image = std : : make_shared < CAnimImage > ( " artifact " , imageIndex ) ;
if ( ! ourArt )
2016-11-04 18:54:09 +02:00
image - > disable ( ) ;
}
2016-11-06 14:02:00 +02:00
void CCommanderArtPlace : : returnArtToHeroCallback ( )
{
2017-07-12 21:01:10 +02:00
ArtifactPosition artifactPos = commanderSlotID ;
2016-11-06 14:02:00 +02:00
ArtifactPosition freeSlot = ourArt - > firstBackpackSlot ( commanderOwner ) ;
ArtifactLocation src ( commanderOwner - > commander . get ( ) , artifactPos ) ;
ArtifactLocation dst ( commanderOwner , freeSlot ) ;
if ( ourArt - > canBePutAt ( dst , true ) )
{
LOCPLINT - > cb - > swapArtifacts ( src , dst ) ;
setArtifact ( nullptr ) ;
parent - > redraw ( ) ;
}
}
2016-11-04 18:54:09 +02:00
void CCommanderArtPlace : : setArtifact ( const CArtifactInstance * art )
{
baseType = - 1 ; //by default we don't store any component
ourArt = art ;
if ( ! art )
{
image - > disable ( ) ;
text = std : : string ( ) ;
return ;
}
image - > enable ( ) ;
image - > setFrame ( art - > artType - > iconIndex ) ;
text = art - > getEffectiveDescription ( ) ;
if ( art - > artType - > id = = ArtifactID : : SPELL_SCROLL )
{
int spellID = art - > getGivenSpellID ( ) ;
if ( spellID > = 0 )
{
//add spell component info (used to provide a pic in r-click popup)
baseType = CComponent : : spell ;
type = spellID ;
bonusValue = 0 ;
}
}
else
{
baseType = CComponent : : artifact ;
type = art - > artType - > id ;
bonusValue = 0 ;
}
}