1
0
mirror of https://github.com/vcmi/vcmi.git synced 2024-12-16 10:19:47 +02:00
vcmi/client/widgets/CArtifactsOfHeroBase.cpp

277 lines
7.3 KiB
C++
Raw Normal View History

2023-04-23 14:10:35 +02:00
/*
* CArtifactsOfHeroBase.cpp, part of VCMI engine
*
* Authors: listed in file AUTHORS in main folder
*
* License: GNU General Public License v2.0 or later
* Full text of license available in license.txt file, in main folder
*
*/
#include "StdInc.h"
#include "CArtifactsOfHeroBase.h"
#include "../gui/CGuiHandler.h"
2023-05-01 00:25:48 +02:00
#include "../gui/Shortcut.h"
2023-04-23 14:10:35 +02:00
#include "Buttons.h"
#include "../CPlayerInterface.h"
#include "../CGameInfo.h"
#include "../../CCallback.h"
2023-05-17 15:52:16 +02:00
#include "../../lib/ArtifactUtils.h"
2023-04-23 14:10:35 +02:00
#include "../../lib/mapObjects/CGHeroInstance.h"
#include "../../lib/networkPacks/ArtifactLocation.h"
2023-04-23 14:10:35 +02:00
CArtifactsOfHeroBase::CArtifactsOfHeroBase()
2024-04-16 16:45:31 +02:00
: curHero(nullptr)
2023-04-23 14:10:35 +02:00
{
}
2023-07-06 21:14:12 +02:00
void CArtifactsOfHeroBase::putBackPickedArtifact()
2023-04-23 14:10:35 +02:00
{
// Artifact located in artifactsTransitionPos should be returned
if(const auto art = getPickedArtifact())
2023-04-23 14:10:35 +02:00
{
auto slot = ArtifactUtils::getArtAnyPosition(curHero, art->getTypeId());
2023-04-23 14:10:35 +02:00
if(slot == ArtifactPosition::PRE_FIRST)
{
LOCPLINT->cb->eraseArtifactByClient(ArtifactLocation(curHero->id, ArtifactPosition::TRANSITION_POS));
2023-04-23 14:10:35 +02:00
}
else
{
LOCPLINT->cb->swapArtifacts(ArtifactLocation(curHero->id, ArtifactPosition::TRANSITION_POS), ArtifactLocation(curHero->id, slot));
2023-04-23 14:10:35 +02:00
}
}
}
void CArtifactsOfHeroBase::init(
2024-03-08 14:21:29 +02:00
const CArtPlace::ClickFunctor & onClickPressedCallback,
const CArtPlace::ClickFunctor & onShowPopupCallback,
2023-04-23 14:10:35 +02:00
const Point & position,
const BpackScrollFunctor & scrollCallback)
2023-04-23 14:10:35 +02:00
{
// CArtifactsOfHeroBase::init may be transform to CArtifactsOfHeroBase::CArtifactsOfHeroBase if OBJECT_CONSTRUCTION_CAPTURING is removed
OBJECT_CONSTRUCTION_CAPTURING(255 - DISPOSE);
pos += position;
for(int g = 0; g < ArtifactPosition::BACKPACK_START; g++)
2023-04-23 14:10:35 +02:00
{
artWorn[ArtifactPosition(g)] = std::make_shared<CHeroArtPlace>(slotPos[g]);
}
backpack.clear();
for(int s = 0; s < 5; s++)
{
auto artPlace = std::make_shared<CHeroArtPlace>(Point(403 + 46 * s, 365));
backpack.push_back(artPlace);
}
for(auto artPlace : artWorn)
{
artPlace.second->slot = artPlace.first;
artPlace.second->setArtifact(nullptr);
2024-03-08 14:21:29 +02:00
artPlace.second->setClickPressedCallback(onClickPressedCallback);
artPlace.second->setShowPopupCallback(onShowPopupCallback);
2023-04-23 14:10:35 +02:00
}
for(auto artPlace : backpack)
{
artPlace->setArtifact(nullptr);
2024-03-08 14:21:29 +02:00
artPlace->setClickPressedCallback(onClickPressedCallback);
artPlace->setShowPopupCallback(onShowPopupCallback);
2023-04-23 14:10:35 +02:00
}
leftBackpackRoll = std::make_shared<CButton>(Point(379, 364), AnimationPath::builtin("hsbtns3.def"), CButton::tooltip(),
2024-03-06 15:16:35 +02:00
[scrollCallback](){scrollCallback(true);}, EShortcut::MOVE_LEFT);
rightBackpackRoll = std::make_shared<CButton>(Point(632, 364), AnimationPath::builtin("hsbtns5.def"), CButton::tooltip(),
2024-03-06 15:16:35 +02:00
[scrollCallback](){scrollCallback(false);}, EShortcut::MOVE_RIGHT);
2023-04-23 14:10:35 +02:00
leftBackpackRoll->block(true);
rightBackpackRoll->block(true);
2023-09-03 20:41:00 +02:00
setRedrawParent(true);
2023-04-23 14:10:35 +02:00
}
2023-12-17 16:30:19 +02:00
void CArtifactsOfHeroBase::clickPrassedArtPlace(CArtPlace & artPlace, const Point & cursorPosition)
2023-04-23 14:10:35 +02:00
{
2024-05-02 00:49:17 +02:00
if(artPlace.isLocked())
return;
2023-12-17 16:30:19 +02:00
if(clickPressedCallback)
2024-05-02 00:49:17 +02:00
clickPressedCallback(artPlace, cursorPosition);
2023-04-23 14:10:35 +02:00
}
2023-12-17 16:30:19 +02:00
void CArtifactsOfHeroBase::showPopupArtPlace(CArtPlace & artPlace, const Point & cursorPosition)
2023-04-23 14:10:35 +02:00
{
2024-05-02 00:49:17 +02:00
if(artPlace.isLocked())
return;
2023-09-17 17:40:14 +02:00
if(showPopupCallback)
2024-05-02 00:49:17 +02:00
showPopupCallback(artPlace, cursorPosition);
2023-04-23 14:10:35 +02:00
}
2023-12-17 16:30:19 +02:00
void CArtifactsOfHeroBase::gestureArtPlace(CArtPlace & artPlace, const Point & cursorPosition)
{
2024-05-02 00:49:17 +02:00
if(artPlace.isLocked())
return;
2023-12-17 16:30:19 +02:00
if(gestureCallback)
2024-05-02 00:49:17 +02:00
gestureCallback(artPlace, cursorPosition);
2023-12-17 16:30:19 +02:00
}
2023-04-23 14:10:35 +02:00
void CArtifactsOfHeroBase::setHero(const CGHeroInstance * hero)
{
curHero = hero;
for(auto slot : artWorn)
{
2024-01-27 23:48:11 +02:00
setSlotData(slot.second, slot.first);
2023-04-23 14:10:35 +02:00
}
updateBackpackSlots();
2023-04-23 14:10:35 +02:00
}
const CGHeroInstance * CArtifactsOfHeroBase::getHero() const
{
return curHero;
}
2024-03-06 15:16:35 +02:00
void CArtifactsOfHeroBase::scrollBackpack(bool left)
2023-04-23 14:10:35 +02:00
{
2024-03-06 15:16:35 +02:00
LOCPLINT->cb->scrollBackpackArtifacts(curHero->id, left);
2023-04-23 14:10:35 +02:00
}
void CArtifactsOfHeroBase::markPossibleSlots(const CArtifactInstance * art, bool assumeDestRemoved)
{
for(auto artPlace : artWorn)
artPlace.second->selectSlot(art->canBePutAt(curHero, artPlace.second->slot, assumeDestRemoved));
2023-04-23 14:10:35 +02:00
}
void CArtifactsOfHeroBase::unmarkSlots()
{
for(auto & artPlace : artWorn)
artPlace.second->selectSlot(false);
for(auto & artPlace : backpack)
artPlace->selectSlot(false);
}
CArtifactsOfHeroBase::ArtPlacePtr CArtifactsOfHeroBase::getArtPlace(const ArtifactPosition & slot)
{
if(ArtifactUtils::isSlotEquipment(slot))
{
if(artWorn.find(slot) == artWorn.end())
{
logGlobal->error("CArtifactsOfHero::getArtPlace: invalid slot %d", slot);
return nullptr;
}
return artWorn[slot];
}
if(ArtifactUtils::isSlotBackpack(slot))
{
for(ArtPlacePtr artPlace : backpack)
if(artPlace->slot == slot)
return artPlace;
return nullptr;
}
else
{
return nullptr;
}
}
2024-05-01 20:18:36 +02:00
CArtifactsOfHeroBase::ArtPlacePtr CArtifactsOfHeroBase::getArtPlace(const Point & cursorPosition)
{
for(const auto & [slot, artPlace] : artWorn)
{
if(artPlace->pos.isInside(cursorPosition))
return artPlace;
}
for(const auto & artPlace : backpack)
{
if(artPlace->pos.isInside(cursorPosition))
return artPlace;
}
return nullptr;
}
2023-04-23 14:10:35 +02:00
void CArtifactsOfHeroBase::updateWornSlots()
{
for(auto place : artWorn)
updateSlot(place.first);
}
void CArtifactsOfHeroBase::updateBackpackSlots()
{
2024-03-06 15:16:35 +02:00
ArtifactPosition slot = ArtifactPosition::BACKPACK_START;
2024-03-08 14:21:29 +02:00
for(const auto & artPlace : backpack)
2024-03-06 15:16:35 +02:00
{
setSlotData(artPlace, slot);
slot = slot + 1;
}
auto scrollingPossible = static_cast<int>(curHero->artifactsInBackpack.size()) > backpack.size();
// Blocking scrolling if there is not enough artifacts to scroll
if(leftBackpackRoll)
leftBackpackRoll->block(!scrollingPossible);
if(rightBackpackRoll)
rightBackpackRoll->block(!scrollingPossible);
2023-04-23 14:10:35 +02:00
}
void CArtifactsOfHeroBase::updateSlot(const ArtifactPosition & slot)
{
2024-01-27 23:48:11 +02:00
setSlotData(getArtPlace(slot), slot);
2023-04-23 14:10:35 +02:00
}
const CArtifactInstance * CArtifactsOfHeroBase::getPickedArtifact()
{
// Returns only the picked up artifact. Not just highlighted like in the trading window.
if(curHero)
2023-04-23 14:10:35 +02:00
return curHero->getArt(ArtifactPosition::TRANSITION_POS);
else
return nullptr;
2023-04-23 14:10:35 +02:00
}
2023-12-17 16:30:19 +02:00
void CArtifactsOfHeroBase::addGestureCallback(CArtPlace::ClickFunctor callback)
{
for(auto & artPlace : artWorn)
{
artPlace.second->setGestureCallback(callback);
artPlace.second->addUsedEvents(GESTURE);
}
}
2024-05-02 00:49:17 +02:00
const CArtifactInstance * CArtifactsOfHeroBase::getArt(const ArtifactPosition & slot)
{
return curHero ? curHero->getArt(slot) : nullptr;
}
2024-01-27 23:48:11 +02:00
void CArtifactsOfHeroBase::setSlotData(ArtPlacePtr artPlace, const ArtifactPosition & slot)
2023-04-23 14:10:35 +02:00
{
// Spurious call from artifactMoved in attempt to update hidden backpack slot
if(!artPlace && ArtifactUtils::isSlotBackpack(slot))
{
return;
}
artPlace->slot = slot;
2024-01-27 23:48:11 +02:00
if(auto slotInfo = curHero->getSlot(slot))
2023-04-23 14:10:35 +02:00
{
artPlace->lockSlot(slotInfo->locked);
artPlace->setArtifact(slotInfo->artifact);
2023-06-29 17:34:07 +02:00
if(!slotInfo->artifact->isCombined())
2023-04-23 14:10:35 +02:00
{
// If the artifact is part of at least one combined artifact, add additional information
std::map<const CArtifact*, int> arts;
2023-07-03 18:15:40 +02:00
for(const auto combinedArt : slotInfo->artifact->artType->getPartOf())
2023-04-23 14:10:35 +02:00
{
arts.insert(std::pair(combinedArt, 0));
2023-07-03 18:15:40 +02:00
for(const auto part : combinedArt->getConstituents())
2023-09-12 17:30:48 +02:00
{
2024-01-27 23:48:11 +02:00
if(curHero->hasArt(part->getId(), false))
2023-04-23 14:10:35 +02:00
arts.at(combinedArt)++;
2023-09-12 17:30:48 +02:00
}
2023-04-23 14:10:35 +02:00
}
artPlace->addCombinedArtInfo(arts);
}
}
else
{
artPlace->setArtifact(nullptr);
}
}