2023-12-04 23:21:49 +02:00
|
|
|
/*
|
|
|
|
* CAltarArtifacts.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 "CAltarArtifacts.h"
|
|
|
|
|
|
|
|
#include "../../gui/CGuiHandler.h"
|
2024-05-18 21:59:28 +02:00
|
|
|
#include "../../gui/Shortcut.h"
|
2023-12-04 23:21:49 +02:00
|
|
|
#include "../../widgets/Buttons.h"
|
|
|
|
#include "../../widgets/TextControls.h"
|
|
|
|
|
|
|
|
#include "../../CGameInfo.h"
|
|
|
|
#include "../../CPlayerInterface.h"
|
|
|
|
|
|
|
|
#include "../../../CCallback.h"
|
|
|
|
|
|
|
|
#include "../../../lib/networkPacks/ArtifactLocation.h"
|
2024-07-20 14:55:17 +02:00
|
|
|
#include "../../../lib/texts/CGeneralTextHandler.h"
|
2023-12-04 23:21:49 +02:00
|
|
|
#include "../../../lib/mapObjects/CGHeroInstance.h"
|
2024-08-20 16:15:50 +02:00
|
|
|
#include "../../../lib/mapObjects/IMarket.h"
|
2023-12-04 23:21:49 +02:00
|
|
|
|
2024-08-17 21:06:48 +02:00
|
|
|
CAltarArtifacts::CAltarArtifacts(const IMarket * market, const CGHeroInstance * hero)
|
2024-03-20 13:28:19 +02:00
|
|
|
: CMarketBase(market, hero)
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
2024-08-09 17:30:04 +02:00
|
|
|
OBJECT_CONSTRUCTION;
|
2023-12-04 23:21:49 +02:00
|
|
|
|
2024-08-17 19:19:48 +02:00
|
|
|
assert(market->getArtifactsStorage());
|
|
|
|
altarArtifactsStorage = market->getArtifactsStorage();
|
2024-01-12 23:57:19 +02:00
|
|
|
|
2024-03-02 20:30:29 +02:00
|
|
|
deal = std::make_shared<CButton>(Point(269, 520), AnimationPath::builtin("ALTSACR.DEF"),
|
2024-05-18 21:59:28 +02:00
|
|
|
CGI->generaltexth->zelp[585], [this]() {CAltarArtifacts::makeDeal(); }, EShortcut::MARKET_DEAL);
|
2024-02-29 12:02:39 +02:00
|
|
|
labels.emplace_back(std::make_shared<CLabel>(450, 32, FONT_SMALL, ETextAlignment::CENTER, Colors::YELLOW, CGI->generaltexth->allTexts[477]));
|
|
|
|
labels.emplace_back(std::make_shared<CLabel>(302, 424, FONT_SMALL, ETextAlignment::CENTER, Colors::YELLOW, CGI->generaltexth->allTexts[478]));
|
2023-12-04 23:21:49 +02:00
|
|
|
|
|
|
|
sacrificeAllButton = std::make_shared<CButton>(Point(393, 520), AnimationPath::builtin("ALTFILL.DEF"),
|
2024-05-18 21:59:28 +02:00
|
|
|
CGI->generaltexth->zelp[571], std::bind(&CExperienceAltar::sacrificeAll, this), EShortcut::MARKET_SACRIFICE_ALL);
|
2023-12-04 23:21:49 +02:00
|
|
|
sacrificeAllButton->block(hero->artifactsInBackpack.empty() && hero->artifactsWorn.empty());
|
|
|
|
|
|
|
|
sacrificeBackpackButton = std::make_shared<CButton>(Point(147, 520), AnimationPath::builtin("ALTEMBK.DEF"),
|
2024-05-18 21:59:28 +02:00
|
|
|
CGI->generaltexth->zelp[570], std::bind(&CAltarArtifacts::sacrificeBackpack, this), EShortcut::MARKET_SACRIFICE_BACKPACK);
|
2023-12-04 23:21:49 +02:00
|
|
|
sacrificeBackpackButton->block(hero->artifactsInBackpack.empty());
|
|
|
|
|
2024-02-25 22:58:53 +02:00
|
|
|
// Hero's artifacts
|
2024-01-16 16:22:40 +02:00
|
|
|
heroArts = std::make_shared<CArtifactsOfHeroAltar>(Point(-365, -11));
|
|
|
|
heroArts->setHero(hero);
|
2024-08-20 16:15:50 +02:00
|
|
|
heroArts->altarId = market->getObjInstanceID();
|
2023-12-04 23:21:49 +02:00
|
|
|
|
2024-02-29 12:02:39 +02:00
|
|
|
// Altar
|
|
|
|
offerTradePanel = std::make_shared<ArtifactsAltarPanel>([this](const std::shared_ptr<CTradeableItem> & altarSlot)
|
|
|
|
{
|
2024-03-19 13:53:23 +02:00
|
|
|
CAltarArtifacts::onSlotClickPressed(altarSlot, offerTradePanel);
|
2024-02-29 12:02:39 +02:00
|
|
|
});
|
|
|
|
offerTradePanel->updateSlotsCallback = std::bind(&CAltarArtifacts::updateAltarSlots, this);
|
|
|
|
offerTradePanel->moveTo(pos.topLeft() + Point(315, 52));
|
2023-12-04 23:21:49 +02:00
|
|
|
|
2024-03-19 13:53:23 +02:00
|
|
|
CMarketBase::updateShowcases();
|
2024-02-29 12:02:39 +02:00
|
|
|
CAltarArtifacts::deselect();
|
2023-12-04 23:21:49 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
TExpType CAltarArtifacts::calcExpAltarForHero()
|
|
|
|
{
|
|
|
|
TExpType expOnAltar(0);
|
2024-01-16 16:22:40 +02:00
|
|
|
for(const auto & tradeSlot : tradeSlotsMap)
|
2024-02-29 12:02:39 +02:00
|
|
|
expOnAltar += calcExpCost(tradeSlot.second->getTypeId());
|
2024-01-16 16:22:40 +02:00
|
|
|
expForHero->setText(std::to_string(expOnAltar));
|
|
|
|
return expOnAltar;
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
|
|
|
|
2024-02-29 12:02:39 +02:00
|
|
|
void CAltarArtifacts::deselect()
|
|
|
|
{
|
2024-03-02 20:30:29 +02:00
|
|
|
CMarketBase::deselect();
|
|
|
|
CExperienceAltar::deselect();
|
2024-02-29 12:02:39 +02:00
|
|
|
tradeSlotsMap.clear();
|
|
|
|
// The event for removing artifacts from the altar will not be triggered. Therefore, we clean the altar immediately.
|
2024-03-19 13:53:23 +02:00
|
|
|
for(const auto & slot : offerTradePanel->slots)
|
|
|
|
slot->clear();
|
|
|
|
offerTradePanel->showcaseSlot->clear();
|
2024-02-29 12:02:39 +02:00
|
|
|
}
|
|
|
|
|
2024-03-02 20:30:29 +02:00
|
|
|
void CAltarArtifacts::update()
|
2024-02-29 22:33:12 +02:00
|
|
|
{
|
2024-03-02 20:30:29 +02:00
|
|
|
CMarketBase::update();
|
|
|
|
CExperienceAltar::update();
|
2024-02-29 22:33:12 +02:00
|
|
|
if(const auto art = hero->getArt(ArtifactPosition::TRANSITION_POS))
|
|
|
|
offerQty = calcExpCost(art->getTypeId());
|
|
|
|
else
|
|
|
|
offerQty = 0;
|
2024-03-19 13:53:23 +02:00
|
|
|
updateShowcases();
|
2024-03-29 12:29:16 +02:00
|
|
|
redraw();
|
2024-02-29 22:33:12 +02:00
|
|
|
}
|
|
|
|
|
2023-12-04 23:21:49 +02:00
|
|
|
void CAltarArtifacts::makeDeal()
|
|
|
|
{
|
|
|
|
std::vector<TradeItemSell> positions;
|
2024-02-29 12:02:39 +02:00
|
|
|
for(const auto & [altarSlot, artInst] : tradeSlotsMap)
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
2024-01-16 21:54:00 +02:00
|
|
|
positions.push_back(artInst->getId());
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
2024-08-20 16:15:50 +02:00
|
|
|
LOCPLINT->cb->trade(market->getObjInstanceID(), EMarketMode::ARTIFACT_EXP, positions, std::vector<TradeItemBuy>(), std::vector<ui32>(), hero);
|
2024-02-29 12:02:39 +02:00
|
|
|
deselect();
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void CAltarArtifacts::sacrificeAll()
|
|
|
|
{
|
2024-05-08 13:16:16 +02:00
|
|
|
LOCPLINT->cb->bulkMoveArtifacts(heroArts->getHero()->id, heroArts->altarId, false, true, true);
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void CAltarArtifacts::sacrificeBackpack()
|
|
|
|
{
|
2024-05-08 13:16:16 +02:00
|
|
|
LOCPLINT->cb->bulkMoveArtifacts(heroArts->getHero()->id, heroArts->altarId, false, false, true);
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
|
|
|
|
2024-01-16 16:22:40 +02:00
|
|
|
std::shared_ptr<CArtifactsOfHeroAltar> CAltarArtifacts::getAOHset() const
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
2024-01-16 16:22:40 +02:00
|
|
|
return heroArts;
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
|
|
|
|
2024-02-29 12:02:39 +02:00
|
|
|
void CAltarArtifacts::updateAltarSlots()
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
2024-08-17 19:19:48 +02:00
|
|
|
assert(altarArtifactsStorage->artifactsInBackpack.size() <= GameConstants::ALTAR_ARTIFACTS_SLOTS);
|
2024-01-16 21:54:00 +02:00
|
|
|
assert(tradeSlotsMap.size() <= GameConstants::ALTAR_ARTIFACTS_SLOTS);
|
2024-01-16 16:22:40 +02:00
|
|
|
|
2024-02-29 12:02:39 +02:00
|
|
|
auto tradeSlotsMapNewArts = tradeSlotsMap;
|
2024-03-19 13:53:23 +02:00
|
|
|
for(const auto & altarSlot : offerTradePanel->slots)
|
2024-01-16 16:22:40 +02:00
|
|
|
if(altarSlot->id != -1)
|
2024-01-27 23:48:11 +02:00
|
|
|
{
|
2024-02-29 12:02:39 +02:00
|
|
|
if(tradeSlotsMap.find(altarSlot) == tradeSlotsMap.end())
|
2024-01-16 16:22:40 +02:00
|
|
|
{
|
|
|
|
altarSlot->setID(-1);
|
2024-02-27 13:39:50 +02:00
|
|
|
altarSlot->subtitle->clear();
|
2024-01-16 16:22:40 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-02-29 12:02:39 +02:00
|
|
|
tradeSlotsMapNewArts.erase(altarSlot);
|
2024-01-16 16:22:40 +02:00
|
|
|
}
|
2024-01-27 23:48:11 +02:00
|
|
|
}
|
2023-12-04 23:21:49 +02:00
|
|
|
|
2024-02-29 12:02:39 +02:00
|
|
|
for(auto & tradeSlot : tradeSlotsMapNewArts)
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
2024-02-29 12:02:39 +02:00
|
|
|
assert(tradeSlot.first->id == -1);
|
2024-08-17 19:19:48 +02:00
|
|
|
assert(altarArtifactsStorage->getArtPos(tradeSlot.second) != ArtifactPosition::PRE_FIRST);
|
2024-02-29 22:33:12 +02:00
|
|
|
tradeSlot.first->setID(tradeSlot.second->getTypeId().num);
|
2024-02-29 12:02:39 +02:00
|
|
|
tradeSlot.first->subtitle->setText(std::to_string(calcExpCost(tradeSlot.second->getTypeId())));
|
2024-01-16 16:22:40 +02:00
|
|
|
}
|
2024-02-29 12:02:39 +02:00
|
|
|
|
2024-08-17 19:19:48 +02:00
|
|
|
auto newArtsFromBulkMove = altarArtifactsStorage->artifactsInBackpack;
|
2024-02-29 12:02:39 +02:00
|
|
|
for(const auto & [altarSlot, art] : tradeSlotsMap)
|
2024-01-16 16:22:40 +02:00
|
|
|
{
|
2024-03-19 13:53:23 +02:00
|
|
|
newArtsFromBulkMove.erase(std::remove_if(newArtsFromBulkMove.begin(), newArtsFromBulkMove.end(), [artForRemove = art](auto & slotInfo)
|
2024-02-29 12:02:39 +02:00
|
|
|
{
|
2024-03-19 13:53:23 +02:00
|
|
|
return slotInfo.artifact == artForRemove;
|
2024-02-29 12:02:39 +02:00
|
|
|
}));
|
|
|
|
}
|
|
|
|
for(const auto & slotInfo : newArtsFromBulkMove)
|
|
|
|
{
|
2024-03-19 13:53:23 +02:00
|
|
|
for(const auto & altarSlot : offerTradePanel->slots)
|
2024-02-29 12:02:39 +02:00
|
|
|
if(altarSlot->id == -1)
|
|
|
|
{
|
2024-02-29 22:33:12 +02:00
|
|
|
altarSlot->setID(slotInfo.artifact->getTypeId().num);
|
2024-02-29 12:02:39 +02:00
|
|
|
altarSlot->subtitle->setText(std::to_string(calcExpCost(slotInfo.artifact->getTypeId())));
|
|
|
|
tradeSlotsMap.try_emplace(altarSlot, slotInfo.artifact);
|
|
|
|
break;
|
|
|
|
}
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
2024-02-29 12:02:39 +02:00
|
|
|
|
2024-01-16 16:22:40 +02:00
|
|
|
calcExpAltarForHero();
|
2024-04-28 14:58:54 +02:00
|
|
|
deal->block(tradeSlotsMap.empty() || !LOCPLINT->makingTurn);
|
2024-01-16 16:22:40 +02:00
|
|
|
}
|
2023-12-04 23:21:49 +02:00
|
|
|
|
2024-01-27 23:48:11 +02:00
|
|
|
void CAltarArtifacts::putBackArtifacts()
|
|
|
|
{
|
|
|
|
// TODO: If the backpack capacity limit is enabled, artifacts may remain on the altar.
|
|
|
|
// Perhaps should be erased in CGameHandler::objectVisitEnded if id of visited object will be available
|
2024-08-17 19:19:48 +02:00
|
|
|
if(!altarArtifactsStorage->artifactsInBackpack.empty())
|
2024-05-08 13:16:16 +02:00
|
|
|
LOCPLINT->cb->bulkMoveArtifacts(heroArts->altarId, heroArts->getHero()->id, false, true, true);
|
2024-01-27 23:48:11 +02:00
|
|
|
}
|
|
|
|
|
2024-03-20 13:28:19 +02:00
|
|
|
CMarketBase::MarketShowcasesParams CAltarArtifacts::getShowcasesParams() const
|
2024-02-29 12:02:39 +02:00
|
|
|
{
|
2024-03-29 12:29:16 +02:00
|
|
|
if(const auto art = hero->getArt(ArtifactPosition::TRANSITION_POS))
|
2024-04-08 11:46:46 +02:00
|
|
|
return MarketShowcasesParams
|
|
|
|
{
|
2024-02-29 12:02:39 +02:00
|
|
|
std::nullopt,
|
2024-03-29 12:29:16 +02:00
|
|
|
ShowcaseParams {std::to_string(offerQty), CGI->artifacts()->getByIndex(art->getTypeId())->getIconIndex()}
|
2024-04-08 11:46:46 +02:00
|
|
|
};
|
|
|
|
return MarketShowcasesParams {std::nullopt, std::nullopt};
|
2024-02-29 12:02:39 +02:00
|
|
|
}
|
|
|
|
|
2024-03-19 13:53:23 +02:00
|
|
|
void CAltarArtifacts::onSlotClickPressed(const std::shared_ptr<CTradeableItem> & altarSlot, std::shared_ptr<TradePanelBase> & curPanel)
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
2024-01-16 16:22:40 +02:00
|
|
|
assert(altarSlot);
|
|
|
|
|
|
|
|
if(const auto pickedArtInst = heroArts->getPickedArtifact())
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
2024-08-27 15:44:30 +02:00
|
|
|
if(pickedArtInst->canBePutAt(altarArtifactsStorage))
|
2024-01-27 23:48:11 +02:00
|
|
|
{
|
2024-01-16 16:22:40 +02:00
|
|
|
if(pickedArtInst->artType->isTradable())
|
|
|
|
{
|
|
|
|
if(altarSlot->id == -1)
|
2024-02-29 12:02:39 +02:00
|
|
|
tradeSlotsMap.try_emplace(altarSlot, pickedArtInst);
|
2024-04-28 14:58:54 +02:00
|
|
|
deal->block(!LOCPLINT->makingTurn);
|
2024-01-16 16:22:40 +02:00
|
|
|
|
|
|
|
LOCPLINT->cb->swapArtifacts(ArtifactLocation(heroArts->getHero()->id, ArtifactPosition::TRANSITION_POS),
|
2024-05-08 13:16:16 +02:00
|
|
|
ArtifactLocation(heroArts->altarId, ArtifactPosition::ALTAR));
|
2024-01-16 16:22:40 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
logGlobal->warn("Cannot put special artifact on altar!");
|
|
|
|
return;
|
|
|
|
}
|
2024-01-27 23:48:11 +02:00
|
|
|
}
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
2024-02-29 12:02:39 +02:00
|
|
|
else if(altarSlot->id != -1)
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
2024-02-29 12:02:39 +02:00
|
|
|
assert(tradeSlotsMap.at(altarSlot));
|
2024-08-17 19:19:48 +02:00
|
|
|
const auto slot = altarArtifactsStorage->getArtPos(tradeSlotsMap.at(altarSlot));
|
2023-12-04 23:21:49 +02:00
|
|
|
assert(slot != ArtifactPosition::PRE_FIRST);
|
2024-05-08 13:16:16 +02:00
|
|
|
LOCPLINT->cb->swapArtifacts(ArtifactLocation(heroArts->altarId, slot),
|
2024-04-30 11:39:20 +02:00
|
|
|
ArtifactLocation(hero->id, GH.isKeyboardCtrlDown() ? ArtifactPosition::FIRST_AVAILABLE : ArtifactPosition::TRANSITION_POS));
|
2024-02-29 12:02:39 +02:00
|
|
|
tradeSlotsMap.erase(altarSlot);
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
2024-01-16 16:22:40 +02:00
|
|
|
}
|
|
|
|
|
2024-03-19 13:53:23 +02:00
|
|
|
TExpType CAltarArtifacts::calcExpCost(ArtifactID id) const
|
2024-01-16 16:22:40 +02:00
|
|
|
{
|
2024-02-19 23:40:43 +02:00
|
|
|
int bidQty = 0;
|
2024-01-16 16:22:40 +02:00
|
|
|
int expOfArt = 0;
|
2024-02-29 12:02:39 +02:00
|
|
|
market->getOffer(id, 0, bidQty, expOfArt, EMarketMode::ARTIFACT_EXP);
|
2024-01-16 16:22:40 +02:00
|
|
|
return hero->calculateXp(expOfArt);
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|