mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-26 18:35:20 +00:00
347 lines
13 KiB
C++
347 lines
13 KiB
C++
#include "waitdialog.hpp"
|
|
|
|
#include <MyGUI_InputManager.h>
|
|
#include <MyGUI_ProgressBar.h>
|
|
#include <MyGUI_ScrollBar.h>
|
|
|
|
#include <components/misc/rng.hpp>
|
|
|
|
#include <components/esm3/loadregn.hpp>
|
|
#include <components/misc/strings/format.hpp>
|
|
#include <components/settings/settings.hpp>
|
|
#include <components/widgets/box.hpp>
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
#include "../mwbase/mechanicsmanager.hpp"
|
|
#include "../mwbase/statemanager.hpp"
|
|
#include "../mwbase/windowmanager.hpp"
|
|
#include "../mwbase/world.hpp"
|
|
|
|
#include "../mwworld/cellstore.hpp"
|
|
#include "../mwworld/class.hpp"
|
|
#include "../mwworld/datetimemanager.hpp"
|
|
#include "../mwworld/esmstore.hpp"
|
|
|
|
#include "../mwmechanics/actorutil.hpp"
|
|
#include "../mwmechanics/creaturestats.hpp"
|
|
#include "../mwmechanics/npcstats.hpp"
|
|
|
|
namespace MWGui
|
|
{
|
|
|
|
WaitDialogProgressBar::WaitDialogProgressBar()
|
|
: WindowBase("openmw_wait_dialog_progressbar.layout")
|
|
{
|
|
getWidget(mProgressBar, "ProgressBar");
|
|
getWidget(mProgressText, "ProgressText");
|
|
}
|
|
|
|
void WaitDialogProgressBar::onOpen()
|
|
{
|
|
center();
|
|
}
|
|
|
|
void WaitDialogProgressBar::setProgress(int cur, int total)
|
|
{
|
|
mProgressBar->setProgressRange(total);
|
|
mProgressBar->setProgressPosition(cur);
|
|
mProgressText->setCaption(MyGUI::utility::toString(cur) + "/" + MyGUI::utility::toString(total));
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------------------------------------
|
|
|
|
WaitDialog::WaitDialog()
|
|
: WindowBase("openmw_wait_dialog.layout")
|
|
, mTimeAdvancer(0.05f)
|
|
, mSleeping(false)
|
|
, mHours(1)
|
|
, mManualHours(1)
|
|
, mFadeTimeRemaining(0)
|
|
, mInterruptAt(-1)
|
|
, mProgressBar()
|
|
{
|
|
getWidget(mDateTimeText, "DateTimeText");
|
|
getWidget(mRestText, "RestText");
|
|
getWidget(mHourText, "HourText");
|
|
getWidget(mUntilHealedButton, "UntilHealedButton");
|
|
getWidget(mWaitButton, "WaitButton");
|
|
getWidget(mCancelButton, "CancelButton");
|
|
getWidget(mHourSlider, "HourSlider");
|
|
|
|
mCancelButton->eventMouseButtonClick += MyGUI::newDelegate(this, &WaitDialog::onCancelButtonClicked);
|
|
mUntilHealedButton->eventMouseButtonClick += MyGUI::newDelegate(this, &WaitDialog::onUntilHealedButtonClicked);
|
|
mWaitButton->eventMouseButtonClick += MyGUI::newDelegate(this, &WaitDialog::onWaitButtonClicked);
|
|
mHourSlider->eventScrollChangePosition += MyGUI::newDelegate(this, &WaitDialog::onHourSliderChangedPosition);
|
|
|
|
mCancelButton->eventKeyButtonPressed += MyGUI::newDelegate(this, &WaitDialog::onKeyButtonPressed);
|
|
mWaitButton->eventKeyButtonPressed += MyGUI::newDelegate(this, &WaitDialog::onKeyButtonPressed);
|
|
mUntilHealedButton->eventKeyButtonPressed += MyGUI::newDelegate(this, &WaitDialog::onKeyButtonPressed);
|
|
|
|
mTimeAdvancer.eventProgressChanged += MyGUI::newDelegate(this, &WaitDialog::onWaitingProgressChanged);
|
|
mTimeAdvancer.eventInterrupted += MyGUI::newDelegate(this, &WaitDialog::onWaitingInterrupted);
|
|
mTimeAdvancer.eventFinished += MyGUI::newDelegate(this, &WaitDialog::onWaitingFinished);
|
|
}
|
|
|
|
void WaitDialog::setPtr(const MWWorld::Ptr& ptr)
|
|
{
|
|
setCanRest(!ptr.isEmpty() || MWBase::Environment::get().getWorld()->canRest() == MWBase::World::Rest_Allowed);
|
|
|
|
if (ptr.isEmpty() && MWBase::Environment::get().getWorld()->canRest() == MWBase::World::Rest_PlayerIsInAir)
|
|
{
|
|
// Resting in air is not allowed unless you're using a bed
|
|
MWBase::Environment::get().getWindowManager()->messageBox("#{sNotifyMessage1}");
|
|
MWBase::Environment::get().getWindowManager()->removeGuiMode(GM_Rest);
|
|
}
|
|
|
|
if (mUntilHealedButton->getVisible())
|
|
MWBase::Environment::get().getWindowManager()->setKeyFocusWidget(mUntilHealedButton);
|
|
else
|
|
MWBase::Environment::get().getWindowManager()->setKeyFocusWidget(mWaitButton);
|
|
}
|
|
|
|
bool WaitDialog::exit()
|
|
{
|
|
bool canExit = !mTimeAdvancer.isRunning(); // Only exit if not currently waiting
|
|
if (canExit)
|
|
{
|
|
clear();
|
|
stopWaiting();
|
|
}
|
|
return canExit;
|
|
}
|
|
|
|
void WaitDialog::clear()
|
|
{
|
|
mSleeping = false;
|
|
mHours = 1;
|
|
mManualHours = 1;
|
|
mFadeTimeRemaining = 0;
|
|
mInterruptAt = -1;
|
|
mTimeAdvancer.stop();
|
|
}
|
|
|
|
void WaitDialog::onOpen()
|
|
{
|
|
if (mTimeAdvancer.isRunning())
|
|
{
|
|
mProgressBar.setVisible(true);
|
|
setVisible(false);
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
mProgressBar.setVisible(false);
|
|
}
|
|
|
|
if (!MWBase::Environment::get().getWindowManager()->getRestEnabled())
|
|
{
|
|
MWBase::Environment::get().getWindowManager()->popGuiMode();
|
|
}
|
|
|
|
MWBase::World::RestPermitted canRest = MWBase::Environment::get().getWorld()->canRest();
|
|
|
|
if (canRest == MWBase::World::Rest_EnemiesAreNearby)
|
|
{
|
|
MWBase::Environment::get().getWindowManager()->messageBox("#{sNotifyMessage2}");
|
|
MWBase::Environment::get().getWindowManager()->popGuiMode();
|
|
}
|
|
else if (canRest == MWBase::World::Rest_PlayerIsUnderwater)
|
|
{
|
|
// resting underwater not allowed
|
|
MWBase::Environment::get().getWindowManager()->messageBox("#{sNotifyMessage1}");
|
|
MWBase::Environment::get().getWindowManager()->popGuiMode();
|
|
}
|
|
|
|
onHourSliderChangedPosition(mHourSlider, 0);
|
|
mHourSlider->setScrollPosition(0);
|
|
|
|
const MWWorld::DateTimeManager& timeManager = *MWBase::Environment::get().getWorld()->getTimeManager();
|
|
std::string_view month = timeManager.getMonthName();
|
|
int hour = static_cast<int>(timeManager.getTimeStamp().getHour());
|
|
bool pm = hour >= 12;
|
|
if (hour >= 13)
|
|
hour -= 12;
|
|
if (hour == 0)
|
|
hour = 12;
|
|
|
|
ESM::EpochTimeStamp currentDate = timeManager.getEpochTimeStamp();
|
|
std::string daysPassed = Misc::StringUtils::format("(#{Calendar:day} %i)", timeManager.getTimeStamp().getDay());
|
|
std::string_view formattedHour(pm ? "#{Calendar:pm}" : "#{Calendar:am}");
|
|
std::string dateTimeText
|
|
= Misc::StringUtils::format("%i %s %s %i %s", currentDate.mDay, month, daysPassed, hour, formattedHour);
|
|
mDateTimeText->setCaptionWithReplacing(dateTimeText);
|
|
}
|
|
|
|
void WaitDialog::onUntilHealedButtonClicked(MyGUI::Widget* sender)
|
|
{
|
|
int autoHours = MWBase::Environment::get().getMechanicsManager()->getHoursToRest();
|
|
|
|
startWaiting(autoHours);
|
|
}
|
|
|
|
void WaitDialog::onWaitButtonClicked(MyGUI::Widget* sender)
|
|
{
|
|
startWaiting(mManualHours);
|
|
}
|
|
|
|
void WaitDialog::startWaiting(int hoursToWait)
|
|
{
|
|
if (Settings::Manager::getBool("autosave", "Saves")) // autosaves when enabled
|
|
MWBase::Environment::get().getStateManager()->quickSave("Autosave");
|
|
|
|
MWBase::World* world = MWBase::Environment::get().getWorld();
|
|
MWBase::Environment::get().getWindowManager()->fadeScreenOut(0.2f);
|
|
mFadeTimeRemaining = 0.4f;
|
|
setVisible(false);
|
|
|
|
mHours = hoursToWait;
|
|
|
|
// FIXME: move this somewhere else?
|
|
mInterruptAt = -1;
|
|
MWWorld::Ptr player = world->getPlayerPtr();
|
|
if (mSleeping && player.getCell()->isExterior())
|
|
{
|
|
const ESM::RefId& regionstr = player.getCell()->getCell()->getRegion();
|
|
if (!regionstr.empty())
|
|
{
|
|
const ESM::Region* region = world->getStore().get<ESM::Region>().find(regionstr);
|
|
if (!region->mSleepList.empty())
|
|
{
|
|
// figure out if player will be woken while sleeping
|
|
int x = Misc::Rng::rollDice(hoursToWait, world->getPrng());
|
|
float fSleepRandMod
|
|
= world->getStore().get<ESM::GameSetting>().find("fSleepRandMod")->mValue.getFloat();
|
|
if (x < fSleepRandMod * hoursToWait)
|
|
{
|
|
float fSleepRestMod
|
|
= world->getStore().get<ESM::GameSetting>().find("fSleepRestMod")->mValue.getFloat();
|
|
int interruptAtHoursRemaining = int(fSleepRestMod * hoursToWait);
|
|
if (interruptAtHoursRemaining != 0)
|
|
{
|
|
mInterruptAt = hoursToWait - interruptAtHoursRemaining;
|
|
mInterruptCreatureList = region->mSleepList;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
mProgressBar.setProgress(0, hoursToWait);
|
|
}
|
|
|
|
void WaitDialog::onCancelButtonClicked(MyGUI::Widget* sender)
|
|
{
|
|
MWBase::Environment::get().getWindowManager()->removeGuiMode(GM_Rest);
|
|
}
|
|
|
|
void WaitDialog::onHourSliderChangedPosition(MyGUI::ScrollBar* sender, size_t position)
|
|
{
|
|
mHourText->setCaptionWithReplacing(MyGUI::utility::toString(position + 1) + " #{sRestMenu2}");
|
|
mManualHours = position + 1;
|
|
MWBase::Environment::get().getWindowManager()->setKeyFocusWidget(mWaitButton);
|
|
}
|
|
|
|
void WaitDialog::onKeyButtonPressed(MyGUI::Widget* sender, MyGUI::KeyCode key, MyGUI::Char character)
|
|
{
|
|
if (key == MyGUI::KeyCode::ArrowUp)
|
|
mHourSlider->setScrollPosition(
|
|
std::min(mHourSlider->getScrollPosition() + 1, mHourSlider->getScrollRange() - 1));
|
|
else if (key == MyGUI::KeyCode::ArrowDown)
|
|
mHourSlider->setScrollPosition(std::max(static_cast<int>(mHourSlider->getScrollPosition()) - 1, 0));
|
|
else
|
|
return;
|
|
onHourSliderChangedPosition(mHourSlider, mHourSlider->getScrollPosition());
|
|
}
|
|
|
|
void WaitDialog::onWaitingProgressChanged(int cur, int total)
|
|
{
|
|
mProgressBar.setProgress(cur, total);
|
|
MWBase::Environment::get().getMechanicsManager()->rest(1, mSleeping);
|
|
MWBase::Environment::get().getWorld()->advanceTime(1);
|
|
|
|
MWWorld::Ptr player = MWBase::Environment::get().getWorld()->getPlayerPtr();
|
|
if (player.getClass().getCreatureStats(player).isDead())
|
|
stopWaiting();
|
|
}
|
|
|
|
void WaitDialog::onWaitingInterrupted()
|
|
{
|
|
MWBase::Environment::get().getWindowManager()->messageBox("#{sSleepInterrupt}");
|
|
MWBase::Environment::get().getWorld()->spawnRandomCreature(mInterruptCreatureList);
|
|
stopWaiting();
|
|
}
|
|
|
|
void WaitDialog::onWaitingFinished()
|
|
{
|
|
stopWaiting();
|
|
|
|
MWWorld::Ptr player = MWMechanics::getPlayer();
|
|
const MWMechanics::NpcStats& pcstats = player.getClass().getNpcStats(player);
|
|
|
|
// trigger levelup if possible
|
|
const MWWorld::Store<ESM::GameSetting>& gmst
|
|
= MWBase::Environment::get().getESMStore()->get<ESM::GameSetting>();
|
|
if (mSleeping && pcstats.getLevelProgress() >= gmst.find("iLevelUpTotal")->mValue.getInteger())
|
|
{
|
|
MWBase::Environment::get().getWindowManager()->pushGuiMode(GM_Levelup);
|
|
}
|
|
}
|
|
|
|
void WaitDialog::setCanRest(bool canRest)
|
|
{
|
|
MWWorld::Ptr player = MWMechanics::getPlayer();
|
|
MWMechanics::CreatureStats& stats = player.getClass().getCreatureStats(player);
|
|
bool full = (stats.getHealth().getCurrent() >= stats.getHealth().getModified())
|
|
&& (stats.getMagicka().getCurrent() >= stats.getMagicka().getModified());
|
|
MWMechanics::NpcStats& npcstats = player.getClass().getNpcStats(player);
|
|
bool werewolf = npcstats.isWerewolf();
|
|
|
|
mUntilHealedButton->setVisible(canRest && !full);
|
|
mWaitButton->setCaptionWithReplacing(canRest ? "#{sRest}" : "#{sWait}");
|
|
mRestText->setCaptionWithReplacing(
|
|
canRest ? "#{sRestMenu3}" : (werewolf ? "#{sWerewolfRestMessage}" : "#{sRestIllegal}"));
|
|
|
|
mSleeping = canRest;
|
|
|
|
Gui::Box* box = dynamic_cast<Gui::Box*>(mMainWidget);
|
|
if (box == nullptr)
|
|
throw std::runtime_error("main widget must be a box");
|
|
box->notifyChildrenSizeChanged();
|
|
center();
|
|
}
|
|
|
|
void WaitDialog::onFrame(float dt)
|
|
{
|
|
mTimeAdvancer.onFrame(dt);
|
|
|
|
if (mFadeTimeRemaining <= 0)
|
|
return;
|
|
|
|
mFadeTimeRemaining -= dt;
|
|
|
|
if (mFadeTimeRemaining <= 0)
|
|
{
|
|
mProgressBar.setVisible(true);
|
|
mTimeAdvancer.run(mHours, mInterruptAt);
|
|
}
|
|
}
|
|
|
|
void WaitDialog::stopWaiting()
|
|
{
|
|
MWBase::Environment::get().getWindowManager()->fadeScreenIn(0.2f);
|
|
mProgressBar.setVisible(false);
|
|
MWBase::Environment::get().getWindowManager()->removeGuiMode(GM_Rest);
|
|
mTimeAdvancer.stop();
|
|
}
|
|
|
|
void WaitDialog::wakeUp()
|
|
{
|
|
mSleeping = false;
|
|
if (mInterruptAt != -1)
|
|
onWaitingInterrupted();
|
|
else
|
|
stopWaiting();
|
|
}
|
|
|
|
}
|