/* * JsonRandom.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 "JsonRandom.h" #include "../JsonNode.h" #include "../CRandomGenerator.h" #include "../StringConstants.h" #include "../VCMI_Lib.h" #include "../CModHandler.h" #include "../CArtHandler.h" #include "../CCreatureHandler.h" #include "../CCreatureSet.h" #include "../spells/CSpellHandler.h" VCMI_LIB_NAMESPACE_BEGIN namespace JsonRandom { si32 loadValue(const JsonNode & value, CRandomGenerator & rng, si32 defaultValue) { if (value.isNull()) return defaultValue; if (value.isNumber()) return static_cast(value.Float()); if (value.isVector()) { const auto & vector = value.Vector(); size_t index= rng.getIntRange(0, vector.size()-1)(); return loadValue(vector[index], rng, 0); } if (!value["amount"].isNull()) return static_cast(loadValue(value["amount"], rng, defaultValue)); si32 min = static_cast(loadValue(value["min"], rng, 0)); si32 max = static_cast(loadValue(value["max"], rng, 0)); return rng.getIntRange(min, max)(); } DLL_LINKAGE std::string loadKey(const JsonNode & value, CRandomGenerator & rng, std::string defaultValue) { if (value.isNull()) return defaultValue; if (value.isString()) return value.String(); if (!value["type"].isNull()) return value["type"].String(); if (value["list"].isNull()) return defaultValue; const auto & resourceList = value["list"].Vector(); if (resourceList.empty()) return defaultValue; si32 index = rng.getIntRange(0, resourceList.size() - 1 )(); return resourceList[index].String(); } TResources loadResources(const JsonNode & value, CRandomGenerator & rng) { TResources ret; if (value.isVector()) { for (const auto & entry : value.Vector()) ret += loadResource(entry, rng); return ret; } for (size_t i=0; imodh->identifiers.getIdentifier(value.meta, "resource", resourceName).get()); TResources ret; ret[resourceID] = resourceAmount; return ret; } std::vector loadPrimary(const JsonNode & value, CRandomGenerator & rng) { std::vector ret; for(const auto & name : PrimarySkill::names) { ret.push_back(loadValue(value[name], rng)); } return ret; } std::map loadSecondary(const JsonNode & value, CRandomGenerator & rng) { std::map ret; for(const auto & pair : value.Struct()) { SecondarySkill id(VLC->modh->identifiers.getIdentifier(pair.second.meta, "skill", pair.first).get()); ret[id] = loadValue(pair.second, rng); } return ret; } ArtifactID loadArtifact(const JsonNode & value, CRandomGenerator & rng) { if (value.getType() == JsonNode::JsonType::DATA_STRING) return ArtifactID(VLC->modh->identifiers.getIdentifier("artifact", value).get()); std::set allowedClasses; std::set allowedPositions; ui32 minValue = 0; ui32 maxValue = std::numeric_limits::max(); if (value["class"].getType() == JsonNode::JsonType::DATA_STRING) allowedClasses.insert(CArtHandler::stringToClass(value["class"].String())); else for(const auto & entry : value["class"].Vector()) allowedClasses.insert(CArtHandler::stringToClass(entry.String())); if (value["slot"].getType() == JsonNode::JsonType::DATA_STRING) allowedPositions.insert(ArtifactPosition(value["class"].String())); else for(const auto & entry : value["slot"].Vector()) allowedPositions.insert(ArtifactPosition(entry.String())); if (!value["minValue"].isNull()) minValue = static_cast(value["minValue"].Float()); if (!value["maxValue"].isNull()) maxValue = static_cast(value["maxValue"].Float()); return VLC->arth->pickRandomArtifact(rng, [=](const ArtifactID & artID) -> bool { CArtifact * art = VLC->arth->objects[artID]; if(!vstd::iswithin(art->price, minValue, maxValue)) return false; if(!allowedClasses.empty() && !allowedClasses.count(art->aClass)) return false; if(!allowedPositions.empty()) { for(const auto & pos : art->possibleSlots[ArtBearer::HERO]) { if(allowedPositions.count(pos)) return true; } return false; } return true; }); } std::vector loadArtifacts(const JsonNode & value, CRandomGenerator & rng) { std::vector ret; for (const JsonNode & entry : value.Vector()) { ret.push_back(loadArtifact(entry, rng)); } return ret; } SpellID loadSpell(const JsonNode & value, CRandomGenerator & rng, std::vector spells) { if (value.getType() == JsonNode::JsonType::DATA_STRING) return SpellID(VLC->modh->identifiers.getIdentifier("spell", value).get()); vstd::erase_if(spells, [=](const SpellID & spell) { return VLC->spellh->objects[spell]->level != si32(value["level"].Float()); }); return SpellID(*RandomGeneratorUtil::nextItem(spells, rng)); } std::vector loadSpells(const JsonNode & value, CRandomGenerator & rng, const std::vector & spells) { // possible extensions: (taken from spell json config) // "type": "adventure",//"adventure", "combat", "ability" // "school": {"air":true, "earth":true, "fire":true, "water":true}, // "level": 1, std::vector ret; for (const JsonNode & entry : value.Vector()) { ret.push_back(loadSpell(entry, rng, spells)); } return ret; } CStackBasicDescriptor loadCreature(const JsonNode & value, CRandomGenerator & rng) { CStackBasicDescriptor stack; stack.type = VLC->creh->objects[VLC->modh->identifiers.getIdentifier("creature", value["type"]).get()]; stack.count = loadValue(value, rng); if (!value["upgradeChance"].isNull() && !stack.type->upgrades.empty()) { if (int(value["upgradeChance"].Float()) > rng.nextInt(99)) // select random upgrade { stack.type = VLC->creh->objects[*RandomGeneratorUtil::nextItem(stack.type->upgrades, rng)]; } } return stack; } std::vector loadCreatures(const JsonNode & value, CRandomGenerator & rng) { std::vector ret; for (const JsonNode & node : value.Vector()) { ret.push_back(loadCreature(node, rng)); } return ret; } std::vector evaluateCreatures(const JsonNode & value) { std::vector ret; for (const JsonNode & node : value.Vector()) { RandomStackInfo info; if (!node["amount"].isNull()) info.minAmount = info.maxAmount = static_cast(node["amount"].Float()); else { info.minAmount = static_cast(node["min"].Float()); info.maxAmount = static_cast(node["max"].Float()); } const CCreature * crea = VLC->creh->objects[VLC->modh->identifiers.getIdentifier("creature", node["type"]).get()]; info.allowedCreatures.push_back(crea); if (node["upgradeChance"].Float() > 0) { for(const auto & creaID : crea->upgrades) info.allowedCreatures.push_back(VLC->creh->objects[creaID]); } ret.push_back(info); } return ret; } //std::vector loadComponents(const JsonNode & value) //{ // std::vector ret; // return ret; // //TODO //} std::vector DLL_LINKAGE loadBonuses(const JsonNode & value) { std::vector ret; for (const JsonNode & entry : value.Vector()) { auto bonus = JsonUtils::parseBonus(entry); ret.push_back(*bonus); } return ret; } } VCMI_LIB_NAMESPACE_END