1
0
mirror of https://github.com/vcmi/vcmi.git synced 2025-01-24 03:47:18 +02:00
vcmi/lib/campaign/CampaignState.h

326 lines
8.7 KiB
C++
Raw Normal View History

2023-06-25 22:28:24 +03:00
/*
* CampaignState.h, 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
*
*/
#pragma once
#include "../GameConstants.h"
#include "../MetaString.h"
#include "../filesystem/ResourcePath.h"
#include "../CGeneralTextHandler.h"
2023-06-25 22:28:24 +03:00
#include "CampaignConstants.h"
#include "CampaignScenarioPrologEpilog.h"
VCMI_LIB_NAMESPACE_BEGIN
struct StartInfo;
class CGHeroInstance;
class CBinaryReader;
class CInputStream;
class CMap;
class CMapHeader;
class CMapInfo;
class JsonNode;
2023-06-26 01:42:53 +03:00
class Point;
2024-01-01 16:37:48 +02:00
class IGameCallback;
2023-06-25 22:28:24 +03:00
2023-06-26 01:42:53 +03:00
class DLL_LINKAGE CampaignRegions
2023-06-25 22:28:24 +03:00
{
std::string campPrefix;
int colorSuffixLength;
struct DLL_LINKAGE RegionDescription
{
std::string infix;
int xpos;
int ypos;
2023-06-25 22:28:24 +03:00
template <typename Handler> void serialize(Handler &h)
2023-06-25 22:28:24 +03:00
{
h & infix;
h & xpos;
h & ypos;
}
static CampaignRegions::RegionDescription fromJson(const JsonNode & node);
};
std::vector<RegionDescription> regions;
ImagePath getNameFor(CampaignScenarioID which, int color, std::string type) const;
2023-06-26 01:42:53 +03:00
public:
ImagePath getBackgroundName() const;
2023-06-26 01:42:53 +03:00
Point getPosition(CampaignScenarioID which) const;
ImagePath getAvailableName(CampaignScenarioID which, int color) const;
ImagePath getSelectedName(CampaignScenarioID which, int color) const;
ImagePath getConqueredName(CampaignScenarioID which, int color) const;
2023-06-26 01:42:53 +03:00
template <typename Handler> void serialize(Handler &h)
2023-06-25 22:28:24 +03:00
{
h & campPrefix;
h & colorSuffixLength;
h & regions;
}
static CampaignRegions fromJson(const JsonNode & node);
static CampaignRegions getLegacy(int campId);
};
class DLL_LINKAGE CampaignHeader : public boost::noncopyable
2023-06-25 22:28:24 +03:00
{
friend class CampaignHandler;
2023-06-25 22:28:24 +03:00
CampaignVersion version = CampaignVersion::NONE;
CampaignRegions campaignRegions;
2023-09-27 22:53:13 +02:00
MetaString name;
MetaString description;
2023-09-04 13:03:15 +03:00
AudioPath music;
2023-06-25 22:28:24 +03:00
std::string filename;
std::string modName;
std::string encoding;
int numberOfScenarios = 0;
bool difficultyChoosenByPlayer = false;
void loadLegacyData(ui8 campId);
TextContainerRegistrable textContainer;
public:
bool playerSelectedDifficulty() const;
bool formatVCMI() const;
2023-09-27 22:53:13 +02:00
std::string getDescriptionTranslated() const;
std::string getNameTranslated() const;
std::string getFilename() const;
std::string getModName() const;
std::string getEncoding() const;
2023-09-04 13:03:15 +03:00
AudioPath getMusic() const;
const CampaignRegions & getRegions() const;
TextContainerRegistrable & getTexts();
template <typename Handler> void serialize(Handler &h)
2023-06-25 22:28:24 +03:00
{
h & version;
h & campaignRegions;
h & numberOfScenarios;
h & name;
h & description;
h & difficultyChoosenByPlayer;
h & filename;
h & modName;
h & music;
2023-06-25 22:28:24 +03:00
h & encoding;
if (h.version >= Handler::Version::RELEASE_143)
h & textContainer;
2023-06-25 22:28:24 +03:00
}
};
struct DLL_LINKAGE CampaignBonus
{
2023-06-26 01:42:53 +03:00
CampaignBonusType type = CampaignBonusType::NONE;
2023-06-25 22:28:24 +03:00
//purpose depends on type
int32_t info1 = 0;
int32_t info2 = 0;
int32_t info3 = 0;
bool isBonusForHero() const;
template <typename Handler> void serialize(Handler &h)
2023-06-25 22:28:24 +03:00
{
h & type;
h & info1;
h & info2;
h & info3;
}
};
2023-06-26 01:42:53 +03:00
struct DLL_LINKAGE CampaignTravel
2023-06-25 22:28:24 +03:00
{
struct DLL_LINKAGE WhatHeroKeeps
{
bool experience = false;
bool primarySkills = false;
bool secondarySkills = false;
bool spells = false;
bool artifacts = false;
template <typename Handler> void serialize(Handler &h)
2023-06-25 22:28:24 +03:00
{
h & experience;
h & primarySkills;
h & secondarySkills;
h & spells;
h & artifacts;
}
};
std::set<CreatureID> monstersKeptByHero;
std::set<ArtifactID> artifactsKeptByHero;
std::vector<CampaignBonus> bonusesToChoose;
WhatHeroKeeps whatHeroKeeps;
CampaignStartOptions startOptions = CampaignStartOptions::NONE; //1 - start bonus, 2 - traveling hero, 3 - hero options
PlayerColor playerColor = PlayerColor::NEUTRAL; //only for startOptions == 1
template <typename Handler> void serialize(Handler &h)
2023-06-25 22:28:24 +03:00
{
h & whatHeroKeeps;
h & monstersKeptByHero;
h & artifactsKeptByHero;
h & startOptions;
h & playerColor;
h & bonusesToChoose;
}
};
2023-06-26 01:42:53 +03:00
struct DLL_LINKAGE CampaignScenario
2023-06-25 22:28:24 +03:00
{
std::string mapName; //*.h3m
2023-09-27 22:53:13 +02:00
MetaString scenarioName; //from header
2023-06-25 22:28:24 +03:00
std::set<CampaignScenarioID> preconditionRegions; //what we need to conquer to conquer this one (stored as bitfield in h3c)
ui8 regionColor = 0;
ui8 difficulty = 0;
2023-09-28 00:10:28 +02:00
MetaString regionText;
2023-06-25 22:28:24 +03:00
CampaignScenarioPrologEpilog prolog;
CampaignScenarioPrologEpilog epilog;
CampaignTravel travelOptions;
void loadPreconditionRegions(ui32 regions);
bool isNotVoid() const;
template <typename Handler> void serialize(Handler &h)
2023-06-25 22:28:24 +03:00
{
h & mapName;
h & scenarioName;
h & preconditionRegions;
h & regionColor;
h & difficulty;
h & regionText;
h & prolog;
h & epilog;
h & travelOptions;
}
};
/// Class that represents loaded campaign information
class DLL_LINKAGE Campaign : public CampaignHeader
{
friend class CampaignHandler;
std::map<CampaignScenarioID, CampaignScenario> scenarios;
public:
const CampaignScenario & scenario(CampaignScenarioID which) const;
std::set<CampaignScenarioID> allScenarios() const;
int scenariosCount() const;
template <typename Handler> void serialize(Handler &h)
{
h & static_cast<CampaignHeader&>(*this);
h & scenarios;
}
};
/// Class that represent campaign that is being played at
/// Contains campaign itself as well as current state of the campaign
class DLL_LINKAGE CampaignState : public Campaign
2023-06-25 22:28:24 +03:00
{
friend class CampaignHandler;
using ScenarioPoolType = std::vector<JsonNode>;
using CampaignPoolType = std::map<CampaignScenarioID, ScenarioPoolType>;
using GlobalPoolType = std::map<HeroTypeID, JsonNode>;
2023-06-25 22:28:24 +03:00
/// List of all maps completed by player, in order of their completion
2023-06-25 22:28:24 +03:00
std::vector<CampaignScenarioID> mapsConquered;
/// List of previously loaded campaign maps, to prevent translation of transferred hero names getting lost after their original map has been completed
std::map<CampaignScenarioID, TextContainerRegistrable> mapTranslations;
std::map<CampaignScenarioID, std::vector<uint8_t> > mapPieces; //binary h3ms, scenario number -> map data
2023-06-25 22:28:24 +03:00
std::map<CampaignScenarioID, ui8> chosenCampaignBonuses;
std::optional<CampaignScenarioID> currentMap;
/// Heroes from specific scenario, ordered by descending strength
CampaignPoolType scenarioHeroPool;
/// Pool of heroes currently reserved for usage in campaign
GlobalPoolType globalHeroPool;
2023-06-25 22:28:24 +03:00
public:
CampaignState() = default;
2023-06-26 01:42:53 +03:00
/// Returns last completed scenario, if any
std::optional<CampaignScenarioID> lastScenario() const;
2023-06-26 01:42:53 +03:00
std::optional<CampaignScenarioID> currentScenario() const;
std::set<CampaignScenarioID> conqueredScenarios() const;
2023-06-26 01:42:53 +03:00
/// Returns bonus selected for specific scenario
std::optional<CampaignBonus> getBonus(CampaignScenarioID which) const;
2023-06-26 01:42:53 +03:00
/// Returns index of selected bonus for specified scenario
std::optional<ui8> getBonusID(CampaignScenarioID which) const;
/// Returns true if selected scenario can be selected and started by player
bool isAvailable(CampaignScenarioID whichScenario) const;
/// Returns true if selected scenario has been already completed by player
bool isConquered(CampaignScenarioID whichScenario) const;
/// Returns true if all available scenarios have been completed and campaign is finished
bool isCampaignFinished() const;
std::unique_ptr<CMap> getMap(CampaignScenarioID scenarioId, IGameCallback * cb);
2023-06-25 22:28:24 +03:00
std::unique_ptr<CMapHeader> getMapHeader(CampaignScenarioID scenarioId) const;
std::shared_ptr<CMapInfo> getMapInfo(CampaignScenarioID scenarioId) const;
void setCurrentMap(CampaignScenarioID which);
void setCurrentMapBonus(ui8 which);
void setCurrentMapAsConquered(std::vector<CGHeroInstance*> heroes);
/// Returns list of heroes that must be reserved for campaign and can only be used for hero placeholders
std::set<HeroTypeID> getReservedHeroes() const;
/// Returns strongest hero from specified scenario, or null if none found
const CGHeroInstance * strongestHero(CampaignScenarioID scenarioId, const PlayerColor & owner) const;
/// Returns heroes that can be instantiated as hero placeholders by power
const std::vector<JsonNode> & getHeroesByPower(CampaignScenarioID scenarioId) const;
/// Returns hero for instantiation as placeholder by type
/// May return empty JsonNode if such hero was not found
const JsonNode & getHeroByType(HeroTypeID heroID) const;
2024-01-01 16:37:48 +02:00
JsonNode crossoverSerialize(CGHeroInstance * hero) const;
CGHeroInstance * crossoverDeserialize(const JsonNode & node, CMap * map) const;
2023-06-25 22:28:24 +03:00
2023-09-21 21:27:06 +02:00
std::string campaignSet;
2023-09-20 03:13:54 +02:00
template <typename Handler> void serialize(Handler &h)
2023-06-25 22:28:24 +03:00
{
h & static_cast<Campaign&>(*this);
h & scenarioHeroPool;
h & globalHeroPool;
2023-06-25 22:28:24 +03:00
h & mapPieces;
h & mapsConquered;
h & currentMap;
h & chosenCampaignBonuses;
2023-09-20 03:13:54 +02:00
h & campaignSet;
if (h.version >= Handler::Version::CAMPAIGN_MAP_TRANSLATIONS)
h & mapTranslations;
2023-06-25 22:28:24 +03:00
}
};
VCMI_LIB_NAMESPACE_END