2017-07-13 10:26:03 +02:00
|
|
|
/*
|
2021-05-16 19:53:11 +02:00
|
|
|
* CommonConstructors.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
|
|
|
|
*
|
|
|
|
*/
|
2014-06-14 18:42:13 +03:00
|
|
|
#include "StdInc.h"
|
|
|
|
#include "CommonConstructors.h"
|
|
|
|
|
|
|
|
#include "CGTownInstance.h"
|
|
|
|
#include "CGHeroInstance.h"
|
2014-06-22 13:39:40 +03:00
|
|
|
#include "CBank.h"
|
2023-01-09 01:17:37 +02:00
|
|
|
#include "../TerrainHandler.h"
|
2014-06-14 18:42:13 +03:00
|
|
|
#include "../mapping/CMap.h"
|
|
|
|
#include "../CHeroHandler.h"
|
2023-01-11 00:49:31 +02:00
|
|
|
#include "../CGeneralTextHandler.h"
|
2014-06-14 18:42:13 +03:00
|
|
|
#include "../CCreatureHandler.h"
|
2014-06-22 13:39:40 +03:00
|
|
|
#include "JsonRandom.h"
|
2014-06-25 17:11:07 +03:00
|
|
|
#include "../CModHandler.h"
|
|
|
|
#include "../IGameCallback.h"
|
2023-04-18 22:14:15 +02:00
|
|
|
#include "../StringConstants.h"
|
2014-06-14 18:42:13 +03:00
|
|
|
|
2022-07-26 15:07:42 +02:00
|
|
|
VCMI_LIB_NAMESPACE_BEGIN
|
|
|
|
|
2014-06-14 18:42:13 +03:00
|
|
|
bool CObstacleConstructor::isStaticObject()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CTownInstanceConstructor::initTypeData(const JsonNode & input)
|
|
|
|
{
|
2014-06-15 19:43:01 +03:00
|
|
|
VLC->modh->identifiers.requestIdentifier("faction", input["faction"], [&](si32 index)
|
|
|
|
{
|
Entities redesign and a few ERM features
* Made most Handlers derived from CHandlerBase and moved service API there.
* Declared existing Entity APIs.
* Added basic script context caching
* Started Lua script module
* Started Lua spell effect API
* Started script state persistence
* Started battle info callback binding
* CommitPackage removed
* Extracted spells::Caster to own header; Expanded Spell API.
* implemented !!MC:S, !!FU:E, !!FU:P, !!MA, !!VR:H, !!VR:C
* !!BU:C, !!BU:E, !!BU:G, !!BU:M implemented
* Allow use of "MC:S@varName@" to declare normal variable (technically v-variable with string key)
* Re-enabled VERM macros.
* !?GM0 added
* !?TM implemented
* Added !!MF:N
* Started !?OB, !!BM, !!HE, !!OW, !!UN
* Added basic support of w-variables
* Added support for ERM indirect variables
* Made !?FU regular trigger
* !!re (ERA loop receiver) implemented
* Fixed ERM receivers with zero args.
2018-03-17 16:58:30 +02:00
|
|
|
faction = (*VLC->townh)[index];
|
2014-06-15 19:43:01 +03:00
|
|
|
});
|
2014-06-14 18:42:13 +03:00
|
|
|
|
|
|
|
filtersJson = input["filters"];
|
2022-11-30 17:36:55 +02:00
|
|
|
|
|
|
|
// change scope of "filters" to scope of object that is being loaded
|
|
|
|
// since this filters require to resolve building ID's
|
2022-11-29 22:34:32 +02:00
|
|
|
filtersJson.setMeta(input["faction"].meta);
|
2014-06-14 18:42:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void CTownInstanceConstructor::afterLoadFinalization()
|
|
|
|
{
|
2014-06-15 19:43:01 +03:00
|
|
|
assert(faction);
|
2023-02-12 22:39:17 +02:00
|
|
|
for(const auto & entry : filtersJson.Struct())
|
2014-06-14 18:42:13 +03:00
|
|
|
{
|
2014-06-15 19:43:01 +03:00
|
|
|
filters[entry.first] = LogicalExpression<BuildingID>(entry.second, [this](const JsonNode & node)
|
2014-06-14 18:42:13 +03:00
|
|
|
{
|
2023-04-16 19:42:56 +02:00
|
|
|
return BuildingID(VLC->modh->identifiers.getIdentifier("building." + faction->getJsonKey(), node.Vector()[0]).value());
|
2014-06-14 18:42:13 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-11 15:12:35 +02:00
|
|
|
bool CTownInstanceConstructor::objectFilter(const CGObjectInstance * object, std::shared_ptr<const ObjectTemplate> templ) const
|
2014-06-14 18:42:13 +03:00
|
|
|
{
|
2023-02-12 22:39:17 +02:00
|
|
|
const auto * town = dynamic_cast<const CGTownInstance *>(object);
|
2014-06-14 18:42:13 +03:00
|
|
|
|
|
|
|
auto buildTest = [&](const BuildingID & id)
|
|
|
|
{
|
|
|
|
return town->hasBuilt(id);
|
|
|
|
};
|
|
|
|
|
2022-09-11 15:12:35 +02:00
|
|
|
return filters.count(templ->stringID) != 0 && filters.at(templ->stringID).test(buildTest);
|
2014-06-14 18:42:13 +03:00
|
|
|
}
|
|
|
|
|
2022-09-11 15:12:35 +02:00
|
|
|
CGObjectInstance * CTownInstanceConstructor::create(std::shared_ptr<const ObjectTemplate> tmpl) const
|
2014-06-14 18:42:13 +03:00
|
|
|
{
|
|
|
|
CGTownInstance * obj = createTyped(tmpl);
|
|
|
|
obj->town = faction->town;
|
|
|
|
obj->tempOwner = PlayerColor::NEUTRAL;
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CTownInstanceConstructor::configureObject(CGObjectInstance * object, CRandomGenerator & rng) const
|
|
|
|
{
|
2023-02-12 22:39:17 +02:00
|
|
|
auto templ = getOverride(CGObjectInstance::cb->getTile(object->pos)->terType->getId(), object);
|
2021-05-16 19:53:11 +02:00
|
|
|
if(templ)
|
2022-09-11 15:12:35 +02:00
|
|
|
object->appearance = templ;
|
2014-06-14 18:42:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void CHeroInstanceConstructor::initTypeData(const JsonNode & input)
|
|
|
|
{
|
2021-05-16 19:53:11 +02:00
|
|
|
VLC->modh->identifiers.requestIdentifier(
|
|
|
|
"heroClass",
|
|
|
|
input["heroClass"],
|
|
|
|
[&](si32 index) { heroClass = VLC->heroh->classes[index]; });
|
2014-06-16 19:27:26 +03:00
|
|
|
|
|
|
|
filtersJson = input["filters"];
|
|
|
|
}
|
|
|
|
|
|
|
|
void CHeroInstanceConstructor::afterLoadFinalization()
|
|
|
|
{
|
2023-02-12 22:39:17 +02:00
|
|
|
for(const auto & entry : filtersJson.Struct())
|
2014-06-16 19:27:26 +03:00
|
|
|
{
|
2019-01-19 12:52:02 +02:00
|
|
|
filters[entry.first] = LogicalExpression<HeroTypeID>(entry.second, [](const JsonNode & node)
|
2014-06-16 19:27:26 +03:00
|
|
|
{
|
2023-04-16 19:42:56 +02:00
|
|
|
return HeroTypeID(VLC->modh->identifiers.getIdentifier("hero", node.Vector()[0]).value());
|
2014-06-16 19:27:26 +03:00
|
|
|
});
|
|
|
|
}
|
2014-06-14 18:42:13 +03:00
|
|
|
}
|
|
|
|
|
2022-09-11 15:12:35 +02:00
|
|
|
bool CHeroInstanceConstructor::objectFilter(const CGObjectInstance * object, std::shared_ptr<const ObjectTemplate> templ) const
|
2014-06-14 18:42:13 +03:00
|
|
|
{
|
2023-02-12 22:39:17 +02:00
|
|
|
const auto * hero = dynamic_cast<const CGHeroInstance *>(object);
|
2014-06-16 19:27:26 +03:00
|
|
|
|
|
|
|
auto heroTest = [&](const HeroTypeID & id)
|
|
|
|
{
|
2023-01-02 13:27:03 +02:00
|
|
|
return hero->type->getId() == id;
|
2014-06-16 19:27:26 +03:00
|
|
|
};
|
|
|
|
|
2022-09-11 15:12:35 +02:00
|
|
|
if(filters.count(templ->stringID))
|
2014-06-16 19:27:26 +03:00
|
|
|
{
|
2022-09-11 15:12:35 +02:00
|
|
|
return filters.at(templ->stringID).test(heroTest);
|
2014-06-16 19:27:26 +03:00
|
|
|
}
|
2014-06-14 18:42:13 +03:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-09-11 15:12:35 +02:00
|
|
|
CGObjectInstance * CHeroInstanceConstructor::create(std::shared_ptr<const ObjectTemplate> tmpl) const
|
2014-06-14 18:42:13 +03:00
|
|
|
{
|
|
|
|
CGHeroInstance * obj = createTyped(tmpl);
|
|
|
|
obj->type = nullptr; //FIXME: set to valid value. somehow.
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CHeroInstanceConstructor::configureObject(CGObjectInstance * object, CRandomGenerator & rng) const
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2023-01-11 00:49:31 +02:00
|
|
|
bool CDwellingInstanceConstructor::hasNameTextID() const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-06-14 18:42:13 +03:00
|
|
|
void CDwellingInstanceConstructor::initTypeData(const JsonNode & input)
|
|
|
|
{
|
2023-01-11 00:49:31 +02:00
|
|
|
if (input.Struct().count("name") == 0)
|
|
|
|
logMod->warn("Dwelling %s missing name!", getJsonKey());
|
|
|
|
|
2023-02-09 15:03:49 +02:00
|
|
|
VLC->generaltexth->registerString( input.meta, getNameTextID(), input["name"].String());
|
2023-01-11 00:49:31 +02:00
|
|
|
|
2014-06-14 18:42:13 +03:00
|
|
|
const JsonVector & levels = input["creatures"].Vector();
|
2021-04-25 14:07:06 +02:00
|
|
|
const auto totalLevels = levels.size();
|
|
|
|
|
|
|
|
availableCreatures.resize(totalLevels);
|
|
|
|
for(auto currentLevel = 0; currentLevel < totalLevels; currentLevel++)
|
2014-06-14 18:42:13 +03:00
|
|
|
{
|
2021-04-25 14:07:06 +02:00
|
|
|
const JsonVector & creaturesOnLevel = levels[currentLevel].Vector();
|
|
|
|
const auto creaturesNumber = creaturesOnLevel.size();
|
|
|
|
availableCreatures[currentLevel].resize(creaturesNumber);
|
|
|
|
|
|
|
|
for(auto currentCreature = 0; currentCreature < creaturesNumber; currentCreature++)
|
2014-06-14 18:42:13 +03:00
|
|
|
{
|
2021-04-25 14:07:06 +02:00
|
|
|
VLC->modh->identifiers.requestIdentifier("creature", creaturesOnLevel[currentCreature], [=] (si32 index)
|
2014-06-14 18:42:13 +03:00
|
|
|
{
|
2021-07-15 23:32:13 +02:00
|
|
|
availableCreatures[currentLevel][currentCreature] = VLC->creh->objects[index];
|
2014-06-14 18:42:13 +03:00
|
|
|
});
|
|
|
|
}
|
2021-04-25 14:07:06 +02:00
|
|
|
assert(!availableCreatures[currentLevel].empty());
|
2014-06-14 18:42:13 +03:00
|
|
|
}
|
2014-06-15 19:43:01 +03:00
|
|
|
guards = input["guards"];
|
2014-06-14 18:42:13 +03:00
|
|
|
}
|
|
|
|
|
2023-02-12 22:39:17 +02:00
|
|
|
bool CDwellingInstanceConstructor::objectFilter(const CGObjectInstance * obj, std::shared_ptr<const ObjectTemplate> tmpl) const
|
2014-06-14 18:42:13 +03:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-09-11 15:12:35 +02:00
|
|
|
CGObjectInstance * CDwellingInstanceConstructor::create(std::shared_ptr<const ObjectTemplate> tmpl) const
|
2014-06-14 18:42:13 +03:00
|
|
|
{
|
|
|
|
CGDwelling * obj = createTyped(tmpl);
|
|
|
|
|
2014-06-15 19:43:01 +03:00
|
|
|
obj->creatures.resize(availableCreatures.size());
|
2023-02-12 22:39:17 +02:00
|
|
|
for(const auto & entry : availableCreatures)
|
2014-06-15 19:43:01 +03:00
|
|
|
{
|
2021-05-16 19:53:11 +02:00
|
|
|
for(const CCreature * cre : entry)
|
2023-04-05 02:26:29 +02:00
|
|
|
obj->creatures.back().second.push_back(cre->getId());
|
2014-06-14 18:42:13 +03:00
|
|
|
}
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
2014-06-15 19:43:01 +03:00
|
|
|
void CDwellingInstanceConstructor::configureObject(CGObjectInstance * object, CRandomGenerator &rng) const
|
2014-06-14 18:42:13 +03:00
|
|
|
{
|
2023-02-12 22:39:17 +02:00
|
|
|
auto * dwelling = dynamic_cast<CGDwelling *>(object);
|
2014-06-15 19:43:01 +03:00
|
|
|
|
|
|
|
dwelling->creatures.clear();
|
2014-08-19 13:16:25 +03:00
|
|
|
dwelling->creatures.reserve(availableCreatures.size());
|
2014-06-14 18:42:13 +03:00
|
|
|
|
2023-02-12 22:39:17 +02:00
|
|
|
for(const auto & entry : availableCreatures)
|
2014-06-15 19:43:01 +03:00
|
|
|
{
|
2014-08-19 13:16:25 +03:00
|
|
|
dwelling->creatures.resize(dwelling->creatures.size() + 1);
|
2021-05-16 19:53:11 +02:00
|
|
|
for(const CCreature * cre : entry)
|
2023-04-05 02:26:29 +02:00
|
|
|
dwelling->creatures.back().second.push_back(cre->getId());
|
2014-06-15 19:43:01 +03:00
|
|
|
}
|
|
|
|
|
2015-06-05 11:28:14 +02:00
|
|
|
bool guarded = false; //TODO: serialize for sanity
|
|
|
|
|
2021-05-16 19:53:11 +02:00
|
|
|
if(guards.getType() == JsonNode::JsonType::DATA_BOOL) //simple switch
|
2014-06-15 23:25:10 +03:00
|
|
|
{
|
2021-05-16 19:53:11 +02:00
|
|
|
if(guards.Bool())
|
2014-06-30 00:16:45 +03:00
|
|
|
{
|
2015-06-05 11:28:14 +02:00
|
|
|
guarded = true;
|
2014-06-30 00:16:45 +03:00
|
|
|
}
|
2014-06-15 23:25:10 +03:00
|
|
|
}
|
2021-05-16 19:53:11 +02:00
|
|
|
else if(guards.getType() == JsonNode::JsonType::DATA_VECTOR) //custom guards (eg. Elemental Conflux)
|
2014-06-15 19:43:01 +03:00
|
|
|
{
|
2021-05-16 19:53:11 +02:00
|
|
|
for(auto & stack : JsonRandom::loadCreatures(guards, rng))
|
2015-06-05 11:28:14 +02:00
|
|
|
{
|
2023-04-05 02:26:29 +02:00
|
|
|
dwelling->putStack(SlotID(dwelling->stacksCount()), new CStackInstance(stack.type->getId(), stack.count));
|
2015-06-05 11:28:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else //default condition - creatures are of level 5 or higher
|
|
|
|
{
|
2021-05-16 19:53:11 +02:00
|
|
|
for(auto creatureEntry : availableCreatures)
|
2015-06-05 11:28:14 +02:00
|
|
|
{
|
2023-04-05 02:26:29 +02:00
|
|
|
if(creatureEntry.at(0)->getLevel() >= 5)
|
2015-06-05 11:28:14 +02:00
|
|
|
{
|
|
|
|
guarded = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-16 19:53:11 +02:00
|
|
|
if(guarded)
|
2015-06-05 11:28:14 +02:00
|
|
|
{
|
2021-05-16 19:53:11 +02:00
|
|
|
for(auto creatureEntry : availableCreatures)
|
2015-06-05 11:28:14 +02:00
|
|
|
{
|
|
|
|
const CCreature * crea = creatureEntry.at(0);
|
2023-04-05 02:26:29 +02:00
|
|
|
dwelling->putStack(SlotID(dwelling->stacksCount()), new CStackInstance(crea->getId(), crea->getGrowth() * 3));
|
2015-06-05 11:28:14 +02:00
|
|
|
}
|
2014-06-15 19:43:01 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDwellingInstanceConstructor::producesCreature(const CCreature * crea) const
|
|
|
|
{
|
2023-02-12 22:39:17 +02:00
|
|
|
for(const auto & entry : availableCreatures)
|
2014-06-15 19:43:01 +03:00
|
|
|
{
|
2021-05-16 19:53:11 +02:00
|
|
|
for(const CCreature * cre : entry)
|
|
|
|
if(crea == cre)
|
2014-06-15 19:43:01 +03:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2014-06-14 18:42:13 +03:00
|
|
|
}
|
2014-06-22 13:39:40 +03:00
|
|
|
|
2014-07-06 23:14:37 +03:00
|
|
|
std::vector<const CCreature *> CDwellingInstanceConstructor::getProducedCreatures() const
|
|
|
|
{
|
|
|
|
std::vector<const CCreature *> creatures; //no idea why it's 2D, to be honest
|
2023-02-12 22:39:17 +02:00
|
|
|
for(const auto & entry : availableCreatures)
|
2014-07-06 23:14:37 +03:00
|
|
|
{
|
2021-05-16 19:53:11 +02:00
|
|
|
for(const CCreature * cre : entry)
|
2014-07-06 23:14:37 +03:00
|
|
|
creatures.push_back(cre);
|
|
|
|
}
|
|
|
|
return creatures;
|
|
|
|
}
|
|
|
|
|
2023-04-18 15:27:39 +02:00
|
|
|
void BoatInstanceConstructor::initTypeData(const JsonNode & input)
|
|
|
|
{
|
2023-04-18 22:14:15 +02:00
|
|
|
layer = EPathfindingLayer::SAIL;
|
|
|
|
int pos = vstd::find_pos(NPathfindingLayer::names, input["layer"].String());
|
|
|
|
if(pos != -1)
|
|
|
|
layer = EPathfindingLayer(pos);
|
2023-04-19 00:11:24 +02:00
|
|
|
onboardAssaultAllowed = input["onboardAssaultAllowed"].Bool();
|
|
|
|
onboardVisitAllowed = input["onboardVisitAllowed"].Bool();
|
2023-04-18 22:14:15 +02:00
|
|
|
actualAnimation = input["actualAnimation"].String();
|
|
|
|
overlayAnimation = input["overlayAnimation"].String();
|
|
|
|
for(int i = 0; i < flagAnimations.size() && i < input["flagAnimations"].Vector().size(); ++i)
|
|
|
|
flagAnimations[i] = input["flagAnimations"].Vector()[i].String();
|
2023-04-18 23:11:51 +02:00
|
|
|
bonuses = JsonRandom::loadBonuses(input["bonuses"]);
|
2023-04-18 15:27:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
CGObjectInstance * BoatInstanceConstructor::create(std::shared_ptr<const ObjectTemplate> tmpl) const
|
|
|
|
{
|
|
|
|
CGBoat * boat = createTyped(tmpl);
|
|
|
|
boat->layer = layer;
|
2023-04-18 22:14:15 +02:00
|
|
|
boat->actualAnimation = actualAnimation;
|
|
|
|
boat->overlayAnimation = overlayAnimation;
|
|
|
|
boat->flagAnimations = flagAnimations;
|
2023-04-18 23:11:51 +02:00
|
|
|
for(auto & b : bonuses)
|
|
|
|
boat->addNewBonus(std::make_shared<Bonus>(b));
|
|
|
|
|
2023-04-18 15:27:39 +02:00
|
|
|
return boat;
|
|
|
|
}
|
|
|
|
|
2023-04-18 22:14:15 +02:00
|
|
|
void BoatInstanceConstructor::configureObject(CGObjectInstance * object, CRandomGenerator & rng) const
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2023-04-28 03:16:10 +02:00
|
|
|
void MarketInstanceConstructor::initTypeData(const JsonNode & input)
|
|
|
|
{
|
|
|
|
for(auto & element : input["modes"].Vector())
|
|
|
|
{
|
|
|
|
if(MappedKeys::MARKET_NAMES_TO_TYPES.count(element.String()))
|
|
|
|
marketModes.insert(MappedKeys::MARKET_NAMES_TO_TYPES.at(element.String()));
|
|
|
|
}
|
|
|
|
|
|
|
|
marketEfficacy = input["efficacy"].isNull() ? -1 : input["efficacy"].Integer();
|
|
|
|
predefinedOffer = input["offer"];
|
|
|
|
|
|
|
|
title = input["title"].String();
|
|
|
|
speech = input["speech"].String();
|
|
|
|
}
|
|
|
|
|
|
|
|
CGObjectInstance * MarketInstanceConstructor::create(std::shared_ptr<const ObjectTemplate> tmpl) const
|
|
|
|
{
|
|
|
|
CGMarket * market = nullptr;
|
|
|
|
if(marketModes.size() == 1)
|
|
|
|
{
|
|
|
|
switch(*marketModes.begin())
|
|
|
|
{
|
|
|
|
case EMarketMode::ARTIFACT_RESOURCE:
|
|
|
|
case EMarketMode::RESOURCE_ARTIFACT:
|
|
|
|
market = new CGBlackMarket;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EMarketMode::RESOURCE_SKILL:
|
|
|
|
market = new CGUniversity;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!market)
|
|
|
|
market = new CGMarket;
|
|
|
|
|
|
|
|
preInitObject(market);
|
|
|
|
|
|
|
|
if(tmpl)
|
|
|
|
market->appearance = tmpl;
|
|
|
|
market->marketModes = marketModes;
|
|
|
|
if(marketEfficacy >= 0)
|
|
|
|
market->marketEfficacy = marketEfficacy;
|
|
|
|
|
2023-04-29 17:08:42 +02:00
|
|
|
market->title = market->getObjectName();
|
|
|
|
if(!title.empty())
|
|
|
|
market->title = VLC->generaltexth->translate(title);
|
|
|
|
|
2023-04-29 14:10:30 +02:00
|
|
|
market->speech = VLC->generaltexth->translate(speech);
|
|
|
|
|
2023-04-28 03:16:10 +02:00
|
|
|
return market;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MarketInstanceConstructor::configureObject(CGObjectInstance * object, CRandomGenerator & rng) const
|
|
|
|
{
|
|
|
|
if(auto * university = dynamic_cast<CGUniversity *>(object))
|
|
|
|
{
|
|
|
|
for(auto skill : JsonRandom::loadSecondary(predefinedOffer, rng))
|
|
|
|
university->skills.push_back(skill.first.getNum());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-01-11 00:49:31 +02:00
|
|
|
bool CBankInstanceConstructor::hasNameTextID() const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-06-22 13:39:40 +03:00
|
|
|
void CBankInstanceConstructor::initTypeData(const JsonNode & input)
|
|
|
|
{
|
2023-01-11 00:49:31 +02:00
|
|
|
if (input.Struct().count("name") == 0)
|
|
|
|
logMod->warn("Bank %s missing name!", getJsonKey());
|
|
|
|
|
2023-02-09 15:03:49 +02:00
|
|
|
VLC->generaltexth->registerString(input.meta, getNameTextID(), input["name"].String());
|
2023-01-11 00:49:31 +02:00
|
|
|
|
2014-06-22 13:39:40 +03:00
|
|
|
levels = input["levels"].Vector();
|
2020-10-01 10:38:06 +02:00
|
|
|
bankResetDuration = static_cast<si32>(input["resetDuration"].Float());
|
2014-06-22 13:39:40 +03:00
|
|
|
}
|
|
|
|
|
2022-09-11 15:12:35 +02:00
|
|
|
CGObjectInstance *CBankInstanceConstructor::create(std::shared_ptr<const ObjectTemplate> tmpl) const
|
2014-06-22 13:39:40 +03:00
|
|
|
{
|
|
|
|
return createTyped(tmpl);
|
|
|
|
}
|
|
|
|
|
|
|
|
BankConfig CBankInstanceConstructor::generateConfig(const JsonNode & level, CRandomGenerator & rng) const
|
|
|
|
{
|
|
|
|
BankConfig bc;
|
|
|
|
|
2020-10-01 10:38:06 +02:00
|
|
|
bc.chance = static_cast<ui32>(level["chance"].Float());
|
2014-06-22 13:39:40 +03:00
|
|
|
|
|
|
|
bc.guards = JsonRandom::loadCreatures(level["guards"], rng);
|
2020-10-01 10:38:06 +02:00
|
|
|
bc.upgradeChance = static_cast<ui32>(level["upgrade_chance"].Float());
|
|
|
|
bc.combatValue = static_cast<ui32>(level["combat_value"].Float());
|
2014-06-22 13:39:40 +03:00
|
|
|
|
|
|
|
std::vector<SpellID> spells;
|
|
|
|
for (size_t i=0; i<6; i++)
|
2020-10-01 10:38:06 +02:00
|
|
|
IObjectInterface::cb->getAllowedSpells(spells, static_cast<ui16>(i));
|
2014-06-22 13:39:40 +03:00
|
|
|
|
2023-04-05 02:26:29 +02:00
|
|
|
bc.resources = ResourceSet(level["reward"]["resources"]);
|
2014-06-22 13:39:40 +03:00
|
|
|
bc.creatures = JsonRandom::loadCreatures(level["reward"]["creatures"], rng);
|
|
|
|
bc.artifacts = JsonRandom::loadArtifacts(level["reward"]["artifacts"], rng);
|
2021-05-16 19:53:11 +02:00
|
|
|
bc.spells = JsonRandom::loadSpells(level["reward"]["spells"], rng, spells);
|
2014-06-22 13:39:40 +03:00
|
|
|
|
2020-10-01 10:38:06 +02:00
|
|
|
bc.value = static_cast<ui32>(level["value"].Float());
|
2014-06-22 13:39:40 +03:00
|
|
|
|
|
|
|
return bc;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CBankInstanceConstructor::configureObject(CGObjectInstance * object, CRandomGenerator & rng) const
|
|
|
|
{
|
2023-02-12 22:39:17 +02:00
|
|
|
auto * bank = dynamic_cast<CBank *>(object);
|
2014-06-22 13:39:40 +03:00
|
|
|
|
|
|
|
bank->resetDuration = bankResetDuration;
|
|
|
|
|
|
|
|
si32 totalChance = 0;
|
2023-02-12 22:39:17 +02:00
|
|
|
for(const auto & node : levels)
|
2020-10-01 10:38:06 +02:00
|
|
|
totalChance += static_cast<si32>(node["chance"].Float());
|
2014-06-22 13:39:40 +03:00
|
|
|
|
|
|
|
assert(totalChance != 0);
|
|
|
|
|
|
|
|
si32 selectedChance = rng.nextInt(totalChance - 1);
|
|
|
|
|
2015-10-02 17:28:33 +02:00
|
|
|
int cumulativeChance = 0;
|
2023-02-12 22:39:17 +02:00
|
|
|
for(const auto & node : levels)
|
2014-06-22 13:39:40 +03:00
|
|
|
{
|
2020-10-01 10:38:06 +02:00
|
|
|
cumulativeChance += static_cast<int>(node["chance"].Float());
|
2021-05-16 19:53:11 +02:00
|
|
|
if(selectedChance < cumulativeChance)
|
2014-06-22 13:39:40 +03:00
|
|
|
{
|
2021-05-16 19:53:11 +02:00
|
|
|
bank->setConfig(generateConfig(node, rng));
|
|
|
|
break;
|
2014-06-22 13:39:40 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-16 19:53:11 +02:00
|
|
|
CBankInfo::CBankInfo(const JsonVector & Config) :
|
2015-12-14 10:26:14 +02:00
|
|
|
config(Config)
|
2014-06-22 13:39:40 +03:00
|
|
|
{
|
2015-12-14 10:26:14 +02:00
|
|
|
assert(!Config.empty());
|
2014-06-22 13:39:40 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static void addStackToArmy(IObjectInfo::CArmyStructure & army, const CCreature * crea, si32 amount)
|
|
|
|
{
|
2023-04-05 02:26:29 +02:00
|
|
|
army.totalStrength += crea->getFightValue() * amount;
|
2014-06-22 13:39:40 +03:00
|
|
|
|
|
|
|
bool walker = true;
|
2021-05-16 19:53:11 +02:00
|
|
|
if(crea->hasBonusOfType(Bonus::SHOOTER))
|
2014-06-22 13:39:40 +03:00
|
|
|
{
|
2023-04-05 02:26:29 +02:00
|
|
|
army.shootersStrength += crea->getFightValue() * amount;
|
2014-06-22 13:39:40 +03:00
|
|
|
walker = false;
|
|
|
|
}
|
2021-05-16 19:53:11 +02:00
|
|
|
if(crea->hasBonusOfType(Bonus::FLYING))
|
2014-06-22 13:39:40 +03:00
|
|
|
{
|
2023-04-05 02:26:29 +02:00
|
|
|
army.flyersStrength += crea->getFightValue() * amount;
|
2014-06-22 13:39:40 +03:00
|
|
|
walker = false;
|
|
|
|
}
|
2021-05-16 19:53:11 +02:00
|
|
|
if(walker)
|
2023-04-05 02:26:29 +02:00
|
|
|
army.walkersStrength += crea->getFightValue() * amount;
|
2014-06-22 13:39:40 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
IObjectInfo::CArmyStructure CBankInfo::minGuards() const
|
|
|
|
{
|
|
|
|
std::vector<IObjectInfo::CArmyStructure> armies;
|
2021-05-16 19:53:11 +02:00
|
|
|
for(auto configEntry : config)
|
2014-06-22 13:39:40 +03:00
|
|
|
{
|
|
|
|
auto stacks = JsonRandom::evaluateCreatures(configEntry["guards"]);
|
|
|
|
IObjectInfo::CArmyStructure army;
|
2021-05-16 19:53:11 +02:00
|
|
|
for(auto & stack : stacks)
|
2014-06-22 13:39:40 +03:00
|
|
|
{
|
|
|
|
assert(!stack.allowedCreatures.empty());
|
|
|
|
auto weakest = boost::range::min_element(stack.allowedCreatures, [](const CCreature * a, const CCreature * b)
|
|
|
|
{
|
2023-04-05 02:26:29 +02:00
|
|
|
return a->getFightValue() < b->getFightValue();
|
2014-06-22 13:39:40 +03:00
|
|
|
});
|
|
|
|
addStackToArmy(army, *weakest, stack.minAmount);
|
|
|
|
}
|
|
|
|
armies.push_back(army);
|
|
|
|
}
|
|
|
|
return *boost::range::min_element(armies);
|
|
|
|
}
|
|
|
|
|
|
|
|
IObjectInfo::CArmyStructure CBankInfo::maxGuards() const
|
|
|
|
{
|
|
|
|
std::vector<IObjectInfo::CArmyStructure> armies;
|
2021-05-16 19:53:11 +02:00
|
|
|
for(auto configEntry : config)
|
2014-06-22 13:39:40 +03:00
|
|
|
{
|
|
|
|
auto stacks = JsonRandom::evaluateCreatures(configEntry["guards"]);
|
|
|
|
IObjectInfo::CArmyStructure army;
|
2021-05-16 19:53:11 +02:00
|
|
|
for(auto & stack : stacks)
|
2014-06-22 13:39:40 +03:00
|
|
|
{
|
|
|
|
assert(!stack.allowedCreatures.empty());
|
|
|
|
auto strongest = boost::range::max_element(stack.allowedCreatures, [](const CCreature * a, const CCreature * b)
|
|
|
|
{
|
2023-04-05 02:26:29 +02:00
|
|
|
return a->getFightValue() < b->getFightValue();
|
2014-06-22 13:39:40 +03:00
|
|
|
});
|
|
|
|
addStackToArmy(army, *strongest, stack.maxAmount);
|
|
|
|
}
|
|
|
|
armies.push_back(army);
|
|
|
|
}
|
|
|
|
return *boost::range::max_element(armies);
|
|
|
|
}
|
|
|
|
|
2014-12-10 13:29:51 +02:00
|
|
|
TPossibleGuards CBankInfo::getPossibleGuards() const
|
|
|
|
{
|
|
|
|
TPossibleGuards out;
|
|
|
|
|
2021-05-16 19:53:11 +02:00
|
|
|
for(const JsonNode & configEntry : config)
|
2014-12-10 13:29:51 +02:00
|
|
|
{
|
|
|
|
const JsonNode & guardsInfo = configEntry["guards"];
|
2015-03-30 23:55:37 +02:00
|
|
|
auto stacks = JsonRandom::evaluateCreatures(guardsInfo);
|
2014-12-10 13:29:51 +02:00
|
|
|
IObjectInfo::CArmyStructure army;
|
|
|
|
|
|
|
|
|
2021-05-16 19:53:11 +02:00
|
|
|
for(auto stack : stacks)
|
2014-12-10 13:29:51 +02:00
|
|
|
{
|
2023-04-05 02:26:29 +02:00
|
|
|
army.totalStrength += stack.allowedCreatures.front()->getAIValue() * (stack.minAmount + stack.maxAmount) / 2;
|
2014-12-10 13:29:51 +02:00
|
|
|
//TODO: add fields for flyers, walkers etc...
|
|
|
|
}
|
|
|
|
|
2020-10-01 10:38:06 +02:00
|
|
|
ui8 chance = static_cast<ui8>(configEntry["chance"].Float());
|
2014-12-10 13:29:51 +02:00
|
|
|
out.push_back(std::make_pair(chance, army));
|
|
|
|
}
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
2021-05-16 19:53:11 +02:00
|
|
|
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())
|
|
|
|
{
|
2023-02-12 22:39:17 +02:00
|
|
|
result.emplace_back(configEntry["chance"].Integer(), TResources(resourcesInfo));
|
2021-05-16 19:53:11 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<PossibleReward<CStackBasicDescriptor>> CBankInfo::getPossibleCreaturesReward() const
|
|
|
|
{
|
|
|
|
std::vector<PossibleReward<CStackBasicDescriptor>> aproximateReward;
|
|
|
|
|
|
|
|
for(const JsonNode & configEntry : config)
|
|
|
|
{
|
|
|
|
const JsonNode & guardsInfo = configEntry["reward"]["creatures"];
|
|
|
|
auto stacks = JsonRandom::evaluateCreatures(guardsInfo);
|
|
|
|
|
|
|
|
for(auto stack : stacks)
|
|
|
|
{
|
2023-02-12 22:39:17 +02:00
|
|
|
const auto * creature = stack.allowedCreatures.front();
|
2021-05-16 19:53:11 +02:00
|
|
|
|
2023-02-12 22:39:17 +02:00
|
|
|
aproximateReward.emplace_back(configEntry["chance"].Integer(), CStackBasicDescriptor(creature, (stack.minAmount + stack.maxAmount) / 2));
|
2021-05-16 19:53:11 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return aproximateReward;
|
|
|
|
}
|
|
|
|
|
2014-06-22 13:39:40 +03:00
|
|
|
bool CBankInfo::givesResources() const
|
|
|
|
{
|
2021-05-16 19:53:11 +02:00
|
|
|
for(const JsonNode & node : config)
|
|
|
|
if(!node["reward"]["resources"].isNull())
|
2014-06-22 13:39:40 +03:00
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CBankInfo::givesArtifacts() const
|
|
|
|
{
|
2021-05-16 19:53:11 +02:00
|
|
|
for(const JsonNode & node : config)
|
|
|
|
if(!node["reward"]["artifacts"].isNull())
|
2014-06-22 13:39:40 +03:00
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CBankInfo::givesCreatures() const
|
|
|
|
{
|
2021-05-16 19:53:11 +02:00
|
|
|
for(const JsonNode & node : config)
|
|
|
|
if(!node["reward"]["creatures"].isNull())
|
2014-06-22 13:39:40 +03:00
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CBankInfo::givesSpells() const
|
|
|
|
{
|
2021-05-16 19:53:11 +02:00
|
|
|
for(const JsonNode & node : config)
|
|
|
|
if(!node["reward"]["spells"].isNull())
|
2014-06-22 13:39:40 +03:00
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-11 15:12:35 +02:00
|
|
|
std::unique_ptr<IObjectInfo> CBankInstanceConstructor::getObjectInfo(std::shared_ptr<const ObjectTemplate> tmpl) const
|
2014-06-22 13:39:40 +03:00
|
|
|
{
|
|
|
|
return std::unique_ptr<IObjectInfo>(new CBankInfo(levels));
|
|
|
|
}
|
2022-07-26 15:07:42 +02:00
|
|
|
|
|
|
|
VCMI_LIB_NAMESPACE_END
|