mirror of
https://github.com/TES3MP/openmw-tes3mp.git
synced 2025-01-15 18:49:54 +00:00
71 lines
2 KiB
C++
71 lines
2 KiB
C++
#include "regionsoundselector.hpp"
|
|
|
|
#include <components/misc/rng.hpp>
|
|
|
|
#include <algorithm>
|
|
#include <numeric>
|
|
|
|
#include "../mwbase/world.hpp"
|
|
#include "../mwworld/esmstore.hpp"
|
|
|
|
namespace MWSound
|
|
{
|
|
namespace
|
|
{
|
|
int addChance(int result, const ESM::Region::SoundRef &v)
|
|
{
|
|
return result + v.mChance;
|
|
}
|
|
}
|
|
|
|
boost::optional<std::string> RegionSoundSelector::getNextRandom(float duration, const std::string& regionName,
|
|
const MWBase::World& world)
|
|
{
|
|
mTimePassed += duration;
|
|
|
|
if (mTimePassed < mTimeToNextEnvSound)
|
|
return {};
|
|
|
|
const float a = Misc::Rng::rollClosedProbability();
|
|
// NOTE: We should use the "Minimum Time Between Environmental Sounds" and
|
|
// "Maximum Time Between Environmental Sounds" fallback settings here.
|
|
mTimeToNextEnvSound = 5.0f * a + 15.0f * (1.0f - a);
|
|
mTimePassed = 0;
|
|
|
|
if (mLastRegionName != regionName)
|
|
{
|
|
mLastRegionName = regionName;
|
|
mSumChance = 0;
|
|
}
|
|
|
|
const ESM::Region* const region = world.getStore().get<ESM::Region>().search(mLastRegionName);
|
|
|
|
if (region == nullptr)
|
|
return {};
|
|
|
|
if (mSumChance == 0)
|
|
{
|
|
mSumChance = std::accumulate(region->mSoundList.begin(), region->mSoundList.end(), 0, addChance);
|
|
if (mSumChance == 0)
|
|
return {};
|
|
}
|
|
|
|
const int r = Misc::Rng::rollDice(mSumChance);
|
|
int pos = 0;
|
|
|
|
const auto isSelected = [&] (const ESM::Region::SoundRef& sound)
|
|
{
|
|
if (r - pos < sound.mChance)
|
|
return true;
|
|
pos += sound.mChance;
|
|
return false;
|
|
};
|
|
|
|
const auto it = std::find_if(region->mSoundList.begin(), region->mSoundList.end(), isSelected);
|
|
|
|
if (it == region->mSoundList.end())
|
|
return {};
|
|
|
|
return it->mSound;
|
|
}
|
|
}
|