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

381 lines
10 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 "../filesystem/ResourcePath.h"
#include "../serializer/Serializeable.h"
#include "../texts/TextLocalizationContainer.h"
2023-06-25 22:28:24 +03:00
#include "CampaignConstants.h"
#include "CampaignScenarioPrologEpilog.h"
2024-07-26 03:33:44 +02:00
#include "../gameState/HighScore.h"
2024-09-27 18:14:48 +02:00
#include "../Point.h"
2023-06-25 22:28:24 +03:00
VCMI_LIB_NAMESPACE_BEGIN
struct StartInfo;
class CGHeroInstance;
class CBinaryReader;
class CInputStream;
class CMap;
class CMapHeader;
class CMapInfo;
class JsonNode;
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;
2024-08-12 17:57:34 +02:00
std::vector<std::string> campSuffix;
std::string campBackground;
2023-06-25 22:28:24 +03:00
int colorSuffixLength;
struct DLL_LINKAGE RegionDescription
{
std::string infix;
2024-09-27 18:14:48 +02:00
Point pos;
std::optional<Point> labelPos;
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;
2024-09-24 11:23:10 +02:00
if (h.version >= Handler::Version::REGION_LABEL)
{
2024-09-27 18:14:48 +02:00
h & pos;
h & labelPos;
}
else
{
h & pos.x;
h & pos.y;
2024-09-24 11:23:10 +02:00
}
2023-06-25 22:28:24 +03:00
}
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;
2024-09-24 11:23:10 +02:00
std::optional<Point> getLabelPosition(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;
2024-08-10 15:27:22 +02:00
if (h.version >= Handler::Version::CAMPAIGN_REGIONS)
{
h & campSuffix;
h & campBackground;
}
2023-06-25 22:28:24 +03:00
}
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;
2024-08-31 14:17:25 +02:00
friend class Campaign;
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;
2024-06-29 13:13:59 +02:00
MetaString author;
MetaString authorContact;
2024-06-29 14:01:25 +02:00
MetaString campaignVersion;
2024-06-29 13:13:59 +02:00
std::time_t creationDateTime;
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;
ImagePath loadingBackground;
2024-09-05 21:31:17 +02:00
ImagePath videoRim;
2024-08-31 17:57:27 +02:00
VideoPath introVideo;
2024-09-05 21:31:17 +02:00
VideoPath outroVideo;
2023-06-25 22:28:24 +03:00
int numberOfScenarios = 0;
bool difficultyChosenByPlayer = false;
void loadLegacyData(ui8 campId);
2024-08-31 13:15:07 +02:00
void loadLegacyData(CampaignRegions regions, int numOfScenario);
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;
2024-08-10 14:59:22 +02:00
std::string getAuthor() const;
std::string getAuthorContact() const;
std::string getCampaignVersion() const;
time_t getCreationDateTime() const;
std::string getFilename() const;
std::string getModName() const;
std::string getEncoding() const;
2023-09-04 13:03:15 +03:00
AudioPath getMusic() const;
ImagePath getLoadingBackground() const;
2024-09-05 21:31:17 +02:00
ImagePath getVideoRim() const;
2024-08-31 17:57:27 +02:00
VideoPath getIntroVideo() const;
2024-09-05 21:31:17 +02:00
VideoPath getOutroVideo() 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;
2024-06-29 13:13:59 +02:00
if (h.version >= Handler::Version::MAP_FORMAT_ADDITIONAL_INFOS)
{
h & author;
h & authorContact;
2024-06-29 14:01:25 +02:00
h & campaignVersion;
2024-06-29 13:13:59 +02:00
h & creationDateTime;
}
h & difficultyChosenByPlayer;
2023-06-25 22:28:24 +03:00
h & filename;
h & modName;
h & music;
2023-06-25 22:28:24 +03:00
h & encoding;
h & textContainer;
if (h.version >= Handler::Version::CHRONICLES_SUPPORT)
{
h & loadingBackground;
2024-09-05 21:31:17 +02:00
h & videoRim;
2024-08-31 17:57:27 +02:00
h & introVideo;
}
2024-09-05 21:31:17 +02:00
if (h.version >= Handler::Version::CAMPAIGN_OUTRO_SUPPORT)
h & outroVideo;
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, public Serializeable
{
friend class CampaignHandler;
std::map<CampaignScenarioID, CampaignScenario> scenarios;
public:
const CampaignScenario & scenario(CampaignScenarioID which) const;
std::set<CampaignScenarioID> allScenarios() const;
int scenariosCount() const;
2024-09-01 12:23:10 +02:00
void overrideCampaign();
void overrideCampaignScenarios();
2024-08-31 14:17:25 +02:00
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
2024-07-26 03:33:44 +02:00
std::vector<HighScoreParameter> highscoreParameters;
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;
h & mapTranslations;
2024-07-26 03:33:44 +02:00
if (h.version >= Handler::Version::HIGHSCORE_PARAMETERS)
h & highscoreParameters;
2023-06-25 22:28:24 +03:00
}
};
VCMI_LIB_NAMESPACE_END