2024-07-27 02:11:26 +02:00
|
|
|
/*
|
|
|
|
* GameStatistics.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 "GameStatistics.h"
|
2024-08-01 21:30:53 +02:00
|
|
|
#include "../CPlayerState.h"
|
|
|
|
#include "../constants/StringConstants.h"
|
|
|
|
#include "CGameState.h"
|
2024-08-01 23:21:41 +02:00
|
|
|
#include "TerrainHandler.h"
|
2024-08-02 01:18:39 +02:00
|
|
|
#include "CHeroHandler.h"
|
2024-08-01 23:56:06 +02:00
|
|
|
#include "StartInfo.h"
|
2024-08-02 19:37:46 +02:00
|
|
|
#include "HighScore.h"
|
2024-08-01 22:36:32 +02:00
|
|
|
#include "../mapObjects/CGHeroInstance.h"
|
|
|
|
#include "../mapObjects/CGTownInstance.h"
|
|
|
|
#include "../mapObjects/CGObjectInstance.h"
|
|
|
|
#include "../mapObjects/MiscObjects.h"
|
|
|
|
#include "../mapping/CMap.h"
|
2024-07-27 02:11:26 +02:00
|
|
|
|
|
|
|
VCMI_LIB_NAMESPACE_BEGIN
|
|
|
|
|
|
|
|
void StatisticDataSet::add(StatisticDataSetEntry entry)
|
|
|
|
{
|
|
|
|
data.push_back(entry);
|
|
|
|
}
|
|
|
|
|
2024-08-01 21:30:53 +02:00
|
|
|
StatisticDataSetEntry StatisticDataSet::createEntry(const PlayerState * ps, const CGameState * gs)
|
|
|
|
{
|
|
|
|
StatisticDataSetEntry data;
|
|
|
|
|
2024-08-02 19:37:46 +02:00
|
|
|
HighScoreParameter param = HighScore::prepareHighScores(gs, ps->color, false);
|
|
|
|
HighScoreCalculation scenarioHighScores;
|
|
|
|
scenarioHighScores.parameters.push_back(param);
|
|
|
|
scenarioHighScores.isCampaign = false;
|
|
|
|
|
2024-08-02 20:40:24 +02:00
|
|
|
data.map = gs->map->name.toString();
|
|
|
|
data.timestamp = std::time(0);
|
2024-08-01 21:30:53 +02:00
|
|
|
data.day = gs->getDate(Date::DAY);
|
|
|
|
data.player = ps->color;
|
|
|
|
data.team = ps->team;
|
2024-08-01 23:21:41 +02:00
|
|
|
data.isHuman = ps->isHuman();
|
|
|
|
data.status = ps->status;
|
2024-08-01 21:30:53 +02:00
|
|
|
data.resources = ps->resources;
|
2024-08-01 22:36:32 +02:00
|
|
|
data.numberHeroes = ps->heroes.size();
|
2024-08-02 19:37:46 +02:00
|
|
|
data.numberTowns = gs->howManyTowns(ps->color);
|
2024-08-01 22:36:32 +02:00
|
|
|
data.numberArtifacts = Statistic::getNumberOfArts(ps);
|
2024-08-02 01:18:39 +02:00
|
|
|
data.armyStrength = Statistic::getArmyStrength(ps, true);
|
2024-08-01 23:56:06 +02:00
|
|
|
data.income = Statistic::getIncome(gs, ps);
|
2024-08-01 23:21:41 +02:00
|
|
|
data.mapVisitedRatio = Statistic::getMapVisitedRatio(gs, ps->color);
|
2024-08-02 01:18:39 +02:00
|
|
|
data.obeliskVisited = Statistic::getObeliskVisited(gs, ps->team);
|
|
|
|
data.mightMagicRatio = Statistic::getMightMagicRatio(ps);
|
|
|
|
data.numMines = Statistic::getNumMines(gs, ps);
|
2024-08-02 19:37:46 +02:00
|
|
|
data.score = scenarioHighScores.calculate().total;
|
2024-08-02 20:40:24 +02:00
|
|
|
data.maxHeroLevel = Statistic::findBestHero(gs, ps->color) ? Statistic::findBestHero(gs, ps->color)->level : 0;
|
2024-08-02 20:06:30 +02:00
|
|
|
data.numBattlesNeutral = gs->statistic.values.numBattlesNeutral.count(ps->color) ? gs->statistic.values.numBattlesNeutral.at(ps->color) : 0;
|
|
|
|
data.numBattlesPlayer = gs->statistic.values.numBattlesPlayer.count(ps->color) ? gs->statistic.values.numBattlesPlayer.at(ps->color) : 0;
|
|
|
|
data.numWinBattlesNeutral = gs->statistic.values.numWinBattlesNeutral.count(ps->color) ? gs->statistic.values.numWinBattlesNeutral.at(ps->color) : 0;
|
|
|
|
data.numWinBattlesPlayer = gs->statistic.values.numWinBattlesPlayer.count(ps->color) ? gs->statistic.values.numWinBattlesPlayer.at(ps->color) : 0;
|
2024-08-02 20:40:24 +02:00
|
|
|
data.numHeroSurrendered = gs->statistic.values.numHeroSurrendered.count(ps->color) ? gs->statistic.values.numHeroSurrendered.at(ps->color) : 0;
|
|
|
|
data.numHeroEscaped = gs->statistic.values.numHeroEscaped.count(ps->color) ? gs->statistic.values.numHeroEscaped.at(ps->color) : 0;
|
2024-08-01 21:30:53 +02:00
|
|
|
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2024-07-27 02:11:26 +02:00
|
|
|
std::string StatisticDataSet::toCsv()
|
|
|
|
{
|
|
|
|
std::stringstream ss;
|
|
|
|
|
2024-08-01 21:30:53 +02:00
|
|
|
auto resources = std::vector<EGameResID>{EGameResID::GOLD, EGameResID::WOOD, EGameResID::MERCURY, EGameResID::ORE, EGameResID::SULFUR, EGameResID::CRYSTAL, EGameResID::GEMS};
|
|
|
|
|
2024-08-02 20:40:24 +02:00
|
|
|
ss << "Map" << ";";
|
|
|
|
ss << "Timestamp" << ";";
|
2024-08-01 23:21:41 +02:00
|
|
|
ss << "Day" << ";";
|
|
|
|
ss << "Player" << ";";
|
|
|
|
ss << "Team" << ";";
|
|
|
|
ss << "IsHuman" << ";";
|
|
|
|
ss << "Status" << ";";
|
|
|
|
ss << "NumberHeroes" << ";";
|
|
|
|
ss << "NumberTowns" << ";";
|
|
|
|
ss << "NumberArtifacts" << ";";
|
|
|
|
ss << "ArmyStrength" << ";";
|
|
|
|
ss << "Income" << ";";
|
2024-08-02 01:18:39 +02:00
|
|
|
ss << "MapVisitedRatio" << ";";
|
|
|
|
ss << "ObeliskVisited" << ";";
|
2024-08-02 19:37:46 +02:00
|
|
|
ss << "MightMagicRatio" << ";";
|
2024-08-02 19:38:33 +02:00
|
|
|
ss << "Score" << ";";
|
|
|
|
ss << "MaxHeroLevel" << ";";
|
|
|
|
ss << "NumBattlesNeutral" << ";";
|
|
|
|
ss << "NumBattlesPlayer" << ";";
|
|
|
|
ss << "NumWinBattlesNeutral" << ";";
|
2024-08-02 20:40:24 +02:00
|
|
|
ss << "NumWinBattlesPlayer" << ";";
|
|
|
|
ss << "NumHeroSurrendered" << ";";
|
|
|
|
ss << "NumHeroEscaped";
|
2024-08-01 21:30:53 +02:00
|
|
|
for(auto & resource : resources)
|
|
|
|
ss << ";" << GameConstants::RESOURCE_NAMES[resource];
|
2024-08-02 01:18:39 +02:00
|
|
|
for(auto & resource : resources)
|
|
|
|
ss << ";" << GameConstants::RESOURCE_NAMES[resource] + "Mines";
|
2024-08-01 21:30:53 +02:00
|
|
|
ss << "\r\n";
|
2024-07-27 02:11:26 +02:00
|
|
|
|
|
|
|
for(auto & entry : data)
|
|
|
|
{
|
2024-08-02 20:40:24 +02:00
|
|
|
ss << entry.map << ";";
|
2024-08-03 16:39:46 +02:00
|
|
|
ss << vstd::getFormattedDateTime(entry.timestamp, "%Y-%m-%dT%H:%M:%S") << ";";
|
2024-08-01 23:21:41 +02:00
|
|
|
ss << entry.day << ";";
|
|
|
|
ss << GameConstants::PLAYER_COLOR_NAMES[entry.player] << ";";
|
|
|
|
ss << entry.team.getNum() << ";";
|
|
|
|
ss << entry.isHuman << ";";
|
|
|
|
ss << (int)entry.status << ";";
|
|
|
|
ss << entry.numberHeroes << ";";
|
|
|
|
ss << entry.numberTowns << ";";
|
|
|
|
ss << entry.numberArtifacts << ";";
|
|
|
|
ss << entry.armyStrength << ";";
|
|
|
|
ss << entry.income << ";";
|
2024-08-02 01:18:39 +02:00
|
|
|
ss << entry.mapVisitedRatio << ";";
|
|
|
|
ss << entry.obeliskVisited << ";";
|
2024-08-02 19:37:46 +02:00
|
|
|
ss << entry.mightMagicRatio << ";";
|
2024-08-02 19:38:33 +02:00
|
|
|
ss << entry.score << ";";
|
|
|
|
ss << entry.maxHeroLevel << ";";
|
|
|
|
ss << entry.numBattlesNeutral << ";";
|
|
|
|
ss << entry.numBattlesPlayer << ";";
|
|
|
|
ss << entry.numWinBattlesNeutral << ";";
|
2024-08-02 20:40:24 +02:00
|
|
|
ss << entry.numWinBattlesPlayer << ";";
|
|
|
|
ss << entry.numHeroSurrendered << ";";
|
|
|
|
ss << entry.numHeroEscaped;
|
2024-08-01 21:30:53 +02:00
|
|
|
for(auto & resource : resources)
|
|
|
|
ss << ";" << entry.resources[resource];
|
2024-08-02 01:18:39 +02:00
|
|
|
for(auto & resource : resources)
|
|
|
|
ss << ";" << entry.numMines[resource];
|
2024-08-01 21:30:53 +02:00
|
|
|
ss << "\r\n";
|
2024-07-27 02:11:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return ss.str();
|
|
|
|
}
|
|
|
|
|
2024-08-02 01:18:39 +02:00
|
|
|
std::vector<const CGMine *> Statistic::getMines(const CGameState * gs, const PlayerState * ps)
|
|
|
|
{
|
|
|
|
std::vector<const CGMine *> tmp;
|
|
|
|
|
|
|
|
/// FIXME: Dirty dirty hack
|
|
|
|
/// Stats helper need some access to gamestate.
|
|
|
|
std::vector<const CGObjectInstance *> ownedObjects;
|
|
|
|
for(const CGObjectInstance * obj : gs->map->objects)
|
|
|
|
{
|
|
|
|
if(obj && obj->tempOwner == ps->color)
|
|
|
|
ownedObjects.push_back(obj);
|
|
|
|
}
|
|
|
|
/// This is code from CPlayerSpecificInfoCallback::getMyObjects
|
|
|
|
/// I'm really need to find out about callback interface design...
|
|
|
|
|
|
|
|
for(const auto * object : ownedObjects)
|
|
|
|
{
|
|
|
|
//Mines
|
|
|
|
if ( object->ID == Obj::MINE )
|
|
|
|
{
|
|
|
|
const auto * mine = dynamic_cast<const CGMine *>(object);
|
|
|
|
assert(mine);
|
|
|
|
|
|
|
|
tmp.push_back(mine);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
2024-08-01 22:36:32 +02:00
|
|
|
//calculates total number of artifacts that belong to given player
|
|
|
|
int Statistic::getNumberOfArts(const PlayerState * ps)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
for(auto h : ps->heroes)
|
|
|
|
{
|
|
|
|
ret += (int)h->artifactsInBackpack.size() + (int)h->artifactsWorn.size();
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get total strength of player army
|
2024-08-02 01:18:39 +02:00
|
|
|
si64 Statistic::getArmyStrength(const PlayerState * ps, bool withTownGarrison)
|
2024-08-01 22:36:32 +02:00
|
|
|
{
|
|
|
|
si64 str = 0;
|
|
|
|
|
|
|
|
for(auto h : ps->heroes)
|
|
|
|
{
|
2024-08-02 01:18:39 +02:00
|
|
|
if(!h->inTownGarrison || withTownGarrison) //original h3 behavior
|
2024-08-01 22:36:32 +02:00
|
|
|
str += h->getArmyStrength();
|
|
|
|
}
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get total gold income
|
2024-08-01 23:56:06 +02:00
|
|
|
int Statistic::getIncome(const CGameState * gs, const PlayerState * ps)
|
2024-08-01 22:36:32 +02:00
|
|
|
{
|
2024-08-01 23:56:06 +02:00
|
|
|
int percentIncome = gs->getStartInfo()->getIthPlayersSettings(ps->color).handicap.percentIncome;
|
2024-08-01 22:36:32 +02:00
|
|
|
int totalIncome = 0;
|
|
|
|
const CGObjectInstance * heroOrTown = nullptr;
|
|
|
|
|
|
|
|
//Heroes can produce gold as well - skill, specialty or arts
|
|
|
|
for(const auto & h : ps->heroes)
|
|
|
|
{
|
2024-08-01 23:56:06 +02:00
|
|
|
totalIncome += h->valOfBonuses(Selector::typeSubtype(BonusType::GENERATE_RESOURCE, BonusSubtypeID(GameResID(GameResID::GOLD)))) * percentIncome / 100;
|
2024-08-01 22:36:32 +02:00
|
|
|
|
|
|
|
if(!heroOrTown)
|
|
|
|
heroOrTown = h;
|
|
|
|
}
|
|
|
|
|
|
|
|
//Add town income of all towns
|
|
|
|
for(const auto & t : ps->towns)
|
|
|
|
{
|
|
|
|
totalIncome += t->dailyIncome()[EGameResID::GOLD];
|
|
|
|
|
|
|
|
if(!heroOrTown)
|
|
|
|
heroOrTown = t;
|
|
|
|
}
|
|
|
|
|
2024-08-02 01:18:39 +02:00
|
|
|
for(const CGMine * mine : getMines(gs, ps))
|
2024-08-01 22:36:32 +02:00
|
|
|
{
|
2024-08-02 01:18:39 +02:00
|
|
|
if (mine->producedResource == EGameResID::GOLD)
|
|
|
|
totalIncome += mine->getProducedQuantity();
|
2024-08-01 22:36:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return totalIncome;
|
|
|
|
}
|
|
|
|
|
2024-08-01 23:21:41 +02:00
|
|
|
double Statistic::getMapVisitedRatio(const CGameState * gs, PlayerColor player)
|
|
|
|
{
|
|
|
|
double visible = 0.0;
|
|
|
|
double numTiles = 0.0;
|
|
|
|
|
|
|
|
for(int layer = 0; layer < (gs->map->twoLevel ? 2 : 1); layer++)
|
2024-08-01 23:56:06 +02:00
|
|
|
for(int y = 0; y < gs->map->height; ++y)
|
|
|
|
for(int x = 0; x < gs->map->width; ++x)
|
2024-08-01 23:21:41 +02:00
|
|
|
{
|
|
|
|
TerrainTile tile = gs->map->getTile(int3(x, y, layer));
|
|
|
|
|
2024-08-01 23:56:06 +02:00
|
|
|
if(tile.blocked && (!tile.visitable))
|
2024-08-01 23:21:41 +02:00
|
|
|
continue;
|
|
|
|
|
|
|
|
if(gs->isVisible(int3(x, y, layer), player))
|
|
|
|
visible++;
|
|
|
|
numTiles++;
|
|
|
|
}
|
|
|
|
|
2024-08-01 23:56:06 +02:00
|
|
|
return visible / numTiles;
|
2024-08-01 23:21:41 +02:00
|
|
|
}
|
|
|
|
|
2024-08-02 19:38:33 +02:00
|
|
|
const CGHeroInstance * Statistic::findBestHero(const CGameState * gs, const PlayerColor & color)
|
2024-08-02 00:04:41 +02:00
|
|
|
{
|
2024-08-02 19:38:33 +02:00
|
|
|
auto &h = gs->players.at(color).heroes;
|
2024-08-02 00:04:41 +02:00
|
|
|
if(h.empty())
|
|
|
|
return nullptr;
|
|
|
|
//best hero will be that with highest exp
|
|
|
|
int best = 0;
|
|
|
|
for(int b=1; b<h.size(); ++b)
|
|
|
|
{
|
|
|
|
if(h[b]->exp > h[best]->exp)
|
|
|
|
{
|
|
|
|
best = b;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return h[best];
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::vector<PlayerColor>> Statistic::getRank(std::vector<TStat> stats)
|
|
|
|
{
|
|
|
|
std::sort(stats.begin(), stats.end(), [](const TStat & a, const TStat & b) { return a.second > b.second; });
|
|
|
|
|
|
|
|
//put first element
|
|
|
|
std::vector< std::vector<PlayerColor> > ret;
|
|
|
|
std::vector<PlayerColor> tmp;
|
|
|
|
tmp.push_back( stats[0].first );
|
|
|
|
ret.push_back( tmp );
|
|
|
|
|
|
|
|
//the rest of elements
|
|
|
|
for(int g=1; g<stats.size(); ++g)
|
|
|
|
{
|
|
|
|
if(stats[g].second == stats[g-1].second)
|
|
|
|
{
|
|
|
|
(ret.end()-1)->push_back( stats[g].first );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
//create next occupied rank
|
|
|
|
std::vector<PlayerColor> tmp;
|
|
|
|
tmp.push_back(stats[g].first);
|
|
|
|
ret.push_back(tmp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2024-08-02 01:18:39 +02:00
|
|
|
int Statistic::getObeliskVisited(const CGameState * gs, const TeamID & t)
|
|
|
|
{
|
|
|
|
if(gs->map->obelisksVisited.count(t))
|
|
|
|
return gs->map->obelisksVisited.at(t);
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
double Statistic::getMightMagicRatio(const PlayerState * ps)
|
|
|
|
{
|
|
|
|
double numMight = 0;
|
|
|
|
|
|
|
|
for(auto h : ps->heroes)
|
|
|
|
if(h->type->heroClass->affinity == CHeroClass::EClassAffinity::MIGHT)
|
|
|
|
numMight++;
|
|
|
|
|
|
|
|
return numMight / ps->heroes.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::map<EGameResID, int> Statistic::getNumMines(const CGameState * gs, const PlayerState * ps)
|
|
|
|
{
|
|
|
|
std::map<EGameResID, int> tmp;
|
|
|
|
|
|
|
|
for(auto & res : EGameResID::ALL_RESOURCES())
|
|
|
|
tmp[res] = 0;
|
|
|
|
|
|
|
|
for(const CGMine * mine : getMines(gs, ps))
|
|
|
|
tmp[mine->producedResource]++;
|
|
|
|
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
2024-07-27 02:11:26 +02:00
|
|
|
VCMI_LIB_NAMESPACE_END
|