2023-06-06 17:32:53 +02:00
|
|
|
/*
|
|
|
|
* CBankInstanceConstructor.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 "CBankInstanceConstructor.h"
|
|
|
|
|
2024-02-11 23:09:01 +02:00
|
|
|
#include "../json/JsonRandom.h"
|
2023-06-06 17:32:53 +02:00
|
|
|
#include "../CGeneralTextHandler.h"
|
|
|
|
#include "../IGameCallback.h"
|
2024-01-31 00:17:40 +02:00
|
|
|
#include "../CRandomGenerator.h"
|
2023-06-06 17:32:53 +02:00
|
|
|
|
|
|
|
VCMI_LIB_NAMESPACE_BEGIN
|
|
|
|
|
|
|
|
bool CBankInstanceConstructor::hasNameTextID() const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CBankInstanceConstructor::initTypeData(const JsonNode & input)
|
|
|
|
{
|
|
|
|
if (input.Struct().count("name") == 0)
|
|
|
|
logMod->warn("Bank %s missing name!", getJsonKey());
|
|
|
|
|
2024-02-13 14:34:16 +02:00
|
|
|
VLC->generaltexth->registerString(input.getModScope(), getNameTextID(), input["name"].String());
|
2023-06-06 17:32:53 +02:00
|
|
|
|
|
|
|
levels = input["levels"].Vector();
|
|
|
|
bankResetDuration = static_cast<si32>(input["resetDuration"].Float());
|
2023-06-21 15:49:44 +02:00
|
|
|
blockVisit = input["blockedVisitable"].Bool();
|
|
|
|
coastVisitable = input["coastVisitable"].Bool();
|
2024-05-29 20:20:33 +02:00
|
|
|
regularUnitPlacement = input["regularUnitPlacement"].Bool();
|
2023-06-06 17:32:53 +02:00
|
|
|
}
|
|
|
|
|
2024-01-09 19:08:22 +02:00
|
|
|
BankConfig CBankInstanceConstructor::generateConfig(IGameCallback * cb, const JsonNode & level, CRandomGenerator & rng) const
|
2023-06-06 17:32:53 +02:00
|
|
|
{
|
|
|
|
BankConfig bc;
|
2024-01-09 19:08:22 +02:00
|
|
|
JsonRandom randomizer(cb);
|
2023-09-30 17:47:47 +02:00
|
|
|
JsonRandom::Variables emptyVariables;
|
2023-06-06 17:32:53 +02:00
|
|
|
|
|
|
|
bc.chance = static_cast<ui32>(level["chance"].Float());
|
2024-01-01 16:37:48 +02:00
|
|
|
bc.guards = randomizer.loadCreatures(level["guards"], rng, emptyVariables);
|
2023-06-06 17:32:53 +02:00
|
|
|
|
|
|
|
bc.resources = ResourceSet(level["reward"]["resources"]);
|
2024-01-01 16:37:48 +02:00
|
|
|
bc.creatures = randomizer.loadCreatures(level["reward"]["creatures"], rng, emptyVariables);
|
|
|
|
bc.artifacts = randomizer.loadArtifacts(level["reward"]["artifacts"], rng, emptyVariables);
|
|
|
|
bc.spells = randomizer.loadSpells(level["reward"]["spells"], rng, emptyVariables);
|
2023-06-06 17:32:53 +02:00
|
|
|
|
|
|
|
return bc;
|
|
|
|
}
|
|
|
|
|
2023-06-07 23:42:47 +02:00
|
|
|
void CBankInstanceConstructor::randomizeObject(CBank * bank, CRandomGenerator & rng) const
|
2023-06-06 17:32:53 +02:00
|
|
|
{
|
|
|
|
bank->resetDuration = bankResetDuration;
|
2023-06-21 15:49:44 +02:00
|
|
|
bank->blockVisit = blockVisit;
|
|
|
|
bank->coastVisitable = coastVisitable;
|
2024-05-29 20:20:33 +02:00
|
|
|
bank->regularUnitPlacement = regularUnitPlacement;
|
2023-06-06 17:32:53 +02:00
|
|
|
|
|
|
|
si32 totalChance = 0;
|
|
|
|
for(const auto & node : levels)
|
|
|
|
totalChance += static_cast<si32>(node["chance"].Float());
|
|
|
|
|
|
|
|
assert(totalChance != 0);
|
|
|
|
|
|
|
|
si32 selectedChance = rng.nextInt(totalChance - 1);
|
|
|
|
|
|
|
|
int cumulativeChance = 0;
|
|
|
|
for(const auto & node : levels)
|
|
|
|
{
|
|
|
|
cumulativeChance += static_cast<int>(node["chance"].Float());
|
|
|
|
if(selectedChance < cumulativeChance)
|
|
|
|
{
|
2024-01-09 19:08:22 +02:00
|
|
|
bank->setConfig(generateConfig(bank->cb, node, rng));
|
2023-06-06 17:32:53 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CBankInfo::CBankInfo(const JsonVector & Config) :
|
|
|
|
config(Config)
|
|
|
|
{
|
|
|
|
assert(!Config.empty());
|
|
|
|
}
|
|
|
|
|
2024-01-01 16:37:48 +02:00
|
|
|
TPossibleGuards CBankInfo::getPossibleGuards(IGameCallback * cb) const
|
2023-06-06 17:32:53 +02:00
|
|
|
{
|
2023-09-30 17:47:47 +02:00
|
|
|
JsonRandom::Variables emptyVariables;
|
2024-01-01 16:37:48 +02:00
|
|
|
JsonRandom randomizer(cb);
|
2023-06-06 17:32:53 +02:00
|
|
|
TPossibleGuards out;
|
|
|
|
|
|
|
|
for(const JsonNode & configEntry : config)
|
|
|
|
{
|
|
|
|
const JsonNode & guardsInfo = configEntry["guards"];
|
2024-01-01 16:37:48 +02:00
|
|
|
auto stacks = randomizer.evaluateCreatures(guardsInfo, emptyVariables);
|
2023-06-06 17:32:53 +02:00
|
|
|
IObjectInfo::CArmyStructure army;
|
|
|
|
|
|
|
|
|
|
|
|
for(auto stack : stacks)
|
|
|
|
{
|
|
|
|
army.totalStrength += stack.allowedCreatures.front()->getAIValue() * (stack.minAmount + stack.maxAmount) / 2;
|
|
|
|
//TODO: add fields for flyers, walkers etc...
|
|
|
|
}
|
|
|
|
|
|
|
|
ui8 chance = static_cast<ui8>(configEntry["chance"].Float());
|
|
|
|
out.push_back(std::make_pair(chance, army));
|
|
|
|
}
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<PossibleReward<TResources>> CBankInfo::getPossibleResourcesReward() const
|
|
|
|
{
|
|
|
|
std::vector<PossibleReward<TResources>> result;
|
|
|
|
|
|
|
|
for(const JsonNode & configEntry : config)
|
|
|
|
{
|
|
|
|
const JsonNode & resourcesInfo = configEntry["reward"]["resources"];
|
|
|
|
|
|
|
|
if(!resourcesInfo.isNull())
|
|
|
|
{
|
|
|
|
result.emplace_back(configEntry["chance"].Integer(), TResources(resourcesInfo));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2024-01-09 19:08:22 +02:00
|
|
|
std::vector<PossibleReward<CStackBasicDescriptor>> CBankInfo::getPossibleCreaturesReward(IGameCallback * cb) const
|
2023-06-06 17:32:53 +02:00
|
|
|
{
|
2023-09-30 17:47:47 +02:00
|
|
|
JsonRandom::Variables emptyVariables;
|
2024-01-09 19:08:22 +02:00
|
|
|
JsonRandom randomizer(cb);
|
2024-06-24 03:23:26 +02:00
|
|
|
std::vector<PossibleReward<CStackBasicDescriptor>> approximateReward;
|
2023-06-06 17:32:53 +02:00
|
|
|
|
|
|
|
for(const JsonNode & configEntry : config)
|
|
|
|
{
|
|
|
|
const JsonNode & guardsInfo = configEntry["reward"]["creatures"];
|
2024-01-01 16:37:48 +02:00
|
|
|
auto stacks = randomizer.evaluateCreatures(guardsInfo, emptyVariables);
|
2023-06-06 17:32:53 +02:00
|
|
|
|
|
|
|
for(auto stack : stacks)
|
|
|
|
{
|
|
|
|
const auto * creature = stack.allowedCreatures.front();
|
|
|
|
|
2024-06-24 03:23:26 +02:00
|
|
|
approximateReward.emplace_back(configEntry["chance"].Integer(), CStackBasicDescriptor(creature, (stack.minAmount + stack.maxAmount) / 2));
|
2023-06-06 17:32:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-24 03:23:26 +02:00
|
|
|
return approximateReward;
|
2023-06-06 17:32:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CBankInfo::givesResources() const
|
|
|
|
{
|
|
|
|
for(const JsonNode & node : config)
|
|
|
|
if(!node["reward"]["resources"].isNull())
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CBankInfo::givesArtifacts() const
|
|
|
|
{
|
|
|
|
for(const JsonNode & node : config)
|
|
|
|
if(!node["reward"]["artifacts"].isNull())
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CBankInfo::givesCreatures() const
|
|
|
|
{
|
|
|
|
for(const JsonNode & node : config)
|
|
|
|
if(!node["reward"]["creatures"].isNull())
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CBankInfo::givesSpells() const
|
|
|
|
{
|
|
|
|
for(const JsonNode & node : config)
|
|
|
|
if(!node["reward"]["spells"].isNull())
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::unique_ptr<IObjectInfo> CBankInstanceConstructor::getObjectInfo(std::shared_ptr<const ObjectTemplate> tmpl) const
|
|
|
|
{
|
|
|
|
return std::unique_ptr<IObjectInfo>(new CBankInfo(levels));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
VCMI_LIB_NAMESPACE_END
|