| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295 | #pragma once#include "ConstTransitivePtr.h"#include "ResourceSet.h"#include "int3.h"#include "GameConstants.h"#include "IHandlerBase.h"#include "LogicalExpression.h"#include "BattleHex.h"/* * CTownHandler.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 * */class CLegacyConfigParser;class JsonNode;class CTown;class CFaction;struct BattleHex;/// a typical building encountered in every castle ;]/// this is structure available to both client and server/// contains all mechanics-related data about town structuresclass DLL_LINKAGE CBuilding{	std::string name;	std::string description;public:	typedef LogicalExpression<BuildingID> TRequired;	CTown * town; // town this building belongs to	TResources resources;	TResources produce;	TRequired requirements;	std::string identifier;	BuildingID bid; //structure ID	BuildingID upgrade; /// indicates that building "upgrade" can be improved by this, -1 = empty	enum EBuildMode	{		BUILD_NORMAL,  // 0 - normal, default		BUILD_AUTO,    // 1 - auto - building appears when all requirements are built		BUILD_SPECIAL, // 2 - special - building can not be built normally		BUILD_GRAIL    // 3 - grail - building reqires grail to be built	} mode;	const std::string &Name() const;	const std::string &Description() const;	//return base of upgrade(s) or this	BuildingID getBase() const;	// returns how many times build has to be upgraded to become build	si32 getDistance(BuildingID build) const;	template <typename Handler> void serialize(Handler &h, const int version)	{		h & identifier & town & bid & resources & produce & name & description & requirements & upgrade & mode;	}	friend class CTownHandler;};/// This is structure used only by client/// Consists of all gui-related data about town structures/// Should be moved from lib to clientstruct DLL_LINKAGE CStructure{	CBuilding * building;  // base building. If null - this structure will be always present on screen	CBuilding * buildable; // building that will be used to determine built building and visible cost. Usually same as "building"	int3 pos;	std::string defName, borderName, areaName, identifier;	bool hiddenUpgrade; // used only if "building" is upgrade, if true - structure on town screen will behave exactly like parent (mouse clicks, hover texts, etc)	template <typename Handler> void serialize(Handler &h, const int version)	{		h & pos & defName & borderName & areaName & identifier & building & buildable & hiddenUpgrade;	}};struct DLL_LINKAGE SPuzzleInfo{	ui16 number; //type of puzzle	si16 x, y; //position	ui16 whenUncovered; //determines the sequnce of discovering (the lesser it is the sooner puzzle will be discovered)	std::string filename; //file with graphic of this puzzle	template <typename Handler> void serialize(Handler &h, const int version)	{		h & number & x & y & whenUncovered & filename;	}};class DLL_LINKAGE CFaction{public:	CFaction();	~CFaction();	std::string name; //town name, by default - from TownName.txt	std::string identifier;	TFaction index;	ETerrainType nativeTerrain;	EAlignment::EAlignment alignment;	CTown * town; //NOTE: can be null	std::string creatureBg120;	std::string creatureBg130;	std::vector<SPuzzleInfo> puzzleMap;	template <typename Handler> void serialize(Handler &h, const int version)	{		h & name & identifier & index & nativeTerrain & alignment & town & creatureBg120 & creatureBg130 & puzzleMap;	}};class DLL_LINKAGE CTown{public:	CTown();	~CTown();	// TODO: remove once save and mod compatability not needed	static std::vector<BattleHex> defaultMoatHexes();	CFaction * faction;	std::vector<std::string> names; //names of the town instances	/// level -> list of creatures on this tier	// TODO: replace with pointers to CCreature	std::vector<std::vector<CreatureID> > creatures;	std::map<BuildingID, ConstTransitivePtr<CBuilding> > buildings;	std::vector<std::string> dwellings; //defs for adventure map dwellings for new towns, [0] means tier 1 creatures etc.	std::vector<std::string> dwellingNames;	// should be removed at least from configs in favor of auto-detection	std::map<int,int> hordeLvl; //[0] - first horde building creature level; [1] - second horde building (-1 if not present)	ui32 mageLevel; //max available mage guild level	ui16 primaryRes;	ArtifactID warMachine;	si32 moatDamage;	std::vector<BattleHex> moatHexes;	// default chance for hero of specific class to appear in tavern, if field "tavern" was not set	// resulting chance = sqrt(town.chance * heroClass.chance)	ui32 defaultTavernChance;	// Client-only data. Should be moved away from lib	struct ClientInfo	{		struct Point		{			si32 x;			si32 y;			template <typename Handler> void serialize(Handler &h, const int version)			{ h & x & y; }		};		//icons [fort is present?][build limit reached?] -> index of icon in def files		int icons[2][2];		std::string iconSmall[2][2]; /// icon names used during loading		std::string iconLarge[2][2];		std::string tavernVideo;		std::string musicTheme;		std::string townBackground;		std::string guildBackground;		std::string guildWindow;		std::string buildingsIcons;		std::string hallBackground;		/// vector[row][column] = list of buildings in this slot		std::vector< std::vector< std::vector<BuildingID> > > hallSlots;		/// list of town screen structures.		/// NOTE: index in vector is meaningless. Vector used instead of list for a bit faster access		std::vector<ConstTransitivePtr<CStructure> > structures;		std::string siegePrefix;		std::vector<Point> siegePositions;		CreatureID siegeShooter; // shooter creature ID		template <typename Handler> void serialize(Handler &h, const int version)		{			h & icons & iconSmall & iconLarge & tavernVideo & musicTheme & townBackground & guildBackground & guildWindow & buildingsIcons & hallBackground;			h & hallSlots & structures;			h & siegePrefix & siegePositions & siegeShooter;		}	} clientInfo;	template <typename Handler> void serialize(Handler &h, const int version)	{		h & names & faction & creatures & dwellings & dwellingNames & buildings & hordeLvl & mageLevel			& primaryRes & warMachine & clientInfo & moatDamage;		if(version >= 758)		{			h & moatHexes;		}		else if(!h.saving)		{			moatHexes = defaultMoatHexes();		}		h & defaultTavernChance;		auto findNull = [](const std::pair<BuildingID, ConstTransitivePtr<CBuilding>> &building)		{ return building.second == nullptr; };		//Fix #1444 corrupted save		while(auto badElem = vstd::tryFindIf(buildings, findNull))		{			logGlobal->errorStream() << "#1444-like bug encountered in CTown::serialize, fixing buildings list by removing bogus entry " << badElem->first << " from " << faction->name;			buildings.erase(badElem->first);		}	}};class DLL_LINKAGE CTownHandler : public IHandlerBase{	struct BuildingRequirementsHelper	{		JsonNode json;		CBuilding * building;		CFaction * faction;	};	std::vector<BuildingRequirementsHelper> requirementsToLoad;	void initializeRequirements();	/// loads CBuilding's into town	void loadBuildingRequirements(CTown &town, CBuilding & building, const JsonNode & source);	void loadBuilding(CTown &town, const std::string & stringID, const JsonNode & source);	void loadBuildings(CTown &town, const JsonNode & source);	/// loads CStructure's into town	void loadStructure(CTown &town, const std::string & stringID, const JsonNode & source);	void loadStructures(CTown &town, const JsonNode & source);	/// loads town hall vector (hallSlots)	void loadTownHall(CTown &town, const JsonNode & source);	void loadSiegeScreen(CTown &town, const JsonNode & source);	void loadClientData(CTown &town, const JsonNode & source);	void loadTown(CTown &town, const JsonNode & source);	void loadPuzzle(CFaction & faction, const JsonNode & source);	CFaction * loadFromJson(const JsonNode & data, const std::string & identifier);public:	std::vector<ConstTransitivePtr<CFaction> > factions;	CTownHandler(); //c-tor, set pointer in VLC to this	~CTownHandler();	std::vector<JsonNode> loadLegacyData(size_t dataSize) override;	void loadObject(std::string scope, std::string name, const JsonNode & data) override;	void loadObject(std::string scope, std::string name, const JsonNode & data, size_t index) override;	void afterLoadFinalization() override;	std::vector<bool> getDefaultAllowed() const override;	std::set<TFaction> getAllowedFactions(bool withTown = true) const;	//json serialization helper	static si32 decodeFaction(const std::string & identifier);	//json serialization helper	static std::string encodeFaction(const si32 index);	template <typename Handler> void serialize(Handler &h, const int version)	{		h & factions;	}};
 |