/* * CStatisticScreen.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 "CStatisticScreen.h" #include "../CGameInfo.h" #include "../gui/CGuiHandler.h" #include "../gui/WindowHandler.h" #include "../eventsSDL/InputHandler.h" #include "../gui/Shortcut.h" #include "../render/Graphics.h" #include "../render/IImage.h" #include "../render/IRenderHandler.h" #include "../widgets/ComboBox.h" #include "../widgets/Images.h" #include "../widgets/GraphicalPrimitiveCanvas.h" #include "../widgets/TextControls.h" #include "../widgets/Buttons.h" #include "../windows/InfoWindows.h" #include "../widgets/Slider.h" #include "../../lib/gameState/GameStatistics.h" #include "../../lib/gameState/CGameState.h" #include "../../lib/texts/CGeneralTextHandler.h" #include "../../lib/texts/TextOperations.h" #include std::string CStatisticScreen::getDay(int d) { return std::to_string(CGameState::getDate(d, Date::MONTH)) + "/" + std::to_string(CGameState::getDate(d, Date::WEEK)) + "/" + std::to_string(CGameState::getDate(d, Date::DAY_OF_WEEK)); } CStatisticScreen::CStatisticScreen(const StatisticDataSet & stat) : CWindowObject(BORDERED), statistic(stat) { OBJECT_CONSTRUCTION; pos = center(Rect(0, 0, 800, 600)); filledBackground = std::make_shared(Rect(0, 0, pos.w, pos.h)); filledBackground->setPlayerColor(PlayerColor(1)); contentArea = Rect(10, 40, 780, 510); layout.emplace_back(std::make_shared(400, 20, FONT_BIG, ETextAlignment::CENTER, Colors::YELLOW, CGI->generaltexth->translate("vcmi.statisticWindow.statistics"))); layout.emplace_back(std::make_shared(contentArea, ColorRGBA(0, 0, 0, 128), ColorRGBA(64, 80, 128, 255), 1)); layout.emplace_back(std::make_shared(Point(725, 558), AnimationPath::builtin("MUBCHCK"), CButton::tooltip(), [this](){ close(); }, EShortcut::GLOBAL_ACCEPT)); buttonSelect = std::make_shared(Point(10, 564), AnimationPath::builtin("GSPBUT2"), CButton::tooltip(), [this](bool on){ onSelectButton(); }); buttonSelect->setTextOverlay(CGI->generaltexth->translate("vcmi.statisticWindow.selectView"), EFonts::FONT_SMALL, Colors::YELLOW); buttonCsvSave = std::make_shared(Point(150, 564), AnimationPath::builtin("GSPBUT2"), CButton::tooltip(), [this](bool on){ GH.input().copyToClipBoard(statistic.toCsv("\t")); }); buttonCsvSave->setTextOverlay(CGI->generaltexth->translate("vcmi.statisticWindow.tsvCopy"), EFonts::FONT_SMALL, Colors::YELLOW); mainContent = getContent(OVERVIEW, EGameResID::NONE); } void CStatisticScreen::onSelectButton() { std::vector texts; for(auto & val : contentInfo) texts.emplace_back(CGI->generaltexth->translate(std::get<0>(val.second))); GH.windows().createAndPushWindow(texts, [this](int selectedIndex) { OBJECT_CONSTRUCTION; if(!std::get<1>(contentInfo[static_cast(selectedIndex)])) mainContent = getContent(static_cast(selectedIndex), EGameResID::NONE); else { auto content = static_cast(selectedIndex); auto possibleRes = std::vector{EGameResID::GOLD, EGameResID::WOOD, EGameResID::MERCURY, EGameResID::ORE, EGameResID::SULFUR, EGameResID::CRYSTAL, EGameResID::GEMS}; std::vector resourceText; for(const auto & res : possibleRes) resourceText.emplace_back(CGI->generaltexth->translate(TextIdentifier("core.restypes", res.getNum()).get())); GH.windows().createAndPushWindow(resourceText, [this, content, possibleRes](int index) { OBJECT_CONSTRUCTION; mainContent = getContent(content, possibleRes[index]); }); } }); } TData CStatisticScreen::extractData(const StatisticDataSet & stat, const ExtractFunctor & selector) const { auto tmpData = stat.data; std::sort(tmpData.begin(), tmpData.end(), [](const StatisticDataSetEntry & v1, const StatisticDataSetEntry & v2){ return v1.player == v2.player ? v1.day < v2.day : v1.player < v2.player; }); PlayerColor tmpColor = PlayerColor::NEUTRAL; std::vector tmpColorSet; TData plotData; EPlayerStatus statusLastRound = EPlayerStatus::INGAME; for(const auto & val : tmpData) { if(tmpColor != val.player) { if(tmpColorSet.size()) { plotData.push_back({graphics->playerColors[tmpColor.getNum()], std::vector(tmpColorSet)}); tmpColorSet.clear(); } tmpColor = val.player; } if(val.status == EPlayerStatus::INGAME || (statusLastRound == EPlayerStatus::INGAME && val.status == EPlayerStatus::LOSER)) tmpColorSet.emplace_back(selector(val)); statusLastRound = val.status; //to keep at least one dataset after loose } if(tmpColorSet.size()) plotData.push_back({graphics->playerColors[tmpColor.getNum()], std::vector(tmpColorSet)}); return plotData; } TIcons CStatisticScreen::extractIcons() const { TIcons icons; auto tmpData = statistic.data; std::sort(tmpData.begin(), tmpData.end(), [](const StatisticDataSetEntry & v1, const StatisticDataSetEntry & v2){ return v1.player == v2.player ? v1.day < v2.day : v1.player < v2.player; }); auto imageTown = GH.renderHandler().loadImage(AnimationPath::builtin("cradvntr"), 3, 0, EImageBlitMode::COLORKEY); auto imageBattle = GH.renderHandler().loadImage(AnimationPath::builtin("cradvntr"), 5, 0, EImageBlitMode::COLORKEY); auto imageDefeated = GH.renderHandler().loadImage(AnimationPath::builtin("crcombat"), 0, 0, EImageBlitMode::COLORKEY); auto imageGrail = GH.renderHandler().loadImage(AnimationPath::builtin("vwsymbol"), 2, 0, EImageBlitMode::COLORKEY); std::map foundDefeated; std::map foundGrail; for(const auto & val : tmpData) { if(val.eventCapturedTown) icons.push_back({ graphics->playerColors[val.player], val.day, imageTown, CGI->generaltexth->translate("vcmi.statisticWindow.icon.townCaptured") }); if(val.eventDefeatedStrongestHero) icons.push_back({ graphics->playerColors[val.player], val.day, imageBattle, CGI->generaltexth->translate("vcmi.statisticWindow.icon.strongestHeroDefeated") }); if(val.status == EPlayerStatus::LOSER && !foundDefeated[val.player]) { foundDefeated[val.player] = true; icons.push_back({ graphics->playerColors[val.player], val.day, imageDefeated, CGI->generaltexth->translate("vcmi.statisticWindow.icon.defeated") }); } if(val.hasGrail && !foundGrail[val.player]) { foundGrail[val.player] = true; icons.push_back({ graphics->playerColors[val.player], val.day, imageGrail, CGI->generaltexth->translate("vcmi.statisticWindow.icon.grailFound") }); } } return icons; } std::shared_ptr CStatisticScreen::getContent(Content c, EGameResID res) { TData plotData; TIcons icons = extractIcons(); switch (c) { case OVERVIEW: return std::make_shared(contentArea.resize(-15), CGI->generaltexth->translate(std::get<0>(contentInfo[c])), statistic); case CHART_RESOURCES: plotData = extractData(statistic, [res](const StatisticDataSetEntry & val) -> float { return val.resources[res]; }); return std::make_shared(contentArea.resize(-5), CGI->generaltexth->translate(std::get<0>(contentInfo[c])) + " - " + CGI->generaltexth->translate(TextIdentifier("core.restypes", res.getNum()).get()), plotData, icons, 0); case CHART_INCOME: plotData = extractData(statistic, [](const StatisticDataSetEntry & val) -> float { return val.income; }); return std::make_shared(contentArea.resize(-5), CGI->generaltexth->translate(std::get<0>(contentInfo[c])), plotData, icons, 0); case CHART_NUMBER_OF_HEROES: plotData = extractData(statistic, [](const StatisticDataSetEntry & val) -> float { return val.numberHeroes; }); return std::make_shared(contentArea.resize(-5), CGI->generaltexth->translate(std::get<0>(contentInfo[c])), plotData, icons, 0); case CHART_NUMBER_OF_TOWNS: plotData = extractData(statistic, [](const StatisticDataSetEntry & val) -> float { return val.numberTowns; }); return std::make_shared(contentArea.resize(-5), CGI->generaltexth->translate(std::get<0>(contentInfo[c])), plotData, icons, 0); case CHART_NUMBER_OF_ARTIFACTS: plotData = extractData(statistic, [](const StatisticDataSetEntry & val) -> float { return val.numberArtifacts; }); return std::make_shared(contentArea.resize(-5), CGI->generaltexth->translate(std::get<0>(contentInfo[c])), plotData, icons, 0); case CHART_NUMBER_OF_DWELLINGS: plotData = extractData(statistic, [](const StatisticDataSetEntry & val) -> float { return val.numberDwellings; }); return std::make_shared(contentArea.resize(-5), CGI->generaltexth->translate(std::get<0>(contentInfo[c])), plotData, icons, 0); case CHART_NUMBER_OF_MINES: plotData = extractData(statistic, [res](StatisticDataSetEntry val) -> float { return val.numMines[res]; }); return std::make_shared(contentArea.resize(-5), CGI->generaltexth->translate(std::get<0>(contentInfo[c])) + " - " + CGI->generaltexth->translate(TextIdentifier("core.restypes", res.getNum()).get()), plotData, icons, 0); case CHART_ARMY_STRENGTH: plotData = extractData(statistic, [](const StatisticDataSetEntry & val) -> float { return val.armyStrength; }); return std::make_shared(contentArea.resize(-5), CGI->generaltexth->translate(std::get<0>(contentInfo[c])), plotData, icons, 0); case CHART_EXPERIENCE: plotData = extractData(statistic, [](const StatisticDataSetEntry & val) -> float { return val.totalExperience; }); return std::make_shared(contentArea.resize(-5), CGI->generaltexth->translate(std::get<0>(contentInfo[c])), plotData, icons, 0); case CHART_RESOURCES_SPENT_ARMY: plotData = extractData(statistic, [res](const StatisticDataSetEntry & val) -> float { return val.spentResourcesForArmy[res]; }); return std::make_shared(contentArea.resize(-5), CGI->generaltexth->translate(std::get<0>(contentInfo[c])) + " - " + CGI->generaltexth->translate(TextIdentifier("core.restypes", res.getNum()).get()), plotData, icons, 0); case CHART_RESOURCES_SPENT_BUILDINGS: plotData = extractData(statistic, [res](const StatisticDataSetEntry & val) -> float { return val.spentResourcesForBuildings[res]; }); return std::make_shared(contentArea.resize(-5), CGI->generaltexth->translate(std::get<0>(contentInfo[c])) + " - " + CGI->generaltexth->translate(TextIdentifier("core.restypes", res.getNum()).get()), plotData, icons, 0); case CHART_MAP_EXPLORED: plotData = extractData(statistic, [](const StatisticDataSetEntry & val) -> float { return val.mapExploredRatio; }); return std::make_shared(contentArea.resize(-5), CGI->generaltexth->translate(std::get<0>(contentInfo[c])), plotData, icons, 1); } return nullptr; } StatisticSelector::StatisticSelector(const std::vector & texts, const std::function & cb) : CWindowObject(BORDERED | NEEDS_ANIMATED_BACKGROUND), texts(texts), cb(cb) { OBJECT_CONSTRUCTION; pos = center(Rect(0, 0, 128 + 16, std::min(static_cast(texts.size()), LINES) * 40)); filledBackground = std::make_shared(Rect(0, 0, pos.w, pos.h)); filledBackground->setPlayerColor(PlayerColor(1)); slider = std::make_shared(Point(pos.w - 16, 0), pos.h, [this](int to){ update(to); redraw(); }, LINES, texts.size(), 0, Orientation::VERTICAL, CSlider::BLUE); slider->setPanningStep(40); slider->setScrollBounds(Rect(-pos.w + slider->pos.w, 0, pos.w, pos.h)); update(0); } void StatisticSelector::update(int to) { OBJECT_CONSTRUCTION; buttons.clear(); for(int i = to; i < LINES + to; i++) { if(i>=texts.size()) continue; auto button = std::make_shared(Point(0, 10 + (i - to) * 40), AnimationPath::builtin("GSPBUT2"), CButton::tooltip(), [this, i](bool on){ close(); cb(i); }); button->setTextOverlay(texts[i], EFonts::FONT_SMALL, Colors::WHITE); buttons.emplace_back(button); } } OverviewPanel::OverviewPanel(Rect position, std::string title, const StatisticDataSet & stat) : CIntObject(), data(stat) { OBJECT_CONSTRUCTION; pos = position + pos.topLeft(); layout.emplace_back(std::make_shared(pos.w / 2, 10, FONT_MEDIUM, ETextAlignment::CENTER, Colors::WHITE, title)); canvas = std::make_shared(Rect(0, Y_OFFS, pos.w - 16, pos.h - Y_OFFS)); dataExtract = { { CGI->generaltexth->translate("vcmi.statisticWindow.param.playerName"), [this](PlayerColor color){ return playerDataFilter(color).front().playerName; } }, { CGI->generaltexth->translate("vcmi.statisticWindow.param.daysSurvived"), [this](PlayerColor color){ auto playerData = playerDataFilter(color); for(int i = 0; i < playerData.size(); i++) if(playerData[i].status == EPlayerStatus::LOSER) return CStatisticScreen::getDay(i + 1); return CStatisticScreen::getDay(playerData.size()); } }, { CGI->generaltexth->translate("vcmi.statisticWindow.param.maxHeroLevel"), [this](PlayerColor color){ int maxLevel = 0; for(const auto & val : playerDataFilter(color)) if(maxLevel < val.maxHeroLevel) maxLevel = val.maxHeroLevel; return std::to_string(maxLevel); } }, { CGI->generaltexth->translate("vcmi.statisticWindow.param.battleWinRatioHero"), [this](PlayerColor color){ auto val = playerDataFilter(color).back(); if(!val.numBattlesPlayer) return std::string(""); float tmp = (static_cast(val.numWinBattlesPlayer) / static_cast(val.numBattlesPlayer)) * 100; return std::to_string(static_cast(tmp)) + " %"; } }, { CGI->generaltexth->translate("vcmi.statisticWindow.param.battleWinRatioNeutral"), [this](PlayerColor color){ auto val = playerDataFilter(color).back(); if(!val.numWinBattlesNeutral) return std::string(""); float tmp = (static_cast(val.numWinBattlesNeutral) / static_cast(val.numBattlesNeutral)) * 100; return std::to_string(static_cast(tmp)) + " %"; } }, { CGI->generaltexth->translate("vcmi.statisticWindow.param.battlesHero"), [this](PlayerColor color){ auto val = playerDataFilter(color).back(); return std::to_string(val.numBattlesPlayer); } }, { CGI->generaltexth->translate("vcmi.statisticWindow.param.battlesNeutral"), [this](PlayerColor color){ auto val = playerDataFilter(color).back(); return std::to_string(val.numBattlesNeutral); } }, { CGI->generaltexth->translate("vcmi.statisticWindow.param.obeliskVisited"), [this](PlayerColor color){ auto val = playerDataFilter(color).back(); return std::to_string(static_cast(val.obeliskVisitedRatio * 100)) + " %"; } }, { CGI->generaltexth->translate("vcmi.statisticWindow.param.maxArmyStrength"), [this](PlayerColor color){ int maxArmyStrength = 0; for(const auto & val : playerDataFilter(color)) if(maxArmyStrength < val.armyStrength) maxArmyStrength = val.armyStrength; return TextOperations::formatMetric(maxArmyStrength, 6); } }, { CGI->generaltexth->translate("vcmi.statisticWindow.param.tradeVolume") + " - " + CGI->generaltexth->translate(TextIdentifier("core.restypes", EGameResID::GOLD).get()), [this](PlayerColor color){ auto val = playerDataFilter(color).back(); return std::to_string(val.tradeVolume[EGameResID::GOLD]); } }, { CGI->generaltexth->translate("vcmi.statisticWindow.param.tradeVolume") + " - " + CGI->generaltexth->translate(TextIdentifier("core.restypes", EGameResID::WOOD).get()), [this](PlayerColor color){ auto val = playerDataFilter(color).back(); return std::to_string(val.tradeVolume[EGameResID::WOOD]); } }, { CGI->generaltexth->translate("vcmi.statisticWindow.param.tradeVolume") + " - " + CGI->generaltexth->translate(TextIdentifier("core.restypes", EGameResID::MERCURY).get()), [this](PlayerColor color){ auto val = playerDataFilter(color).back(); return std::to_string(val.tradeVolume[EGameResID::MERCURY]); } }, { CGI->generaltexth->translate("vcmi.statisticWindow.param.tradeVolume") + " - " + CGI->generaltexth->translate(TextIdentifier("core.restypes", EGameResID::ORE).get()), [this](PlayerColor color){ auto val = playerDataFilter(color).back(); return std::to_string(val.tradeVolume[EGameResID::ORE]); } }, { CGI->generaltexth->translate("vcmi.statisticWindow.param.tradeVolume") + " - " + CGI->generaltexth->translate(TextIdentifier("core.restypes", EGameResID::SULFUR).get()), [this](PlayerColor color){ auto val = playerDataFilter(color).back(); return std::to_string(val.tradeVolume[EGameResID::SULFUR]); } }, { CGI->generaltexth->translate("vcmi.statisticWindow.param.tradeVolume") + " - " + CGI->generaltexth->translate(TextIdentifier("core.restypes", EGameResID::CRYSTAL).get()), [this](PlayerColor color){ auto val = playerDataFilter(color).back(); return std::to_string(val.tradeVolume[EGameResID::CRYSTAL]); } }, { CGI->generaltexth->translate("vcmi.statisticWindow.param.tradeVolume") + " - " + CGI->generaltexth->translate(TextIdentifier("core.restypes", EGameResID::GEMS).get()), [this](PlayerColor color){ auto val = playerDataFilter(color).back(); return std::to_string(val.tradeVolume[EGameResID::GEMS]); } }, }; int usedLines = dataExtract.size(); slider = std::make_shared(Point(pos.w - 16, Y_OFFS), pos.h - Y_OFFS, [this](int to){ update(to); setRedrawParent(true); redraw(); }, LINES - 1, usedLines, 0, Orientation::VERTICAL, CSlider::BLUE); slider->setPanningStep(canvas->pos.h / LINES); slider->setScrollBounds(Rect(-pos.w + slider->pos.w, 0, pos.w, canvas->pos.h)); fieldSize = Point(canvas->pos.w / (graphics->playerColors.size() + 2), canvas->pos.h / LINES); for(int x = 0; x < graphics->playerColors.size() + 1; x++) for(int y = 0; y < LINES; y++) { int xStart = (x + (x == 0 ? 0 : 1)) * fieldSize.x; int yStart = y * fieldSize.y; if(x == 0 || y == 0) canvas->addBox(Point(xStart, yStart), Point(x == 0 ? 2 * fieldSize.x : fieldSize.x, fieldSize.y), ColorRGBA(0, 0, 0, 100)); canvas->addRectangle(Point(xStart, yStart), Point(x == 0 ? 2 * fieldSize.x : fieldSize.x, fieldSize.y), ColorRGBA(127, 127, 127, 255)); } update(0); } std::vector OverviewPanel::playerDataFilter(PlayerColor color) { std::vector tmpData; std::copy_if(data.data.begin(), data.data.end(), std::back_inserter(tmpData), [color](const StatisticDataSetEntry & e){ return e.player == color; }); return tmpData; } void OverviewPanel::update(int to) { OBJECT_CONSTRUCTION; content.clear(); for(int y = to; y < LINES - 1 + to; y++) { if(y >= dataExtract.size()) continue; for(int x = 0; x < PlayerColor::PLAYER_LIMIT_I + 1; x++) { if(y == to && x < PlayerColor::PLAYER_LIMIT_I) content.emplace_back(std::make_shared(AnimationPath::builtin("ITGFLAGS"), x, 0, 180 + x * fieldSize.x, 35)); int xStart = (x + (x == 0 ? 0 : 1)) * fieldSize.x + (x == 0 ? fieldSize.x : (fieldSize.x / 2)); int yStart = Y_OFFS + (y + 1 - to) * fieldSize.y + (fieldSize.y / 2); PlayerColor tmpColor(x - 1); if(playerDataFilter(tmpColor).size() || x == 0) content.emplace_back(std::make_shared(xStart, yStart, FONT_TINY, ETextAlignment::CENTER, Colors::WHITE, (x == 0 ? dataExtract[y].first : dataExtract[y].second(tmpColor)), x == 0 ? (fieldSize.x * 2) : fieldSize.x)); } } } int computeGridStep(int maxAmount, int linesLimit) { for (int lineInterval = 1;;lineInterval *= 10) { for (int factor : { 1, 2, 5 } ) { int lineIntervalToTest = lineInterval * factor; if (maxAmount / lineIntervalToTest <= linesLimit) return lineIntervalToTest; } } } LineChart::LineChart(Rect position, std::string title, TData data, TIcons icons, float maxY) : CIntObject(), maxVal(0), maxDay(0) { OBJECT_CONSTRUCTION; addUsedEvents(LCLICK | MOVE | GESTURE); pos = position + pos.topLeft(); layout.emplace_back(std::make_shared(pos.w / 2, 20, FONT_MEDIUM, ETextAlignment::CENTER, Colors::WHITE, title)); chartArea = pos.resize(-50); chartArea.moveTo(Point(50, 50)); canvas = std::make_shared(Rect(0, 0, pos.w, pos.h)); statusBar = CGStatusBar::create(0, 0, ImagePath::builtin("radialMenu/statusBar")); static_cast>(statusBar)->setEnabled(false); // additional calculations bool skipMaxValCalc = maxY > 0; maxVal = maxY; for(const auto & line : data) { for(auto & val : line.second) if(maxVal < val && !skipMaxValCalc) maxVal = val; if(maxDay < line.second.size()) maxDay = line.second.size(); } //calculate nice maxVal int gridLineCount = 10; int gridStep = computeGridStep(maxVal, gridLineCount); niceMaxVal = gridStep * std::ceil(maxVal / gridStep); niceMaxVal = std::max(1, niceMaxVal); // avoid zero size Y axis (if all values are 0) // calculate points in chart auto getPoint = [this](int i, std::vector data){ float x = (static_cast(chartArea.w) / static_cast(maxDay - 1)) * static_cast(i); float y = static_cast(chartArea.h) - (static_cast(chartArea.h) / niceMaxVal) * data[i]; return Point(x, y); }; // draw grid (vertical lines) int dayGridInterval = maxDay < 700 ? 7 : 28; for(const auto & line : data) { for(int i = 0; i < line.second.size(); i += dayGridInterval) { Point p = getPoint(i, line.second) + chartArea.topLeft(); canvas->addLine(Point(p.x, chartArea.topLeft().y), Point(p.x, chartArea.topLeft().y + chartArea.h), ColorRGBA(70, 70, 70)); } } // draw grid (horizontal lines) if(maxVal > 0) { int gridStepPx = int((static_cast(chartArea.h) / niceMaxVal) * gridStep); for(int i = 0; i < std::ceil(maxVal / gridStep) + 1; i++) { canvas->addLine(chartArea.topLeft() + Point(0, chartArea.h - gridStepPx * i), chartArea.topLeft() + Point(chartArea.w, chartArea.h - gridStepPx * i), ColorRGBA(70, 70, 70)); layout.emplace_back(std::make_shared(chartArea.topLeft().x - 5, chartArea.topLeft().y + 10 + chartArea.h - gridStepPx * i, FONT_SMALL, ETextAlignment::CENTERRIGHT, Colors::WHITE, TextOperations::formatMetric(i * gridStep, 5))); } } // draw for(const auto & line : data) { Point lastPoint(-1, -1); for(int i = 0; i < line.second.size(); i++) { Point p = getPoint(i, line.second) + chartArea.topLeft(); if(lastPoint.x != -1) canvas->addLine(lastPoint, p, line.first); // icons for(auto & icon : icons) if(std::get<0>(icon) == line.first && std::get<1>(icon) == i + 1) // color && day { pictures.emplace_back(std::make_shared(std::get<2>(icon), Point(p.x - (std::get<2>(icon)->width() / 2), p.y - (std::get<2>(icon)->height() / 2)))); pictures.back()->addRClickCallback([icon](){ CRClickPopup::createAndPush(std::get<3>(icon)); }); } lastPoint = p; } } // Axis canvas->addLine(chartArea.topLeft() + Point(0, -10), chartArea.topLeft() + Point(0, chartArea.h + 10), Colors::WHITE); canvas->addLine(chartArea.topLeft() + Point(-10, chartArea.h), chartArea.topLeft() + Point(chartArea.w + 10, chartArea.h), Colors::WHITE); Point p = chartArea.topLeft() + Point(chartArea.w + 10, chartArea.h + 10); layout.emplace_back(std::make_shared(p.x, p.y, FONT_SMALL, ETextAlignment::CENTER, Colors::WHITE, CStatisticScreen::getDay(maxDay))); p = chartArea.bottomLeft() + Point(chartArea.w / 2, + 20); layout.emplace_back(std::make_shared(p.x, p.y, FONT_MEDIUM, ETextAlignment::CENTER, Colors::WHITE, CGI->generaltexth->translate("core.genrltxt.64"))); } void LineChart::updateStatusBar(const Point & cursorPosition) { statusBar->moveTo(cursorPosition + Point(-statusBar->pos.w / 2, 20)); statusBar->fitToRect(pos, 10); Rect r(pos.x + chartArea.x, pos.y + chartArea.y, chartArea.w, chartArea.h); statusBar->setEnabled(r.isInside(cursorPosition)); if(r.isInside(cursorPosition)) { float x = (static_cast(maxDay - 1) / static_cast(chartArea.w)) * (static_cast(cursorPosition.x) - static_cast(r.x)) + 1.0f; float y = niceMaxVal - (niceMaxVal / static_cast(chartArea.h)) * (static_cast(cursorPosition.y) - static_cast(r.y)); statusBar->write(CGI->generaltexth->translate("core.genrltxt.64") + ": " + CStatisticScreen::getDay(x) + " " + CGI->generaltexth->translate("vcmi.statisticWindow.value") + ": " + (static_cast(y) > 0 ? std::to_string(static_cast(y)) : std::to_string(y))); } setRedrawParent(true); redraw(); } void LineChart::mouseMoved(const Point & cursorPosition, const Point & lastUpdateDistance) { updateStatusBar(cursorPosition); } void LineChart::gesturePanning(const Point & initialPosition, const Point & currentPosition, const Point & lastUpdateDistance) { updateStatusBar(currentPosition); }