2023-12-04 23:21:49 +02:00
|
|
|
/*
|
|
|
|
* CAltarCreatures.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 "CAltarCreatures.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"
|
|
|
|
|
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
|
|
|
|
|
|
|
CAltarCreatures::CAltarCreatures(const IMarket * market, const CGHeroInstance * hero)
|
2024-03-20 13:28:19 +02:00
|
|
|
: CMarketBase(market, hero)
|
2024-03-02 20:30:29 +02:00
|
|
|
, CMarketSlider(std::bind(&CAltarCreatures::onOfferSliderMoved, this, _1))
|
2024-03-20 13:28:19 +02:00
|
|
|
, CMarketTraderText(Point(28, 31), FONT_MEDIUM, Colors::YELLOW)
|
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-03-02 20:30:29 +02:00
|
|
|
deal = std::make_shared<CButton>(dealButtonPosWithSlider, AnimationPath::builtin("ALTSACR.DEF"),
|
2024-05-18 21:59:28 +02:00
|
|
|
CGI->generaltexth->zelp[584], [this]() {CAltarCreatures::makeDeal();}, EShortcut::MARKET_DEAL);
|
2023-12-04 23:21:49 +02:00
|
|
|
labels.emplace_back(std::make_shared<CLabel>(155, 30, FONT_SMALL, ETextAlignment::CENTER, Colors::YELLOW,
|
|
|
|
boost::str(boost::format(CGI->generaltexth->allTexts[272]) % hero->getNameTranslated())));
|
|
|
|
labels.emplace_back(std::make_shared<CLabel>(450, 30, FONT_SMALL, ETextAlignment::CENTER, Colors::YELLOW, CGI->generaltexth->allTexts[479]));
|
|
|
|
texts.emplace_back(std::make_unique<CTextBox>(CGI->generaltexth->allTexts[480], Rect(320, 56, 256, 40), 0, FONT_SMALL, ETextAlignment::CENTER, Colors::YELLOW));
|
2024-03-02 20:30:29 +02:00
|
|
|
offerSlider->moveTo(pos.topLeft() + Point(231, 481));
|
|
|
|
maxAmount->setHelp(CGI->generaltexth->zelp[578]);
|
2023-12-04 23:21:49 +02:00
|
|
|
|
|
|
|
unitsOnAltar.resize(GameConstants::ARMY_SIZE, 0);
|
|
|
|
expPerUnit.resize(GameConstants::ARMY_SIZE, 0);
|
|
|
|
sacrificeAllButton = std::make_shared<CButton>(
|
2024-05-18 21:59:28 +02:00
|
|
|
Point(393, 520), AnimationPath::builtin("ALTARMY.DEF"), CGI->generaltexth->zelp[579], std::bind(&CExperienceAltar::sacrificeAll, this), EShortcut::MARKET_SACRIFICE_ALL);
|
2023-12-04 23:21:49 +02:00
|
|
|
|
|
|
|
// Hero creatures panel
|
2024-02-25 22:58:53 +02:00
|
|
|
assert(bidTradePanel);
|
|
|
|
bidTradePanel->moveTo(pos.topLeft() + Point(45, 110));
|
2024-03-19 13:53:23 +02:00
|
|
|
bidTradePanel->showcaseSlot->moveTo(pos.topLeft() + Point(149, 422));
|
|
|
|
bidTradePanel->showcaseSlot->subtitle->moveBy(Point(0, 3));
|
2024-02-25 22:58:53 +02:00
|
|
|
for(const auto & slot : bidTradePanel->slots)
|
2024-03-19 13:53:23 +02:00
|
|
|
slot->clickPressedCallback = [this](const std::shared_ptr<CTradeableItem> & heroSlot) {CAltarCreatures::onSlotClickPressed(heroSlot, bidTradePanel);};
|
2023-12-04 23:21:49 +02:00
|
|
|
|
|
|
|
// Altar creatures panel
|
2024-02-25 22:58:53 +02:00
|
|
|
offerTradePanel = std::make_shared<CreaturesPanel>([this](const std::shared_ptr<CTradeableItem> & altarSlot)
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
2024-03-19 13:53:23 +02:00
|
|
|
CAltarCreatures::onSlotClickPressed(altarSlot, offerTradePanel);
|
2024-02-25 22:58:53 +02:00
|
|
|
}, bidTradePanel->slots);
|
|
|
|
offerTradePanel->moveTo(pos.topLeft() + Point(334, 110));
|
2024-03-19 13:53:23 +02:00
|
|
|
offerTradePanel->showcaseSlot->moveTo(pos.topLeft() + Point(395, 422));
|
|
|
|
offerTradePanel->showcaseSlot->subtitle->moveBy(Point(0, 3));
|
2024-02-25 22:58:53 +02:00
|
|
|
offerTradePanel->updateSlotsCallback = [this]()
|
2024-02-22 22:49:30 +02:00
|
|
|
{
|
2024-02-25 22:58:53 +02:00
|
|
|
for(const auto & altarSlot : offerTradePanel->slots)
|
2024-02-22 22:49:30 +02:00
|
|
|
updateAltarSlot(altarSlot);
|
|
|
|
};
|
2024-02-25 22:58:53 +02:00
|
|
|
bidTradePanel->deleteSlotsCheck = offerTradePanel->deleteSlotsCheck = std::bind(&CCreaturesSelling::slotDeletingCheck, this, _1);
|
2024-02-19 23:40:43 +02:00
|
|
|
|
2023-12-04 23:21:49 +02:00
|
|
|
readExpValues();
|
|
|
|
CAltarCreatures::deselect();
|
|
|
|
};
|
|
|
|
|
|
|
|
void CAltarCreatures::readExpValues()
|
|
|
|
{
|
2024-02-19 23:40:43 +02:00
|
|
|
int bidQty = 0;
|
2024-02-25 22:58:53 +02:00
|
|
|
for(const auto & heroSlot : bidTradePanel->slots)
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
|
|
|
if(heroSlot->id >= 0)
|
2024-02-19 23:40:43 +02:00
|
|
|
market->getOffer(heroSlot->id, 0, bidQty, expPerUnit[heroSlot->serial], EMarketMode::CREATURE_EXP);
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-02 20:30:29 +02:00
|
|
|
void CAltarCreatures::highlightingChanged()
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
|
|
|
int sliderAmount = 0;
|
2024-03-20 13:28:19 +02:00
|
|
|
if(bidTradePanel->isHighlighted())
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
|
|
|
std::optional<SlotID> lastSlot;
|
|
|
|
for(auto slot = SlotID(0); slot.num < GameConstants::ARMY_SIZE; slot++)
|
|
|
|
{
|
|
|
|
if(hero->getStackCount(slot) > unitsOnAltar[slot.num])
|
|
|
|
{
|
|
|
|
if(lastSlot.has_value())
|
|
|
|
{
|
|
|
|
lastSlot = std::nullopt;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
lastSlot = slot;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-03-19 13:53:23 +02:00
|
|
|
sliderAmount = hero->getStackCount(SlotID(bidTradePanel->highlightedSlot->serial));
|
|
|
|
if(lastSlot.has_value() && lastSlot.value() == SlotID(bidTradePanel->highlightedSlot->serial))
|
2023-12-04 23:21:49 +02:00
|
|
|
sliderAmount--;
|
|
|
|
}
|
|
|
|
offerSlider->setAmount(sliderAmount);
|
|
|
|
offerSlider->block(!offerSlider->getAmount());
|
2024-03-20 13:28:19 +02:00
|
|
|
if(bidTradePanel->isHighlighted())
|
2024-03-19 13:53:23 +02:00
|
|
|
offerSlider->scrollTo(unitsOnAltar[bidTradePanel->highlightedSlot->serial]);
|
2024-02-19 23:40:43 +02:00
|
|
|
maxAmount->block(offerSlider->getAmount() == 0);
|
2024-03-19 13:53:23 +02:00
|
|
|
updateShowcases();
|
2024-03-20 13:28:19 +02:00
|
|
|
CMarketTraderText::highlightingChanged();
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
|
|
|
|
2024-03-02 20:30:29 +02:00
|
|
|
void CAltarCreatures::update()
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
2024-03-02 20:30:29 +02:00
|
|
|
CMarketBase::update();
|
|
|
|
CExperienceAltar::update();
|
2024-02-25 22:58:53 +02:00
|
|
|
assert(bidTradePanel->slots.size() == offerTradePanel->slots.size());
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void CAltarCreatures::deselect()
|
|
|
|
{
|
2024-03-02 20:30:29 +02:00
|
|
|
CMarketBase::deselect();
|
|
|
|
CExperienceAltar::deselect();
|
|
|
|
CMarketSlider::deselect();
|
2024-03-20 13:28:19 +02:00
|
|
|
CMarketTraderText::deselect();
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
TExpType CAltarCreatures::calcExpAltarForHero()
|
|
|
|
{
|
|
|
|
TExpType expOnAltar(0);
|
|
|
|
auto oneUnitExp = expPerUnit.begin();
|
|
|
|
for(const auto units : unitsOnAltar)
|
|
|
|
expOnAltar += *oneUnitExp++ * units;
|
|
|
|
auto resultExp = hero->calculateXp(expOnAltar);
|
|
|
|
expForHero->setText(std::to_string(resultExp));
|
|
|
|
return resultExp;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CAltarCreatures::makeDeal()
|
|
|
|
{
|
|
|
|
std::vector<TradeItemSell> ids;
|
|
|
|
std::vector<ui32> toSacrifice;
|
|
|
|
|
|
|
|
for(int i = 0; i < unitsOnAltar.size(); i++)
|
|
|
|
{
|
|
|
|
if(unitsOnAltar[i])
|
|
|
|
{
|
|
|
|
ids.push_back(SlotID(i));
|
|
|
|
toSacrifice.push_back(unitsOnAltar[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-20 16:15:50 +02:00
|
|
|
LOCPLINT->cb->trade(market->getObjInstanceID(), EMarketMode::CREATURE_EXP, ids, {}, toSacrifice, hero);
|
2023-12-04 23:21:49 +02:00
|
|
|
|
|
|
|
for(int & units : unitsOnAltar)
|
|
|
|
units = 0;
|
|
|
|
|
2024-10-19 21:48:06 +02:00
|
|
|
for(auto & heroSlot : offerTradePanel->slots)
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
2024-10-19 21:48:06 +02:00
|
|
|
heroSlot->setID(CreatureID::NONE);
|
2024-02-27 13:39:50 +02:00
|
|
|
heroSlot->subtitle->clear();
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
2024-03-02 20:30:29 +02:00
|
|
|
deselect();
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
|
|
|
|
2024-03-20 13:28:19 +02:00
|
|
|
CMarketBase::MarketShowcasesParams CAltarCreatures::getShowcasesParams() const
|
2024-02-27 13:39:50 +02:00
|
|
|
{
|
2024-03-20 13:28:19 +02:00
|
|
|
std::optional<ShowcaseParams> bidSelected = std::nullopt;
|
|
|
|
std::optional<ShowcaseParams> offerSelected = std::nullopt;
|
|
|
|
if(bidTradePanel->isHighlighted())
|
2024-10-19 21:48:06 +02:00
|
|
|
bidSelected = ShowcaseParams {std::to_string(offerSlider->getValue()), CGI->creatures()->getByIndex(bidTradePanel->getHighlightedItemId())->getIconIndex()};
|
2024-03-20 13:28:19 +02:00
|
|
|
if(offerTradePanel->isHighlighted() && offerSlider->getValue() > 0)
|
2024-10-19 21:48:06 +02:00
|
|
|
offerSelected = ShowcaseParams {offerTradePanel->highlightedSlot->subtitle->getText(), CGI->creatures()->getByIndex(offerTradePanel->getHighlightedItemId())->getIconIndex()};
|
2024-04-08 11:46:46 +02:00
|
|
|
return MarketShowcasesParams {bidSelected, offerSelected};
|
2024-02-27 13:39:50 +02:00
|
|
|
}
|
|
|
|
|
2023-12-04 23:21:49 +02:00
|
|
|
void CAltarCreatures::sacrificeAll()
|
|
|
|
{
|
|
|
|
std::optional<SlotID> lastSlot;
|
2024-10-19 21:48:06 +02:00
|
|
|
for(const auto & heroSlot : bidTradePanel->slots)
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
|
|
|
auto stackCount = hero->getStackCount(SlotID(heroSlot->serial));
|
|
|
|
if(stackCount > unitsOnAltar[heroSlot->serial])
|
|
|
|
{
|
|
|
|
if(!lastSlot.has_value())
|
|
|
|
lastSlot = SlotID(heroSlot->serial);
|
|
|
|
unitsOnAltar[heroSlot->serial] = stackCount;
|
|
|
|
}
|
|
|
|
}
|
2024-02-03 22:59:05 +02:00
|
|
|
if(hero->needsLastStack())
|
|
|
|
{
|
|
|
|
assert(lastSlot.has_value());
|
|
|
|
unitsOnAltar[lastSlot.value().num]--;
|
|
|
|
}
|
2023-12-04 23:21:49 +02:00
|
|
|
|
2024-03-20 13:28:19 +02:00
|
|
|
if(offerTradePanel->isHighlighted())
|
2024-03-19 13:53:23 +02:00
|
|
|
offerSlider->scrollTo(unitsOnAltar[offerTradePanel->highlightedSlot->serial]);
|
2024-03-02 20:30:29 +02:00
|
|
|
offerTradePanel->update();
|
2024-03-19 13:53:23 +02:00
|
|
|
updateShowcases();
|
2023-12-04 23:21:49 +02:00
|
|
|
|
2024-04-28 14:58:54 +02:00
|
|
|
deal->block(calcExpAltarForHero() == 0 || !LOCPLINT->makingTurn);
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
|
|
|
|
2024-02-22 22:49:30 +02:00
|
|
|
void CAltarCreatures::updateAltarSlot(const std::shared_ptr<CTradeableItem> & slot)
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
|
|
|
auto units = unitsOnAltar[slot->serial];
|
2024-10-19 21:48:06 +02:00
|
|
|
const auto [oppositeSlot, oppositePanel] = getOpposite(slot);
|
|
|
|
slot->setID(units > 0 ? oppositeSlot->id : CreatureID::NONE);
|
2024-02-27 13:39:50 +02:00
|
|
|
slot->subtitle->setText(units > 0 ?
|
|
|
|
boost::str(boost::format(CGI->generaltexth->allTexts[122]) % std::to_string(hero->calculateXp(units * expPerUnit[slot->serial]))) : "");
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void CAltarCreatures::onOfferSliderMoved(int newVal)
|
|
|
|
{
|
2024-03-20 13:28:19 +02:00
|
|
|
if(bidTradePanel->isHighlighted())
|
2024-03-19 13:53:23 +02:00
|
|
|
unitsOnAltar[bidTradePanel->highlightedSlot->serial] = newVal;
|
2024-03-20 13:28:19 +02:00
|
|
|
if(offerTradePanel->isHighlighted())
|
2024-03-19 13:53:23 +02:00
|
|
|
updateAltarSlot(offerTradePanel->highlightedSlot);
|
2024-04-28 14:58:54 +02:00
|
|
|
deal->block(calcExpAltarForHero() == 0 || !LOCPLINT->makingTurn);
|
2024-03-02 20:30:29 +02:00
|
|
|
highlightingChanged();
|
2024-02-19 23:40:43 +02:00
|
|
|
redraw();
|
2023-12-04 23:21:49 +02:00
|
|
|
}
|
|
|
|
|
2024-03-19 13:53:23 +02:00
|
|
|
void CAltarCreatures::onSlotClickPressed(const std::shared_ptr<CTradeableItem> & newSlot, std::shared_ptr<TradePanelBase> & curPanel)
|
2023-12-04 23:21:49 +02:00
|
|
|
{
|
2024-03-19 13:53:23 +02:00
|
|
|
assert(newSlot);
|
|
|
|
assert(curPanel);
|
|
|
|
if(newSlot == curPanel->highlightedSlot)
|
2023-12-04 23:21:49 +02:00
|
|
|
return;
|
|
|
|
|
2024-03-19 13:53:23 +02:00
|
|
|
curPanel->onSlotClickPressed(newSlot);
|
2024-10-19 21:48:06 +02:00
|
|
|
auto [oppositeSlot, oppositePanel] = getOpposite(newSlot);
|
|
|
|
oppositePanel->onSlotClickPressed(oppositeSlot);
|
2024-03-02 20:30:29 +02:00
|
|
|
highlightingChanged();
|
2023-12-04 23:21:49 +02:00
|
|
|
redraw();
|
|
|
|
}
|
2024-03-20 13:28:19 +02:00
|
|
|
|
|
|
|
std::string CAltarCreatures::getTraderText()
|
|
|
|
{
|
|
|
|
if(bidTradePanel->isHighlighted() && offerTradePanel->isHighlighted())
|
|
|
|
{
|
2024-04-08 11:46:46 +02:00
|
|
|
MetaString message = MetaString::createFromTextID("core.genrltxt.484");
|
2024-10-19 21:48:06 +02:00
|
|
|
message.replaceNamePlural(CreatureID(bidTradePanel->getHighlightedItemId()));
|
2024-04-08 11:46:46 +02:00
|
|
|
return message.toString();
|
2024-03-20 13:28:19 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
}
|
2024-10-19 21:48:06 +02:00
|
|
|
|
|
|
|
std::tuple<const std::shared_ptr<CTradeableItem>, std::shared_ptr<TradePanelBase>> CAltarCreatures::getOpposite(
|
|
|
|
const std::shared_ptr<CTradeableItem> & curSlot)
|
|
|
|
{
|
|
|
|
assert(curSlot);
|
|
|
|
|
|
|
|
auto oppositePanel = bidTradePanel;
|
|
|
|
if(vstd::contains(bidTradePanel->slots, curSlot))
|
|
|
|
oppositePanel = offerTradePanel;
|
|
|
|
|
|
|
|
std::shared_ptr<CTradeableItem> oppositeSlot;
|
|
|
|
for(const auto & slot : oppositePanel->slots)
|
|
|
|
if (slot->serial == curSlot->serial)
|
|
|
|
{
|
|
|
|
oppositeSlot = slot;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return std::make_tuple(oppositeSlot, oppositePanel);
|
|
|
|
}
|