/* * 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" #include "../../widgets/Buttons.h" #include "../../widgets/TextControls.h" #include "../../CGameInfo.h" #include "../../CPlayerInterface.h" #include "../../../CCallback.h" #include "../../../lib/networkPacks/ArtifactLocation.h" #include "../../../lib/CGeneralTextHandler.h" #include "../../../lib/mapObjects/CGHeroInstance.h" #include "../../../lib/mapObjects/CGMarket.h" CAltarArtifacts::CAltarArtifacts(const IMarket * market, const CGHeroInstance * hero) : CTradeBase(market, hero, [this](){return CAltarArtifacts::getSelectionParams();}) { OBJECT_CONSTRUCTION_CAPTURING(255 - DISPOSE); assert(market); auto altarObj = dynamic_cast(market); altarId = altarObj->id; altarArtifacts = altarObj; deal = std::make_shared(dealButtonPos, AnimationPath::builtin("ALTSACR.DEF"), CGI->generaltexth->zelp[585], [this]() {CAltarArtifacts::makeDeal(); }); labels.emplace_back(std::make_shared(450, 32, FONT_SMALL, ETextAlignment::CENTER, Colors::YELLOW, CGI->generaltexth->allTexts[477])); labels.emplace_back(std::make_shared(302, 424, FONT_SMALL, ETextAlignment::CENTER, Colors::YELLOW, CGI->generaltexth->allTexts[478])); sacrificeAllButton = std::make_shared(Point(393, 520), AnimationPath::builtin("ALTFILL.DEF"), CGI->generaltexth->zelp[571], std::bind(&CExperienceAltar::sacrificeAll, this)); sacrificeAllButton->block(hero->artifactsInBackpack.empty() && hero->artifactsWorn.empty()); sacrificeBackpackButton = std::make_shared(Point(147, 520), AnimationPath::builtin("ALTEMBK.DEF"), CGI->generaltexth->zelp[570], std::bind(&CAltarArtifacts::sacrificeBackpack, this)); sacrificeBackpackButton->block(hero->artifactsInBackpack.empty()); // Hero's artifacts heroArts = std::make_shared(Point(-365, -11)); heroArts->setHero(hero); // Altar offerTradePanel = std::make_shared([this](const std::shared_ptr & altarSlot) { CAltarArtifacts::onSlotClickPressed(altarSlot, hRight); }); offerTradePanel->updateSlotsCallback = std::bind(&CAltarArtifacts::updateAltarSlots, this); offerTradePanel->moveTo(pos.topLeft() + Point(315, 52)); CTradeBase::updateSelected(); CAltarArtifacts::deselect(); }; TExpType CAltarArtifacts::calcExpAltarForHero() { TExpType expOnAltar(0); for(const auto & tradeSlot : tradeSlotsMap) expOnAltar += calcExpCost(tradeSlot.second->getTypeId()); expForHero->setText(std::to_string(expOnAltar)); return expOnAltar; } void CAltarArtifacts::deselect() { CTradeBase::deselect(); expForHero->setText(std::to_string(0)); tradeSlotsMap.clear(); // The event for removing artifacts from the altar will not be triggered. Therefore, we clean the altar immediately. for(auto & slot : offerTradePanel->slots) { slot->setID(-1); slot->subtitle->clear(); } } void CAltarArtifacts::makeDeal() { std::vector positions; for(const auto & [altarSlot, artInst] : tradeSlotsMap) { positions.push_back(artInst->getId()); } LOCPLINT->cb->trade(market, EMarketMode::ARTIFACT_EXP, positions, std::vector(), std::vector(), hero); deselect(); } void CAltarArtifacts::sacrificeAll() { LOCPLINT->cb->bulkMoveArtifacts(heroArts->getHero()->id, altarId, false, true, true); } void CAltarArtifacts::sacrificeBackpack() { LOCPLINT->cb->bulkMoveArtifacts(heroArts->getHero()->id, altarId, false, false, true); } void CAltarArtifacts::setSelectedArtifact(std::optional id) { if(id.has_value()) { hRight = offerTradePanel->selectedSlot; hRight->setID(id.value().num); offerQty = calcExpCost(id.value()); } else { hRight.reset(); offerQty = 0; } updateSelected(); } std::shared_ptr CAltarArtifacts::getAOHset() const { return heroArts; } ObjectInstanceID CAltarArtifacts::getObjId() const { return altarId; } void CAltarArtifacts::updateAltarSlots() { assert(altarArtifacts->artifactsInBackpack.size() <= GameConstants::ALTAR_ARTIFACTS_SLOTS); assert(tradeSlotsMap.size() <= GameConstants::ALTAR_ARTIFACTS_SLOTS); auto tradeSlotsMapNewArts = tradeSlotsMap; for(auto & altarSlot : offerTradePanel->slots) if(altarSlot->id != -1) { if(tradeSlotsMap.find(altarSlot) == tradeSlotsMap.end()) { altarSlot->setID(-1); altarSlot->subtitle->clear(); } else { tradeSlotsMapNewArts.erase(altarSlot); } } for(auto & tradeSlot : tradeSlotsMapNewArts) { assert(tradeSlot.first->id == -1); assert(altarArtifacts->getSlotByInstance(tradeSlot.second) != ArtifactPosition::PRE_FIRST); tradeSlot.first->setID(tradeSlot.second->getTypeId()); tradeSlot.first->subtitle->setText(std::to_string(calcExpCost(tradeSlot.second->getTypeId()))); } auto newArtsFromBulkMove = altarArtifacts->artifactsInBackpack; for(const auto & [altarSlot, art] : tradeSlotsMap) { newArtsFromBulkMove.erase(std::remove_if(newArtsFromBulkMove.begin(), newArtsFromBulkMove.end(), [art = art](auto & slotInfo) { return slotInfo.artifact == art; })); } for(const auto & slotInfo : newArtsFromBulkMove) { for(auto & altarSlot : offerTradePanel->slots) if(altarSlot->id == -1) { altarSlot->setID(slotInfo.artifact->getTypeId()); altarSlot->subtitle->setText(std::to_string(calcExpCost(slotInfo.artifact->getTypeId()))); tradeSlotsMap.try_emplace(altarSlot, slotInfo.artifact); break; } } calcExpAltarForHero(); deal->block(tradeSlotsMap.empty()); } 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 if(!altarArtifacts->artifactsInBackpack.empty()) LOCPLINT->cb->bulkMoveArtifacts(altarId, heroArts->getHero()->id, false, true, true); } CTradeBase::SelectionParams CAltarArtifacts::getSelectionParams() const { if(hRight) return std::make_tuple( std::nullopt, SelectionParamOneSide {std::to_string(offerQty), GameResID(hRight->id)} ); return std::make_tuple(std::nullopt, std::nullopt); } void CAltarArtifacts::onSlotClickPressed(const std::shared_ptr & altarSlot, std::shared_ptr & hCurSlot) { assert(altarSlot); if(const auto pickedArtInst = heroArts->getPickedArtifact()) { if(pickedArtInst->canBePutAt(altarArtifacts)) { if(pickedArtInst->artType->isTradable()) { if(altarSlot->id == -1) tradeSlotsMap.try_emplace(altarSlot, pickedArtInst); deal->block(false); LOCPLINT->cb->swapArtifacts(ArtifactLocation(heroArts->getHero()->id, ArtifactPosition::TRANSITION_POS), ArtifactLocation(altarId, ArtifactPosition::ALTAR)); } else { logGlobal->warn("Cannot put special artifact on altar!"); return; } } } else if(altarSlot->id != -1) { assert(tradeSlotsMap.at(altarSlot)); const auto slot = altarArtifacts->getSlotByInstance(tradeSlotsMap.at(altarSlot)); assert(slot != ArtifactPosition::PRE_FIRST); LOCPLINT->cb->swapArtifacts(ArtifactLocation(altarId, slot), ArtifactLocation(hero->id, ArtifactPosition::TRANSITION_POS)); tradeSlotsMap.erase(altarSlot); } } TExpType CAltarArtifacts::calcExpCost(ArtifactID id) { int bidQty = 0; int expOfArt = 0; market->getOffer(id, 0, bidQty, expOfArt, EMarketMode::ARTIFACT_EXP); return hero->calculateXp(expOfArt); }