2023-08-13 12:06:35 +02:00
|
|
|
/*
|
|
|
|
* TurnTimerWidget.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
|
|
|
|
*
|
|
|
|
*/
|
2023-08-13 21:50:40 +02:00
|
|
|
#include "StdInc.h"
|
2023-08-13 12:06:35 +02:00
|
|
|
#include "TurnTimerWidget.h"
|
|
|
|
|
2023-08-16 23:10:03 +02:00
|
|
|
#include "../CGameInfo.h"
|
|
|
|
#include "../CMusicHandler.h"
|
2023-08-13 12:06:35 +02:00
|
|
|
#include "../CPlayerInterface.h"
|
2023-08-26 03:58:22 +02:00
|
|
|
#include "../battle/BattleInterface.h"
|
|
|
|
#include "../battle/BattleStacksController.h"
|
2023-08-13 23:13:37 +02:00
|
|
|
|
2023-08-13 12:06:35 +02:00
|
|
|
#include "../render/EFont.h"
|
2023-08-19 17:45:25 +02:00
|
|
|
#include "../render/Graphics.h"
|
2023-08-13 12:06:35 +02:00
|
|
|
#include "../gui/CGuiHandler.h"
|
|
|
|
#include "../gui/TextAlignment.h"
|
|
|
|
#include "../widgets/Images.h"
|
|
|
|
#include "../widgets/TextControls.h"
|
|
|
|
#include "../../CCallback.h"
|
2023-08-26 03:58:22 +02:00
|
|
|
#include "../../lib/CStack.h"
|
2023-08-13 12:06:35 +02:00
|
|
|
#include "../../lib/CPlayerState.h"
|
2023-08-23 14:07:50 +02:00
|
|
|
#include "../../lib/filesystem/ResourcePath.h"
|
2023-08-13 23:13:37 +02:00
|
|
|
|
|
|
|
TurnTimerWidget::DrawRect::DrawRect(const Rect & r, const ColorRGBA & c):
|
|
|
|
CIntObject(), rect(r), color(c)
|
|
|
|
{
|
|
|
|
}
|
2023-08-13 12:06:35 +02:00
|
|
|
|
2023-08-13 23:13:37 +02:00
|
|
|
void TurnTimerWidget::DrawRect::showAll(Canvas & to)
|
|
|
|
{
|
|
|
|
to.drawColor(rect, color);
|
|
|
|
|
|
|
|
CIntObject::showAll(to);
|
|
|
|
}
|
2023-08-13 12:06:35 +02:00
|
|
|
|
|
|
|
TurnTimerWidget::TurnTimerWidget():
|
2023-08-13 23:13:37 +02:00
|
|
|
InterfaceObjectConfigurable(TIME),
|
2023-08-26 04:15:28 +02:00
|
|
|
turnTime(0), lastTurnTime(0), cachedTurnTime(0), lastPlayer(PlayerColor::CANNOT_DETERMINE)
|
2023-08-13 12:06:35 +02:00
|
|
|
{
|
2023-08-13 23:13:37 +02:00
|
|
|
REGISTER_BUILDER("drawRect", &TurnTimerWidget::buildDrawRect);
|
2023-08-13 12:06:35 +02:00
|
|
|
|
|
|
|
recActions &= ~DEACTIVATE;
|
2023-08-13 23:13:37 +02:00
|
|
|
|
2023-09-01 23:26:14 +02:00
|
|
|
const JsonNode config(JsonPath::builtin("config/widgets/turnTimer.json"));
|
2023-08-13 23:13:37 +02:00
|
|
|
|
|
|
|
build(config);
|
2023-08-16 23:10:03 +02:00
|
|
|
|
|
|
|
std::transform(variables["notificationTime"].Vector().begin(),
|
|
|
|
variables["notificationTime"].Vector().end(),
|
|
|
|
std::inserter(notifications, notifications.begin()),
|
|
|
|
[](const JsonNode & node){ return node.Integer(); });
|
2023-08-13 12:06:35 +02:00
|
|
|
}
|
|
|
|
|
2023-08-13 23:13:37 +02:00
|
|
|
std::shared_ptr<TurnTimerWidget::DrawRect> TurnTimerWidget::buildDrawRect(const JsonNode & config) const
|
2023-08-13 12:06:35 +02:00
|
|
|
{
|
2023-08-13 23:13:37 +02:00
|
|
|
logGlobal->debug("Building widget TurnTimerWidget::DrawRect");
|
|
|
|
auto rect = readRect(config["rect"]);
|
|
|
|
auto color = readColor(config["color"]);
|
|
|
|
return std::make_shared<TurnTimerWidget::DrawRect>(rect, color);
|
2023-08-13 12:06:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void TurnTimerWidget::show(Canvas & to)
|
|
|
|
{
|
|
|
|
showAll(to);
|
|
|
|
}
|
|
|
|
|
2023-08-25 00:08:48 +02:00
|
|
|
void TurnTimerWidget::setTime(PlayerColor player, int time)
|
2023-08-13 12:06:35 +02:00
|
|
|
{
|
2023-08-16 23:10:03 +02:00
|
|
|
int newTime = time / 1000;
|
2023-08-26 04:15:28 +02:00
|
|
|
if(player == LOCPLINT->playerID
|
|
|
|
&& newTime != turnTime
|
2023-08-20 02:06:52 +02:00
|
|
|
&& notifications.count(newTime))
|
2023-08-25 00:08:48 +02:00
|
|
|
{
|
2023-09-04 12:03:15 +02:00
|
|
|
CCS->soundh->playSound(AudioPath::fromJson(variables["notificationSound"]));
|
2023-08-25 00:08:48 +02:00
|
|
|
}
|
|
|
|
|
2023-08-16 23:10:03 +02:00
|
|
|
turnTime = newTime;
|
2023-08-25 00:08:48 +02:00
|
|
|
|
2023-08-13 23:13:37 +02:00
|
|
|
if(auto w = widget<CLabel>("timer"))
|
|
|
|
{
|
|
|
|
std::ostringstream oss;
|
|
|
|
oss << turnTime / 60 << ":" << std::setw(2) << std::setfill('0') << turnTime % 60;
|
|
|
|
w->setText(oss.str());
|
2023-08-19 17:45:25 +02:00
|
|
|
|
|
|
|
if(graphics && LOCPLINT && LOCPLINT->cb
|
|
|
|
&& variables["textColorFromPlayerColor"].Bool()
|
2023-08-25 00:08:48 +02:00
|
|
|
&& player.isValidPlayer())
|
2023-08-19 17:45:25 +02:00
|
|
|
{
|
2023-08-25 00:08:48 +02:00
|
|
|
w->setColor(graphics->playerColors[player]);
|
2023-08-19 17:45:25 +02:00
|
|
|
}
|
2023-08-13 23:13:37 +02:00
|
|
|
}
|
2023-08-13 12:06:35 +02:00
|
|
|
}
|
|
|
|
|
2023-08-30 01:33:59 +02:00
|
|
|
void TurnTimerWidget::updateTimer(PlayerColor player, uint32_t msPassed)
|
|
|
|
{
|
|
|
|
const auto & time = LOCPLINT->cb->getPlayerTurnTime(player);
|
|
|
|
if(time.isActive)
|
|
|
|
cachedTurnTime -= msPassed;
|
|
|
|
|
|
|
|
if(cachedTurnTime < 0)
|
|
|
|
cachedTurnTime = 0; //do not go below zero
|
|
|
|
|
|
|
|
if(lastPlayer != player)
|
|
|
|
{
|
|
|
|
lastPlayer = player;
|
|
|
|
lastTurnTime = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto timeCheckAndUpdate = [&](int time)
|
|
|
|
{
|
|
|
|
if(time / 1000 != lastTurnTime / 1000)
|
|
|
|
{
|
|
|
|
//do not update timer on this tick
|
|
|
|
lastTurnTime = time;
|
|
|
|
cachedTurnTime = time;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
setTime(player, cachedTurnTime);
|
|
|
|
};
|
|
|
|
|
|
|
|
auto * playerInfo = LOCPLINT->cb->getPlayer(player);
|
|
|
|
if(player.isValidPlayer() || (playerInfo && playerInfo->isHuman()))
|
|
|
|
{
|
|
|
|
if(time.isBattle)
|
|
|
|
timeCheckAndUpdate(time.creatureTimer);
|
|
|
|
else
|
|
|
|
timeCheckAndUpdate(time.turnTimer);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
timeCheckAndUpdate(0);
|
|
|
|
}
|
|
|
|
|
2023-08-13 12:06:35 +02:00
|
|
|
void TurnTimerWidget::tick(uint32_t msPassed)
|
|
|
|
{
|
2023-08-25 00:08:48 +02:00
|
|
|
if(!LOCPLINT || !LOCPLINT->cb)
|
|
|
|
return;
|
|
|
|
|
2023-08-30 01:33:59 +02:00
|
|
|
if(LOCPLINT->battleInt)
|
2023-08-13 12:06:35 +02:00
|
|
|
{
|
2023-08-30 01:33:59 +02:00
|
|
|
if(auto * stack = LOCPLINT->battleInt->stacksController->getActiveStack())
|
|
|
|
updateTimer(stack->getOwner(), msPassed);
|
|
|
|
else
|
|
|
|
updateTimer(PlayerColor::NEUTRAL, msPassed);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-10-06 17:27:12 +02:00
|
|
|
if(LOCPLINT->makingTurn)
|
|
|
|
updateTimer(LOCPLINT->playerID, msPassed);
|
|
|
|
else
|
2023-08-16 23:10:03 +02:00
|
|
|
{
|
2023-10-06 17:27:12 +02:00
|
|
|
for(PlayerColor p(0); p < PlayerColor::PLAYER_LIMIT; ++p)
|
|
|
|
{
|
|
|
|
if(LOCPLINT->cb->isPlayerMakingTurn(p))
|
2023-10-06 17:56:46 +02:00
|
|
|
{
|
2023-10-06 17:27:12 +02:00
|
|
|
updateTimer(p, msPassed);
|
2023-10-06 17:56:46 +02:00
|
|
|
break;
|
|
|
|
}
|
2023-10-06 17:27:12 +02:00
|
|
|
}
|
2023-08-14 21:31:44 +02:00
|
|
|
}
|
2023-08-13 12:06:35 +02:00
|
|
|
}
|
|
|
|
}
|